Added overflow stop for mobile menu, shifting some stuff.
This commit is contained in:
@ -22,6 +22,8 @@
|
|||||||
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
import { connect } from 'react-redux';
|
||||||
|
|
||||||
import Background from './background/Background';
|
import Background from './background/Background';
|
||||||
import Header from './header/Header';
|
import Header from './header/Header';
|
||||||
import Footer from './footer/Footer';
|
import Footer from './footer/Footer';
|
||||||
@ -31,19 +33,37 @@ import Routes from './page/Routes';
|
|||||||
class App extends React.Component {
|
class App extends React.Component {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
super(props);
|
super(props);
|
||||||
|
|
||||||
|
this.onEnteringBound = this.onEntering.bind(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
onEntering() {
|
||||||
|
this.refs.app.scroll({
|
||||||
|
top: 0,
|
||||||
|
left: 0,
|
||||||
|
behavior: 'smooth'
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
|
let clazz = "o-app";
|
||||||
|
if(this.props.menuOpen) clazz += " is-menu-open "
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<HashRouter>
|
<HashRouter>
|
||||||
<div className="o-app">
|
<div className={clazz} ref="app">
|
||||||
<Background style="normal" />
|
|
||||||
<Header />
|
<Header />
|
||||||
<Routes />
|
<Routes onEntering={this.onEnteringBound} />
|
||||||
</div>
|
</div>
|
||||||
</HashRouter>
|
</HashRouter>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default App;
|
const mapStateToProps = function(state) {
|
||||||
|
return {
|
||||||
|
menuOpen: state.menu.open
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default connect(mapStateToProps)(App);
|
||||||
|
44
public/actions/MenuActions.js
Normal file
44
public/actions/MenuActions.js
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
// Copyright (c) 2018 Dominic Masters
|
||||||
|
//
|
||||||
|
// MIT License
|
||||||
|
//
|
||||||
|
// Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
// a copy of this software and associated documentation files (the
|
||||||
|
// "Software"), to deal in the Software without restriction, including
|
||||||
|
// without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
// permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
// the following conditions:
|
||||||
|
//
|
||||||
|
// The above copyright notice and this permission notice shall be
|
||||||
|
// included in all copies or substantial portions of the Software.
|
||||||
|
//
|
||||||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
|
export const OPEN_MENU = "OPEN_MENU";
|
||||||
|
export const CLOSE_MENU = "CLOSE_MENU";
|
||||||
|
export const TOGGLE_MENU = "TOGGLE_MENU";
|
||||||
|
|
||||||
|
export function openMenu() {
|
||||||
|
return {
|
||||||
|
type: OPEN_MENU
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function closeMenu() {
|
||||||
|
return {
|
||||||
|
type: CLOSE_MENU
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function toggleMenu() {
|
||||||
|
return {
|
||||||
|
type: TOGGLE_MENU
|
||||||
|
}
|
||||||
|
}
|
@ -22,12 +22,16 @@
|
|||||||
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
import { connect } from 'react-redux';
|
||||||
import { NavLink } from 'react-router-dom';
|
import { NavLink } from 'react-router-dom';
|
||||||
|
import * as MenuActions from './../../actions/MenuActions';
|
||||||
|
|
||||||
const HamburerMenuItem = function(props) {
|
const HamburerMenuItem = function(props) {
|
||||||
return (
|
return (
|
||||||
<li className="o-hamburger-menu__link">
|
<li className="o-hamburger-menu__link">
|
||||||
<NavLink to={ props.to }>Home</NavLink>
|
<NavLink to={ props.to } className="o-hamburger-menu__link-link">
|
||||||
|
Home
|
||||||
|
</NavLink>
|
||||||
</li>
|
</li>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -35,23 +39,11 @@ const HamburerMenuItem = function(props) {
|
|||||||
class HamburgerMenu extends React.Component {
|
class HamburgerMenu extends React.Component {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
super(props);
|
super(props);
|
||||||
|
|
||||||
this.state ={
|
|
||||||
open: false
|
|
||||||
}
|
|
||||||
|
|
||||||
this.toggleMenuBound = this.toggleMenu.bind(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
toggleMenu() {
|
|
||||||
this.setState({
|
|
||||||
open: !this.state.open
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
let clazz = "o-hamburger-menu";
|
let clazz = "o-hamburger-menu";
|
||||||
if(this.state && this.state.open) clazz += " is-open";
|
if(this.props.open) clazz += " is-open";
|
||||||
if(this.props.className) clazz += " " + this.props.className;
|
if(this.props.className) clazz += " " + this.props.className;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@ -59,7 +51,7 @@ class HamburgerMenu extends React.Component {
|
|||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
className="o-hamburger-menu__button"
|
className="o-hamburger-menu__button"
|
||||||
onClick={this.toggleMenuBound}
|
onClick={this.props.toggleMenu}
|
||||||
>
|
>
|
||||||
<img
|
<img
|
||||||
src={ require('./../../images/icons/hamburger.svg') }
|
src={ require('./../../images/icons/hamburger.svg') }
|
||||||
@ -67,14 +59,31 @@ class HamburgerMenu extends React.Component {
|
|||||||
/>
|
/>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<ul className="o-hamburger-menu__menu">
|
<div className="o-hamburger-menu__menu">
|
||||||
|
<ul className="o-hamburger-menu__links">
|
||||||
<HamburerMenuItem to="/" />
|
<HamburerMenuItem to="/" />
|
||||||
<HamburerMenuItem to="/" />
|
<HamburerMenuItem to="/" />
|
||||||
<HamburerMenuItem to="/" />
|
<HamburerMenuItem to="/" />
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default HamburgerMenu;
|
|
||||||
|
const mapStateToProps = function(state) {
|
||||||
|
return {
|
||||||
|
open: state.menu.open
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const mapDispatchToProps = function(dispatch) {
|
||||||
|
return {
|
||||||
|
toggleMenu: function(theme) {
|
||||||
|
dispatch(MenuActions.toggleMenu());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default connect(mapStateToProps, mapDispatchToProps)(HamburgerMenu);
|
||||||
|
@ -27,6 +27,7 @@ import { withRouter } from 'react-router';
|
|||||||
import PropTypes from 'prop-types'
|
import PropTypes from 'prop-types'
|
||||||
import { HashRouter, Route, Switch } from 'react-router-dom';
|
import { HashRouter, Route, Switch } from 'react-router-dom';
|
||||||
|
|
||||||
|
import Header from './../header/Header';
|
||||||
import Footer from './../footer/Footer';
|
import Footer from './../footer/Footer';
|
||||||
|
|
||||||
//Pages
|
//Pages
|
||||||
@ -49,8 +50,16 @@ const RouteWrapper = (props) => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const Routes = ({location}) => {
|
class Routes extends React.Component {
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { match, location, history } = this.props;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<Route>
|
||||||
<TransitionGroup className="o-page-transition__container">
|
<TransitionGroup className="o-page-transition__container">
|
||||||
<CSSTransition
|
<CSSTransition
|
||||||
key={ location.pathname }
|
key={ location.pathname }
|
||||||
@ -58,13 +67,7 @@ const Routes = ({location}) => {
|
|||||||
classNames="o-page-transition"
|
classNames="o-page-transition"
|
||||||
mountOnEnter={ true }
|
mountOnEnter={ true }
|
||||||
unmountOnExit={ true }
|
unmountOnExit={ true }
|
||||||
onEntering={() => {
|
onEntering={ this.props.onEntering }
|
||||||
window.scroll({
|
|
||||||
top: 0,
|
|
||||||
left: 0,
|
|
||||||
behavior: 'smooth'
|
|
||||||
});
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
<Switch location={ location }>
|
<Switch location={ location }>
|
||||||
<RouteWrapper exact path="/" page={ Homepage } />
|
<RouteWrapper exact path="/" page={ Homepage } />
|
||||||
@ -73,9 +76,9 @@ const Routes = ({location}) => {
|
|||||||
</Switch>
|
</Switch>
|
||||||
</CSSTransition>
|
</CSSTransition>
|
||||||
</TransitionGroup>
|
</TransitionGroup>
|
||||||
|
</Route>
|
||||||
);
|
);
|
||||||
};
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export default withRouter(() => {
|
export default withRouter(Routes);
|
||||||
return <Route render={Routes} />
|
|
||||||
});
|
|
||||||
|
@ -29,11 +29,15 @@ import Image from './../../image/Image';
|
|||||||
import { Title, Subtitle } from './../../typography/Typography';
|
import { Title, Subtitle } from './../../typography/Typography';
|
||||||
|
|
||||||
export default function() {
|
export default function() {
|
||||||
|
let lines = [];
|
||||||
|
for(let i = 0; i < 100; i++) {
|
||||||
|
lines.push(<br key={i} />);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Page style="home-page" className="p-home-page">
|
<Page style="home-page" className="p-home-page">
|
||||||
Welcome home
|
Welcome home
|
||||||
|
{ lines }
|
||||||
</Page>
|
</Page>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
51
public/reducers/MenuReducer.js
Normal file
51
public/reducers/MenuReducer.js
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
// Copyright (c) 2018 Dominic Masters
|
||||||
|
//
|
||||||
|
// MIT License
|
||||||
|
//
|
||||||
|
// Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
// a copy of this software and associated documentation files (the
|
||||||
|
// "Software"), to deal in the Software without restriction, including
|
||||||
|
// without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
// permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
// the following conditions:
|
||||||
|
//
|
||||||
|
// The above copyright notice and this permission notice shall be
|
||||||
|
// included in all copies or substantial portions of the Software.
|
||||||
|
//
|
||||||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
|
import {
|
||||||
|
OPEN_MENU,
|
||||||
|
CLOSE_MENU,
|
||||||
|
TOGGLE_MENU
|
||||||
|
} from './../actions/MenuActions';
|
||||||
|
|
||||||
|
const initialState = {
|
||||||
|
open: false
|
||||||
|
}
|
||||||
|
|
||||||
|
const menu = function(state, action) {
|
||||||
|
if(typeof state === typeof undefined) {
|
||||||
|
state = initialState;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch(action.type) {
|
||||||
|
case OPEN_MENU:
|
||||||
|
return {open: true}
|
||||||
|
case CLOSE_MENU:
|
||||||
|
return {open: false}
|
||||||
|
case TOGGLE_MENU:
|
||||||
|
return {open: !state.open}
|
||||||
|
default:
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default menu;
|
@ -24,9 +24,11 @@
|
|||||||
import { combineReducers } from 'redux';
|
import { combineReducers } from 'redux';
|
||||||
|
|
||||||
import LanguageReducer from './LanguageReducer';
|
import LanguageReducer from './LanguageReducer';
|
||||||
|
import MenuReducer from './MenuReducer';
|
||||||
|
|
||||||
const rootReducer = combineReducers({
|
const rootReducer = combineReducers({
|
||||||
language: LanguageReducer
|
language: LanguageReducer,
|
||||||
|
menu: MenuReducer
|
||||||
});
|
});
|
||||||
|
|
||||||
export default rootReducer;
|
export default rootReducer;
|
||||||
|
@ -11,9 +11,10 @@
|
|||||||
body {
|
body {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
min-height: 100vh;
|
width: 100vw;
|
||||||
|
height: 100vh;
|
||||||
|
|
||||||
font-family: $s-font--stack-default;
|
font-family: $s-font--stack-default;
|
||||||
font-size: $s-font--size--base;
|
font-size: $s-font--size--base;
|
||||||
overflow-y: scroll;//Really makes the transitions feel smoother
|
overflow: hidden;//Really makes the transitions feel smoother
|
||||||
}
|
}
|
||||||
|
@ -10,9 +10,8 @@
|
|||||||
*/
|
*/
|
||||||
.o-app {
|
.o-app {
|
||||||
position: relative;
|
position: relative;
|
||||||
min-height: 100vh;
|
height: 100vh;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
overflow-y: scroll;
|
||||||
@extend %t-flexbox;
|
overflow-x: hidden;
|
||||||
@include t-flex-direction(column);
|
|
||||||
}
|
}
|
||||||
|
@ -3,18 +3,29 @@
|
|||||||
* Mobile-Centric openable menu with a hamburger icon to toggle.
|
* Mobile-Centric openable menu with a hamburger icon to toggle.
|
||||||
*
|
*
|
||||||
* Dependencies:
|
* Dependencies:
|
||||||
|
* styles/settings/animation.scss
|
||||||
* styles/settings/colors.scss
|
* styles/settings/colors.scss
|
||||||
* styles/settings/z.scss
|
* styles/settings/z.scss
|
||||||
*
|
*
|
||||||
* Version:
|
* Version:
|
||||||
* 1.0.0 - 2018/06/12
|
* 1.0.0 - 2018/06/12
|
||||||
*/
|
*/
|
||||||
|
$o-hamburger-menu--max: 200%;
|
||||||
|
$o-hamburger-menu--pos-x: 100%;
|
||||||
|
$o-hamburger-menu--pos-y: 0%;
|
||||||
|
@include t-keyframes(o-hamburger-menu--open) {
|
||||||
|
from { clip-path: circle(0% at $o-hamburger-menu--pos-x $o-hamburger-menu--pos-y); }
|
||||||
|
to { clip-path: circle($o-hamburger-menu--max at $o-hamburger-menu--pos-x $o-hamburger-menu--pos-y); }
|
||||||
|
}
|
||||||
|
|
||||||
|
@include t-keyframes(o-hamburger-menu--close) {
|
||||||
|
from { clip-path: circle($o-hamburger-menu--max at $o-hamburger-menu--pos-x $o-hamburger-menu--pos-y); }
|
||||||
|
to { clip-path: circle(0% at $o-hamburger-menu--pos-x $o-hamburger-menu--pos-y); }
|
||||||
|
}
|
||||||
|
|
||||||
.o-hamburger-menu {
|
.o-hamburger-menu {
|
||||||
border: 1px solid red;
|
|
||||||
|
|
||||||
&__menu {
|
&__menu {
|
||||||
@extend %t-list-blank;
|
|
||||||
|
|
||||||
display: none;
|
display: none;
|
||||||
position: fixed;
|
position: fixed;
|
||||||
top: 0;
|
top: 0;
|
||||||
@ -22,12 +33,37 @@
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
background: $s-color--menu__background;
|
background: $s-color--menu__background;
|
||||||
|
transition: all 1s $s-animation--ease-out;
|
||||||
|
|
||||||
z-index: $s-z--menu;
|
z-index: $s-z--menu;
|
||||||
|
|
||||||
|
@include t-animation-fill-mode(forwards);
|
||||||
|
@include t-animation-timing-function($s-animation--ease-out);
|
||||||
|
@include t-animation-duration(0.4s);
|
||||||
|
}
|
||||||
|
|
||||||
|
&__links {
|
||||||
|
@extend %t-list-blank;
|
||||||
|
padding: 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
&__link {
|
&__link {
|
||||||
@extend %t-list-litem-blank;
|
@extend %t-list-litem-blank;
|
||||||
|
display: block;
|
||||||
|
font-size: 1.25em;
|
||||||
|
width: 100%;
|
||||||
|
position: relative;
|
||||||
|
border: 1px solid red;
|
||||||
|
|
||||||
|
&-link {
|
||||||
|
position: relative;
|
||||||
|
display: block;
|
||||||
|
padding: 1.5em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:hover:before {
|
||||||
|
//@include t-translate-x(5%); Disabled due to not being needed on mobile
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&__button {
|
&__button {
|
||||||
@ -47,6 +83,14 @@
|
|||||||
&.is-open {
|
&.is-open {
|
||||||
.o-hamburger-menu__menu {
|
.o-hamburger-menu__menu {
|
||||||
display: block;
|
display: block;
|
||||||
|
@include t-animation-name(o-hamburger-menu--open);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&.is-closing {
|
||||||
|
.o-hamburger-menu__menu {
|
||||||
|
display: block;
|
||||||
|
@include t-animation-name(o-hamburger-menu--close);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -20,7 +20,7 @@
|
|||||||
.o-page-transition__container {
|
.o-page-transition__container {
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
min-height: 100%;
|
||||||
position: relative;
|
position: relative;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -42,7 +42,7 @@ $s-color--navbar__bar-hover: $s-color--pastel-blue;
|
|||||||
$s-color--navbar__bar-active: $s-color--pastel-green;
|
$s-color--navbar__bar-active: $s-color--pastel-green;
|
||||||
|
|
||||||
//Menu Colors
|
//Menu Colors
|
||||||
$s-color--menu__background: red;
|
$s-color--menu__background: rgba(0, 0, 0, 0.8);
|
||||||
|
|
||||||
//loader
|
//loader
|
||||||
$s-color--loader: $s-color--swatch-blue;
|
$s-color--loader: $s-color--swatch-blue;
|
||||||
|
Reference in New Issue
Block a user