Added overflow stop for mobile menu, shifting some stuff.

This commit is contained in:
2018-06-18 21:55:24 +10:00
parent c52c04db63
commit 07aa74f359
12 changed files with 244 additions and 67 deletions

View File

@ -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);

View 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
}
}

View File

@ -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);

View File

@ -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} />
});

View File

@ -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>
); );
} }

View 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;

View File

@ -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;

View File

@ -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
} }

View File

@ -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);
} }

View File

@ -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);
} }
} }
} }

View File

@ -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;
} }

View File

@ -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;