diff options
author | Fabio Berger <me@fabioberger.com> | 2018-09-28 21:28:39 +0800 |
---|---|---|
committer | Fabio Berger <me@fabioberger.com> | 2018-09-28 21:28:39 +0800 |
commit | 155a4a8f067e36912c3eb6bc279ca58ce90a53c0 (patch) | |
tree | c4df379c0db3da6fb57610c3a42f0f3d7c86c34e | |
parent | 12cfa6b45063b40e39a2f2e6c169fc2102843a16 (diff) | |
download | dexon-sol-tools-155a4a8f067e36912c3eb6bc279ca58ce90a53c0.tar.gz dexon-sol-tools-155a4a8f067e36912c3eb6bc279ca58ce90a53c0.tar.zst dexon-sol-tools-155a4a8f067e36912c3eb6bc279ca58ce90a53c0.zip |
Add sidebar menu, proper scrolling and mobile-optimize
3 files changed, 203 insertions, 91 deletions
diff --git a/packages/website/ts/components/documentation/docs_content_top_bar.tsx b/packages/website/ts/components/documentation/docs_content_top_bar.tsx index dede6f636..1db9e794a 100644 --- a/packages/website/ts/components/documentation/docs_content_top_bar.tsx +++ b/packages/website/ts/components/documentation/docs_content_top_bar.tsx @@ -61,7 +61,7 @@ export class DocsContentTopBar extends React.Component<DocsContentTopBarProps, D }, ]; return ( - <Container height={75} className="pb2 mb3"> + <Container height={80}> <Container className="flex items-center lg-pt3 md-pt3 sm-pt1 relative" width="100%"> <div className="col col-2 sm-hide xs-hide"> <Link diff --git a/packages/website/ts/components/documentation/tutorial_button.tsx b/packages/website/ts/components/documentation/tutorial_button.tsx index 0cb7d9de5..22480888e 100644 --- a/packages/website/ts/components/documentation/tutorial_button.tsx +++ b/packages/website/ts/components/documentation/tutorial_button.tsx @@ -40,10 +40,10 @@ export class TutorialButton extends React.Component<TutorialButtonProps, Tutoria backgroundColor: this.state.isHovering ? '#e7f1fd' : colors.white, }} > - <div> + <div className="col col-1 flex items-center sm-pr3"> <img src={this.props.tutorialInfo.iconUrl} height={40} /> </div> - <div className="pl2"> + <div className="pl2 col col-10"> <Text Tag="div" fontSize="18" fontColor={colors.lightLinkBlue} fontWeight="bold"> {this.props.translate.get(this.props.tutorialInfo.title as Key, Deco.Cap)} </Text> @@ -51,11 +51,13 @@ export class TutorialButton extends React.Component<TutorialButtonProps, Tutoria {this.props.translate.get(this.props.tutorialInfo.description as Key, Deco.Cap)} </Text> </div> - <div className="absolute" style={{ top: 31, right: 31 }}> - <i - className="zmdi zmdi-chevron-right bold" - style={{ fontSize: 26, color: colors.lightLinkBlue }} - /> + <div className="col col-1 flex items-center justify-end"> + <div className="right"> + <i + className="zmdi zmdi-chevron-right bold" + style={{ fontSize: 26, color: colors.lightLinkBlue }} + /> + </div> </div> </div> </Link> diff --git a/packages/website/ts/pages/documentation/home.tsx b/packages/website/ts/pages/documentation/home.tsx index cf2ba0eec..338230358 100644 --- a/packages/website/ts/pages/documentation/home.tsx +++ b/packages/website/ts/pages/documentation/home.tsx @@ -1,7 +1,14 @@ -import { colors, NestedSidebarMenu } from '@0xproject/react-shared'; +import { + colors, + constants, + constants as sharedConstants, + NestedSidebarMenu, + utils as sharedUtils, +} from '@0xproject/react-shared'; import * as _ from 'lodash'; import * as React from 'react'; import DocumentTitle = require('react-document-title'); +import { Element as ScrollElement } from 'react-scroll'; import { DocsContentTopBar } from 'ts/components/documentation/docs_content_top_bar'; import { DocsLogo } from 'ts/components/documentation/docs_logo'; import { TutorialButton } from 'ts/components/documentation/tutorial_button'; @@ -22,6 +29,8 @@ interface Package { } const THROTTLE_TIMEOUT = 100; +const TOP_BAR_HEIGHT = 80; +const SCROLLER_WIDTH = 4; const TUTORIALS: TutorialInfo[] = [ { title: Key.DevelopOnEthereum, @@ -48,8 +57,13 @@ const TUTORIALS: TutorialInfo[] = [ location: `${WebsitePaths.Wiki}#Find,-Submit,-Fill-Order-From-Relayer`, }, ]; +enum Categories { + ZeroExProtocol = '0x Protocol', + Ethereum = 'Ethereum', + CommunityMaintained = 'Community Maintained', +} const CATEGORY_TO_PACKAGES: { [category: string]: Package[] } = { - '0x Protocol': [ + [Categories.ZeroExProtocol]: [ { name: '0x.js', description: @@ -103,7 +117,7 @@ const CATEGORY_TO_PACKAGES: { [category: string]: Package[] } = { shouldOpenInNewTab: true, }, ], - Ethereum: [ + [Categories.Ethereum]: [ { name: 'abi-gen', description: @@ -143,7 +157,7 @@ const CATEGORY_TO_PACKAGES: { [category: string]: Package[] } = { to: WebsitePaths.Web3Wrapper, }, ], - 'Community Maintained': [ + [Categories.CommunityMaintained]: [ { name: '0x Event Extractor', description: @@ -267,13 +281,20 @@ export interface HomeProps { dispatcher: Dispatcher; } -export interface HomeState {} +export interface HomeState { + isHoveringSidebar: boolean; + isHoveringMainContent: boolean; +} export class Home extends React.Component<HomeProps, HomeState> { private readonly _throttledScreenWidthUpdate: () => void; constructor(props: HomeProps) { super(props); this._throttledScreenWidthUpdate = _.throttle(this._updateScreenWidth.bind(this), THROTTLE_TIMEOUT); + this.state = { + isHoveringSidebar: false, + isHoveringMainContent: false, + }; } public componentDidMount(): void { window.addEventListener('resize', this._throttledScreenWidthUpdate); @@ -283,8 +304,33 @@ export class Home extends React.Component<HomeProps, HomeState> { window.removeEventListener('resize', this._throttledScreenWidthUpdate); } public render(): React.ReactNode { + const mainContainerStyle: React.CSSProperties = { + position: 'absolute', + top: 80, + left: 0, + bottom: 0, + right: 0, + overflowX: 'hidden', + overflowY: 'scroll', + minHeight: `calc(100vh - ${TOP_BAR_HEIGHT}px)`, + WebkitOverflowScrolling: 'touch', + }; const isSmallScreen = this.props.screenWidth === ScreenWidths.Sm; - const mainContentPadding = isSmallScreen ? 0 : 50; + const mainContentPadding = isSmallScreen ? 20 : 50; + const topLevelMenu = { + 'Starter guides': _.map(TUTORIALS, tutorialInfo => + this.props.translate.get(tutorialInfo.title as Key, Deco.Cap), + ), + [Categories.ZeroExProtocol]: _.map(CATEGORY_TO_PACKAGES[Categories.ZeroExProtocol], pkg => pkg.name), + [Categories.Ethereum]: _.map(CATEGORY_TO_PACKAGES[Categories.Ethereum], pkg => pkg.name), + [Categories.CommunityMaintained]: _.map( + CATEGORY_TO_PACKAGES[Categories.CommunityMaintained], + pkg => pkg.name, + ), + }; + _.each(TUTORIALS, tutorialInfo => { + const id = sharedUtils.getIdFromName(this.props.translate.get(tutorialInfo.title as Key, Deco.Cap)); + }); return ( <Container className="flex items-center" @@ -294,54 +340,95 @@ export class Home extends React.Component<HomeProps, HomeState> { } 50%, ${colors.white} 100%)`} > <DocumentTitle title="0x Docs Home" /> - <div className="flex mx-auto"> + <div className="flex mx-auto" style={{ height: '100vh' }}> <Container - className="sm-hide xs-hide" + className="sm-hide xs-hide relative" width={234} paddingLeft={22} paddingRight={22} paddingTop={2} backgroundColor={colors.grey100} > - <DocsLogo height={36} containerStyle={{ paddingTop: 28 }} /> + <DocsLogo height={36} containerStyle={{ paddingTop: 30, paddingBottom: 12 }} /> + <div + style={{ + ...mainContainerStyle, + paddingTop: 35, + overflow: this.state.isHoveringSidebar ? 'auto' : 'hidden', + }} + onMouseEnter={this._onSidebarHover.bind(this)} + onMouseLeave={this._onSidebarHoverOff.bind(this)} + > + <NestedSidebarMenu + topLevelMenu={topLevelMenu} + menuSubsectionsBySection={{}} + shouldDisplaySectionHeaders={true} + shouldReformatMenuItemNames={false} + /> + </div> </Container> <Container + className="relative" width={isSmallScreen ? '100vw' : 716} paddingBottom="100px" - paddingLeft={mainContentPadding} - paddingRight={mainContentPadding} backgroundColor={colors.white} > - <DocsContentTopBar location={this.props.location} translate={this.props.translate} /> - <div> - {this._renderSectionTitle('Start building on 0x')} - <Container paddingTop="12px"> - {this._renderSectionDescription( - 'Follow one of our "Getting started" guides to learn more about building ontop of 0x.', - )} - <Container marginTop="36px"> - {_.map(TUTORIALS, tutorialInfo => ( - <TutorialButton - translate={this.props.translate} - tutorialInfo={tutorialInfo} - key={`tutorial-${tutorialInfo.title}`} - /> - ))} + <Container paddingLeft={mainContentPadding} paddingRight={mainContentPadding}> + <DocsContentTopBar location={this.props.location} translate={this.props.translate} /> + </Container> + <div + id={sharedConstants.SCROLL_CONTAINER_ID} + className="absolute" + style={{ + ...mainContainerStyle, + paddingTop: 30, + paddingLeft: mainContentPadding, + paddingRight: this.state.isHoveringMainContent + ? mainContentPadding - SCROLLER_WIDTH + : mainContentPadding, + overflow: this.state.isHoveringMainContent ? 'auto' : 'hidden', + }} + onMouseEnter={this._onMainContentHover.bind(this)} + onMouseLeave={this._onMainContentHoverOff.bind(this)} + > + <div> + {this._renderSectionTitle('Start building on 0x')} + <Container paddingTop="12px"> + {this._renderSectionDescription( + 'Follow one of our "Getting started" guides to learn more about building ontop of 0x.', + )} + <Container marginTop="36px"> + {_.map(TUTORIALS, tutorialInfo => ( + <ScrollElement + name={sharedUtils.getIdFromName( + this.props.translate.get(tutorialInfo.title as Key, Deco.Cap), + )} + key={`tutorial-${tutorialInfo.title}`} + > + <TutorialButton + translate={this.props.translate} + tutorialInfo={tutorialInfo} + /> + </ScrollElement> + ))} + </Container> </Container> - </Container> - </div> - <div className="mt4"> - {this._renderSectionTitle(this.props.translate.get(Key.LibrariesAndTools, Deco.CapWords))} - <Container paddingTop="12px"> - {this._renderSectionDescription( - 'A list of available tools maintained by the 0x core developers and wider community for building on top of 0x Protocol and Ethereum', + </div> + <div className="mt4"> + {this._renderSectionTitle( + this.props.translate.get(Key.LibrariesAndTools, Deco.CapWords), )} - <Container marginTop="36px"> - {_.map(CATEGORY_TO_PACKAGES, (pkgs, category) => - this._renderPackageCategory(category, pkgs), + <Container paddingTop="12px"> + {this._renderSectionDescription( + 'A list of available tools maintained by the 0x core developers and wider community for building on top of 0x Protocol and Ethereum', )} + <Container marginTop="36px"> + {_.map(CATEGORY_TO_PACKAGES, (pkgs, category) => + this._renderPackageCategory(category, pkgs), + )} + </Container> </Container> - </Container> + </div> </div> </Container> </div> @@ -357,56 +444,59 @@ export class Home extends React.Component<HomeProps, HomeState> { ); } private _renderPackage(pkg: Package): React.ReactNode { + const id = sharedUtils.getIdFromName(pkg.name); return ( - <div className="pb2" key={`package-${pkg.name}`}> - <div - style={{ - width: '100%', - height: 1, - backgroundColor: colors.grey300, - marginTop: 11, - }} - /> - <div className="clearfix mt2 pt1"> - <div className="col col-4"> - <Link - to={pkg.to} - className="text-decoration-none" - style={{ color: colors.lightLinkBlue }} - isExternal={!!pkg.isExternal} - shouldOpenInNewTab={!!pkg.shouldOpenInNewTab} - > - <Text Tag="div" fontColor={colors.lightLinkBlue} fontWeight="bold"> - {pkg.name} + <ScrollElement name={id} key={`package-${pkg.name}`}> + <div className="pb2"> + <div + style={{ + width: '100%', + height: 1, + backgroundColor: colors.grey300, + marginTop: 11, + }} + /> + <div className="clearfix mt2 pt1"> + <div className="md-col lg-col md-col-4 lg-col-4"> + <Link + to={pkg.to} + className="text-decoration-none" + style={{ color: colors.lightLinkBlue }} + isExternal={!!pkg.isExternal} + shouldOpenInNewTab={!!pkg.shouldOpenInNewTab} + > + <Text Tag="div" fontColor={colors.lightLinkBlue} fontWeight="bold"> + {pkg.name} + </Text> + </Link> + </div> + <div className="md-col lg-col md-col-6 lg-col-6 sm-py2"> + <Text Tag="div" fontColor={colors.grey700}> + {pkg.description} </Text> - </Link> - </div> - <div className="col col-6"> - <Text Tag="div" fontColor={colors.grey700}> - {pkg.description} - </Text> - </div> - <div className="col col-2 relative"> - <Link - to={pkg.to} - className="text-decoration-none absolute" - style={{ right: 0, color: colors.lightLinkBlue }} - isExternal={!!pkg.isExternal} - shouldOpenInNewTab={!!pkg.shouldOpenInNewTab} - > - <div className="flex"> - <div>{this.props.translate.get(Key.More, Deco.Cap)}</div> - <Container paddingTop="1px" paddingLeft="6px"> - <i - className="zmdi zmdi-chevron-right bold" - style={{ fontSize: 18, color: colors.lightLinkBlue }} - /> - </Container> - </div> - </Link> + </div> + <div className="md-col lg-col md-col-2 lg-col-2 sm-pb2 relative"> + <Link + to={pkg.to} + className="text-decoration-none absolute" + style={{ right: 0, color: colors.lightLinkBlue }} + isExternal={!!pkg.isExternal} + shouldOpenInNewTab={!!pkg.shouldOpenInNewTab} + > + <div className="flex"> + <div>{this.props.translate.get(Key.More, Deco.Cap)}</div> + <Container paddingTop="1px" paddingLeft="6px"> + <i + className="zmdi zmdi-chevron-right bold" + style={{ fontSize: 18, color: colors.lightLinkBlue }} + /> + </Container> + </div> + </Link> + </div> </div> </div> - </div> + </ScrollElement> ); } private _renderSectionTitle(text: string): React.ReactNode { @@ -427,4 +517,24 @@ export class Home extends React.Component<HomeProps, HomeState> { const newScreenWidth = utils.getScreenWidth(); this.props.dispatcher.updateScreenWidth(newScreenWidth); } + private _onSidebarHover(_event: React.FormEvent<HTMLInputElement>): void { + this.setState({ + isHoveringSidebar: true, + }); + } + private _onSidebarHoverOff(): void { + this.setState({ + isHoveringSidebar: false, + }); + } + private _onMainContentHover(_event: React.FormEvent<HTMLInputElement>): void { + this.setState({ + isHoveringMainContent: true, + }); + } + private _onMainContentHoverOff(): void { + this.setState({ + isHoveringMainContent: false, + }); + } } |