mirror of
https://github.com/shlinkio/shlink-web-client.git
synced 2025-01-09 01:37:24 +03:00
Added swipable menu
This commit is contained in:
parent
42d718960f
commit
cb9dc9d65e
11 changed files with 139 additions and 48 deletions
|
@ -29,6 +29,7 @@
|
||||||
"react-moment": "^0.7.6",
|
"react-moment": "^0.7.6",
|
||||||
"react-redux": "^5.0.7",
|
"react-redux": "^5.0.7",
|
||||||
"react-router-dom": "^4.2.2",
|
"react-router-dom": "^4.2.2",
|
||||||
|
"react-swipeable": "^4.3.0",
|
||||||
"react-tagsinput": "^3.19.0",
|
"react-tagsinput": "^3.19.0",
|
||||||
"reactstrap": "^6.0.1",
|
"reactstrap": "^6.0.1",
|
||||||
"redux": "^4.0.0",
|
"redux": "^4.0.0",
|
||||||
|
|
|
@ -9,7 +9,7 @@ import CreateServer from './servers/CreateServer';
|
||||||
export default class App extends React.Component {
|
export default class App extends React.Component {
|
||||||
render() {
|
render() {
|
||||||
return (
|
return (
|
||||||
<div className="container-fluid">
|
<div className="container-fluid app-container">
|
||||||
<MainHeader/>
|
<MainHeader/>
|
||||||
|
|
||||||
<div className="app">
|
<div className="app">
|
||||||
|
|
|
@ -1,5 +1,10 @@
|
||||||
@import './utils/base';
|
@import './utils/base';
|
||||||
|
|
||||||
|
.app-container {
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
.app {
|
.app {
|
||||||
padding-top: $headerHeight;
|
padding-top: $headerHeight;
|
||||||
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,12 +6,23 @@ import { NavLink } from 'react-router-dom';
|
||||||
import DeleteServerButton from '../servers/DeleteServerButton';
|
import DeleteServerButton from '../servers/DeleteServerButton';
|
||||||
import './AsideMenu.scss';
|
import './AsideMenu.scss';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
|
import { serverType } from '../servers/prop-types';
|
||||||
|
|
||||||
export default function AsideMenu({ selectedServer }) {
|
const defaultProps = {
|
||||||
|
className: '',
|
||||||
|
showOnMobile: false,
|
||||||
|
};
|
||||||
|
const propTypes = {
|
||||||
|
selectedServer: serverType,
|
||||||
|
className: PropTypes.string,
|
||||||
|
showOnMobile: PropTypes.bool,
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function AsideMenu({ selectedServer, className, showOnMobile }) {
|
||||||
const serverId = selectedServer ? selectedServer.id : '';
|
const serverId = selectedServer ? selectedServer.id : '';
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<aside className="aside-menu col-lg-2 col-md-3">
|
<aside className={`aside-menu ${!showOnMobile ? 'aside-menu--hidden' : ''} ${className}`}>
|
||||||
<nav className="nav flex-column aside-menu__nav">
|
<nav className="nav flex-column aside-menu__nav">
|
||||||
<NavLink
|
<NavLink
|
||||||
className="aside-menu__item"
|
className="aside-menu__item"
|
||||||
|
@ -39,11 +50,5 @@ export default function AsideMenu({ selectedServer }) {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
AsideMenu.propTypes = {
|
AsideMenu.defaultProps = defaultProps;
|
||||||
selectedServer: PropTypes.shape({
|
AsideMenu.propTypes = propTypes;
|
||||||
id: PropTypes.string,
|
|
||||||
name: PropTypes.string,
|
|
||||||
url: PropTypes.string,
|
|
||||||
apiKey: PropTypes.string,
|
|
||||||
}),
|
|
||||||
};
|
|
||||||
|
|
|
@ -1,22 +1,38 @@
|
||||||
@import '../utils/base';
|
@import '../utils/base';
|
||||||
|
@import '../utils/mixins/box-shadow';
|
||||||
|
|
||||||
|
$asideMenuMobileWidth: 280px;
|
||||||
|
|
||||||
.aside-menu {
|
.aside-menu {
|
||||||
background-color: #f7f7f7;
|
background-color: #f7f7f7;
|
||||||
padding-top: 10px;
|
position: fixed !important;
|
||||||
|
padding-top: 13px;
|
||||||
padding-bottom: 10px;
|
padding-bottom: 10px;
|
||||||
|
top: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
display: block;
|
||||||
|
z-index: 1050;
|
||||||
|
overflow-x: hidden;
|
||||||
|
overflow-y: auto;
|
||||||
|
|
||||||
@media (min-width: $mdMin) {
|
@media (min-width: $mdMin) {
|
||||||
position: fixed !important;
|
|
||||||
top: $headerHeight;
|
top: $headerHeight;
|
||||||
bottom: 0;
|
|
||||||
left: 0;
|
|
||||||
z-index: 1000;
|
|
||||||
display: block;
|
|
||||||
padding: 30px 15px 15px;
|
padding: 30px 15px 15px;
|
||||||
overflow-x: hidden;
|
|
||||||
overflow-y: auto;
|
|
||||||
border-right: 1px solid #eee;
|
border-right: 1px solid #eee;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@media (max-width: $smMax) {
|
||||||
|
width: $asideMenuMobileWidth !important;
|
||||||
|
@include box-shadow(-10px 0px 50px 11px rgba(0, 0, 0, 0.55));
|
||||||
|
transition: left 300ms;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.aside-menu--hidden {
|
||||||
|
@media (max-width: $smMax) {
|
||||||
|
left: -($asideMenuMobileWidth + 35px);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.aside-menu__nav {
|
.aside-menu__nav {
|
||||||
|
|
|
@ -1,48 +1,79 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { Route, Switch } from 'react-router-dom';
|
import { Route, Switch, withRouter } from 'react-router-dom';
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
|
import { compose } from 'redux';
|
||||||
import { selectServer } from '../servers/reducers/selectedServer';
|
import { selectServer } from '../servers/reducers/selectedServer';
|
||||||
import CreateShortUrl from '../short-urls/CreateShortUrl';
|
import CreateShortUrl from '../short-urls/CreateShortUrl';
|
||||||
import ShortUrls from '../short-urls/ShortUrls';
|
import ShortUrls from '../short-urls/ShortUrls';
|
||||||
import ShortUrlsVisits from '../short-urls/ShortUrlVisits';
|
import ShortUrlsVisits from '../short-urls/ShortUrlVisits';
|
||||||
import AsideMenu from './AsideMenu';
|
import AsideMenu from './AsideMenu';
|
||||||
import { pick } from 'ramda';
|
import { pick } from 'ramda';
|
||||||
|
import Swipeable from 'react-swipeable';
|
||||||
|
import './MenuLayout.scss';
|
||||||
|
|
||||||
export class MenuLayout extends React.Component {
|
export class MenuLayout extends React.Component {
|
||||||
|
state = { showSideBar: false };
|
||||||
|
|
||||||
// FIXME Shouldn't use componentWillMount, but this code has to be run before children components are rendered
|
// FIXME Shouldn't use componentWillMount, but this code has to be run before children components are rendered
|
||||||
componentWillMount() {
|
componentWillMount() {
|
||||||
const { serverId } = this.props.match.params;
|
const { serverId } = this.props.match.params;
|
||||||
this.props.selectServer(serverId);
|
this.props.selectServer(serverId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
componentDidUpdate(prevProps) {
|
||||||
|
const { location } = this.props;
|
||||||
|
|
||||||
|
// Hide sidebar when location changes
|
||||||
|
if (location !== prevProps.location) {
|
||||||
|
this.setState({ showSideBar: false });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { selectedServer } = this.props;
|
const { selectedServer } = this.props;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="row">
|
<Swipeable
|
||||||
<AsideMenu selectedServer={selectedServer} />
|
delta={40}
|
||||||
<div className="col-lg-10 offset-lg-2 col-md-9 offset-md-3">
|
onSwipedLeft={() => this.setState({ showSideBar: false })}
|
||||||
<Switch>
|
onSwipedRight={() => this.setState({ showSideBar: true })}
|
||||||
<Route
|
className="menu-layout__swipeable"
|
||||||
exact
|
>
|
||||||
path="/server/:serverId/list-short-urls/:page"
|
<div className="row menu-layout__swipeable-inner">
|
||||||
component={ShortUrls}
|
<AsideMenu
|
||||||
/>
|
className="col-lg-2 col-md-3"
|
||||||
<Route
|
selectedServer={selectedServer}
|
||||||
exact
|
showOnMobile={this.state.showSideBar}
|
||||||
path="/server/:serverId/create-short-url"
|
/>
|
||||||
component={CreateShortUrl}
|
<div
|
||||||
/>
|
className="col-lg-10 offset-lg-2 col-md-9 offset-md-3"
|
||||||
<Route
|
onClick={() => this.setState({ showSideBar: false })}
|
||||||
exact
|
>
|
||||||
path="/server/:serverId/short-code/:shortCode/visits"
|
<Switch>
|
||||||
component={ShortUrlsVisits}
|
<Route
|
||||||
/>
|
exact
|
||||||
</Switch>
|
path="/server/:serverId/list-short-urls/:page"
|
||||||
|
component={ShortUrls}
|
||||||
|
/>
|
||||||
|
<Route
|
||||||
|
exact
|
||||||
|
path="/server/:serverId/create-short-url"
|
||||||
|
component={CreateShortUrl}
|
||||||
|
/>
|
||||||
|
<Route
|
||||||
|
exact
|
||||||
|
path="/server/:serverId/short-code/:shortCode/visits"
|
||||||
|
component={ShortUrlsVisits}
|
||||||
|
/>
|
||||||
|
</Switch>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</Swipeable>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default connect(pick(['selectedServer', 'shortUrlsListParams']), { selectServer })(MenuLayout);
|
export default compose(
|
||||||
|
connect(pick(['selectedServer', 'shortUrlsListParams']), { selectServer }),
|
||||||
|
withRouter
|
||||||
|
)(MenuLayout);
|
||||||
|
|
12
src/common/MenuLayout.scss
Normal file
12
src/common/MenuLayout.scss
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
.menu-layout__swipeable {
|
||||||
|
$offset: 15px;
|
||||||
|
height: 100%;
|
||||||
|
margin-right: -$offset;
|
||||||
|
margin-left: -$offset;
|
||||||
|
padding-left: $offset;
|
||||||
|
padding-right: $offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menu-layout__swipeable-inner {
|
||||||
|
height: 100%;
|
||||||
|
}
|
|
@ -1,5 +1,11 @@
|
||||||
@import './utils/base';
|
@import './utils/base';
|
||||||
|
|
||||||
|
html,
|
||||||
|
body,
|
||||||
|
#root {
|
||||||
|
height: 100%
|
||||||
|
}
|
||||||
|
|
||||||
* {
|
* {
|
||||||
outline: none !important;
|
outline: none !important;
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@ import { withRouter } from 'react-router-dom';
|
||||||
import { Modal, ModalBody, ModalFooter, ModalHeader } from 'reactstrap';
|
import { Modal, ModalBody, ModalFooter, ModalHeader } from 'reactstrap';
|
||||||
import { compose } from 'redux';
|
import { compose } from 'redux';
|
||||||
import { deleteServer } from './reducers/server';
|
import { deleteServer } from './reducers/server';
|
||||||
|
import { serverType } from './prop-types';
|
||||||
|
|
||||||
export const DeleteServerModal = ({ server, toggle, isOpen, deleteServer, history }) => {
|
export const DeleteServerModal = ({ server, toggle, isOpen, deleteServer, history }) => {
|
||||||
const closeModal = () => {
|
const closeModal = () => {
|
||||||
|
@ -34,12 +35,7 @@ export const DeleteServerModal = ({ server, toggle, isOpen, deleteServer, histor
|
||||||
DeleteServerModal.propTypes = {
|
DeleteServerModal.propTypes = {
|
||||||
toggle: PropTypes.func.isRequired,
|
toggle: PropTypes.func.isRequired,
|
||||||
isOpen: PropTypes.bool.isRequired,
|
isOpen: PropTypes.bool.isRequired,
|
||||||
server: PropTypes.shape({
|
server: serverType,
|
||||||
id: PropTypes.string,
|
|
||||||
name: PropTypes.string,
|
|
||||||
url: PropTypes.string,
|
|
||||||
apiKey: PropTypes.string,
|
|
||||||
}),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export default compose(
|
export default compose(
|
||||||
|
|
8
src/servers/prop-types/index.js
Normal file
8
src/servers/prop-types/index.js
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
import PropTypes from 'prop-types';
|
||||||
|
|
||||||
|
export const serverType = PropTypes.shape({
|
||||||
|
id: PropTypes.string,
|
||||||
|
name: PropTypes.string,
|
||||||
|
url: PropTypes.string,
|
||||||
|
apiKey: PropTypes.string,
|
||||||
|
});
|
11
yarn.lock
11
yarn.lock
|
@ -2160,6 +2160,10 @@ detect-node@^2.0.3:
|
||||||
version "2.0.3"
|
version "2.0.3"
|
||||||
resolved "https://registry.yarnpkg.com/detect-node/-/detect-node-2.0.3.tgz#a2033c09cc8e158d37748fbde7507832bd6ce127"
|
resolved "https://registry.yarnpkg.com/detect-node/-/detect-node-2.0.3.tgz#a2033c09cc8e158d37748fbde7507832bd6ce127"
|
||||||
|
|
||||||
|
detect-passive-events@^1.0.4:
|
||||||
|
version "1.0.4"
|
||||||
|
resolved "https://registry.yarnpkg.com/detect-passive-events/-/detect-passive-events-1.0.4.tgz#6ed477e6e5bceb79079735dcd357789d37f9a91a"
|
||||||
|
|
||||||
detect-port-alt@1.1.6:
|
detect-port-alt@1.1.6:
|
||||||
version "1.1.6"
|
version "1.1.6"
|
||||||
resolved "https://registry.yarnpkg.com/detect-port-alt/-/detect-port-alt-1.1.6.tgz#24707deabe932d4a3cf621302027c2b266568275"
|
resolved "https://registry.yarnpkg.com/detect-port-alt/-/detect-port-alt-1.1.6.tgz#24707deabe932d4a3cf621302027c2b266568275"
|
||||||
|
@ -6228,6 +6232,13 @@ react-router@^4.3.1:
|
||||||
prop-types "^15.6.1"
|
prop-types "^15.6.1"
|
||||||
warning "^4.0.1"
|
warning "^4.0.1"
|
||||||
|
|
||||||
|
react-swipeable@^4.3.0:
|
||||||
|
version "4.3.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/react-swipeable/-/react-swipeable-4.3.0.tgz#818c0818ac565f5b169348f442153f343abff8bc"
|
||||||
|
dependencies:
|
||||||
|
detect-passive-events "^1.0.4"
|
||||||
|
prop-types "^15.5.8"
|
||||||
|
|
||||||
react-tagsinput@^3.19.0:
|
react-tagsinput@^3.19.0:
|
||||||
version "3.19.0"
|
version "3.19.0"
|
||||||
resolved "https://registry.yarnpkg.com/react-tagsinput/-/react-tagsinput-3.19.0.tgz#6e3b45595f2d295d4657bf194491988f948caabf"
|
resolved "https://registry.yarnpkg.com/react-tagsinput/-/react-tagsinput-3.19.0.tgz#6e3b45595f2d295d4657bf194491988f948caabf"
|
||||||
|
|
Loading…
Reference in a new issue