feat: fix links in navbar and dropdown

This commit is contained in:
fragosti 2018-12-17 14:37:34 -08:00
parent 9df0ae90bc
commit a77e5a1a12
4 changed files with 71 additions and 51 deletions

View File

@ -1,13 +1,13 @@
import * as _ from 'lodash'; import * as _ from 'lodash';
import * as React from 'react'; import * as React from 'react';
import { Link as ReactRounterLink } from 'react-router-dom'; import { NavLink as ReactRounterLink } from 'react-router-dom';
import { Link as ScrollLink } from 'react-scroll'; import { Link as ScrollLink } from 'react-scroll';
import * as validUrl from 'valid-url'; import * as validUrl from 'valid-url';
import { LinkType } from '../types'; import { LinkType } from '../types';
import { constants } from '../utils/constants'; import { constants } from '../utils/constants';
interface BaseLinkProps { export interface BaseLinkProps {
to: string; to: string;
shouldOpenInNewTab?: boolean; shouldOpenInNewTab?: boolean;
className?: string; className?: string;
@ -18,11 +18,15 @@ interface BaseLinkProps {
fontColor?: string; fontColor?: string;
} }
interface ScrollLinkProps extends BaseLinkProps { export interface ScrollLinkProps extends BaseLinkProps {
onActivityChanged?: (isActive: boolean) => void; onActivityChanged?: (isActive: boolean) => void;
} }
type LinkProps = BaseLinkProps & ScrollLinkProps; export interface ReactLinkProps extends BaseLinkProps {
activeStyle?: React.CSSProperties;
}
export type LinkProps = ReactLinkProps & ScrollLinkProps;
export interface LinkState {} export interface LinkState {}
@ -94,6 +98,7 @@ export class Link extends React.Component<LinkProps, LinkState> {
onMouseOver={this.props.onMouseOver} onMouseOver={this.props.onMouseOver}
onMouseEnter={this.props.onMouseEnter} onMouseEnter={this.props.onMouseEnter}
onMouseLeave={this.props.onMouseLeave} onMouseLeave={this.props.onMouseLeave}
activeStyle={this.props.activeStyle}
> >
{this.props.children} {this.props.children}
</ReactRounterLink> </ReactRounterLink>

View File

@ -3,7 +3,7 @@ export { MarkdownLinkBlock } from './components/markdown_link_block';
export { MarkdownCodeBlock } from './components/markdown_code_block'; export { MarkdownCodeBlock } from './components/markdown_code_block';
export { MarkdownSection } from './components/markdown_section'; export { MarkdownSection } from './components/markdown_section';
export { SectionHeader } from './components/section_header'; export { SectionHeader } from './components/section_header';
export { Link } from './components/link'; export { Link, LinkProps } from './components/link';
export { HeaderSizes, Styles, EtherscanLinkSuffixes, Networks, ALink } from './types'; export { HeaderSizes, Styles, EtherscanLinkSuffixes, Networks, ALink } from './types';

View File

@ -1,8 +1,8 @@
import { Link } from '@0x/react-shared';
import _ from 'lodash'; import _ from 'lodash';
import * as React from 'react'; import * as React from 'react';
import MediaQuery from 'react-responsive'; import MediaQuery from 'react-responsive';
import { NavLink as ReactRouterLink } from 'react-router-dom'; import styled, { css, withTheme } from 'styled-components';
import styled, { withTheme } from 'styled-components';
import Headroom from 'react-headroom'; import Headroom from 'react-headroom';
@ -29,6 +29,7 @@ interface NavItemProps {
text?: string; text?: string;
dropdownWidth?: number; dropdownWidth?: number;
dropdownComponent?: React.ReactNode; dropdownComponent?: React.ReactNode;
shouldOpenInNewTab?: boolean;
} }
interface DropdownWrapInterface { interface DropdownWrapInterface {
@ -43,14 +44,12 @@ const navItems: NavItemProps[] = [
}, },
{ {
id: 'products', id: 'products',
url: '#',
text: 'Products', text: 'Products',
dropdownComponent: DropdownProducts, dropdownComponent: DropdownProducts,
dropdownWidth: 280, dropdownWidth: 280,
}, },
{ {
id: 'developers', id: 'developers',
url: '#',
text: 'Developers', text: 'Developers',
dropdownComponent: DropdownDevelopers, dropdownComponent: DropdownDevelopers,
dropdownWidth: 480, dropdownWidth: 480,
@ -62,7 +61,8 @@ const navItems: NavItemProps[] = [
}, },
{ {
id: 'blog', id: 'blog',
url: 'https://blog.0x.org/latest', url: 'https://blog.0xproject.com/latest',
shouldOpenInNewTab: true,
text: 'Blog', text: 'Blog',
}, },
]; ];
@ -72,7 +72,7 @@ class HeaderBase extends React.Component<HeaderProps> {
if (this.props.isNavToggled) { if (this.props.isNavToggled) {
this.props.toggleMobileNav(); this.props.toggleMobileNav();
} }
} };
public render(): React.ReactNode { public render(): React.ReactNode {
const { isNavToggled, toggleMobileNav, theme } = this.props; const { isNavToggled, toggleMobileNav, theme } = this.props;
@ -81,25 +81,16 @@ class HeaderBase extends React.Component<HeaderProps> {
<Headroom onUnpin={this.onUnpin} downTolerance={4} upTolerance={10}> <Headroom onUnpin={this.onUnpin} downTolerance={4} upTolerance={10}>
<StyledHeader isNavToggled={isNavToggled}> <StyledHeader isNavToggled={isNavToggled}>
<HeaderWrap> <HeaderWrap>
<ReactRouterLink to={WebsitePaths.Home}> <Link to={WebsitePaths.Home}>
<Logo /> <Logo />
</ReactRouterLink> </Link>
<NavLinks> <NavLinks>
{_.map(navItems, (link, index) => ( {_.map(navItems, (link, index) => <NavItem key={`navlink-${index}`} link={link} />)}
<NavItem
key={`navlink-${index}`}
link={link}
/>
))}
</NavLinks> </NavLinks>
<MediaQuery minWidth={990}> <MediaQuery minWidth={990}>
<TradeButton <TradeButton bgColor={theme.headerButtonBg} color="#ffffff" href="/portal">
bgColor={theme.headerButtonBg}
color="#ffffff"
href="/portal"
>
Trade on 0x Trade on 0x
</TradeButton> </TradeButton>
</MediaQuery> </MediaQuery>
@ -118,23 +109,30 @@ export const Header = withTheme(HeaderBase);
const NavItem = (props: { link: NavItemProps; key: string }) => { const NavItem = (props: { link: NavItemProps; key: string }) => {
const { link } = props; const { link } = props;
const Subnav = link.dropdownComponent; const Subnav = link.dropdownComponent;
const linkElement = _.isUndefined(link.url) ? (
return ( <StyledAnchor href="#">{link.text}</StyledAnchor>
<LinkWrap> ) : (
<StyledNavLink to={link.url}> <StyledNavLink to={link.url} shouldOpenInNewTab={link.shouldOpenInNewTab}>
{link.text} {link.text}
</StyledNavLink> </StyledNavLink>
);
return (
<LinkWrap>
{linkElement}
{link.dropdownComponent && {link.dropdownComponent && (
<DropdownWrap width={link.dropdownWidth}> <DropdownWrap width={link.dropdownWidth}>
<Subnav /> <Subnav />
</DropdownWrap> </DropdownWrap>
} )}
</LinkWrap> </LinkWrap>
); );
}; };
const StyledHeader = styled.header<HeaderProps>` const StyledHeader =
styled.header <
HeaderProps >
`
padding: 30px; padding: 30px;
background-color: ${props => props.theme.bgColor}; background-color: ${props => props.theme.bgColor};
`; `;
@ -157,9 +155,7 @@ const LinkWrap = styled.li`
} }
`; `;
const StyledNavLink = styled(ReactRouterLink).attrs({ const linkStyles = css`
activeStyle: { opacity: 1 },
})`
color: ${props => props.theme.textColor}; color: ${props => props.theme.textColor};
opacity: 0.5; opacity: 0.5;
transition: opacity 0.35s; transition: opacity 0.35s;
@ -171,6 +167,16 @@ const StyledNavLink = styled(ReactRouterLink).attrs({
} }
`; `;
const StyledNavLink = styled(Link).attrs({
activeStyle: { opacity: 1 },
})`
${linkStyles};
`;
const StyledAnchor = styled.a`
${linkStyles};
`;
const HeaderWrap = styled(FlexWrap)` const HeaderWrap = styled(FlexWrap)`
justify-content: space-between; justify-content: space-between;
align-items: center; align-items: center;
@ -192,7 +198,10 @@ const NavLinks = styled.ul`
} }
`; `;
const DropdownWrap = styled.div<DropdownWrapInterface>` const DropdownWrap =
styled.div <
DropdownWrapInterface >
`
width: ${props => props.width || 280}px; width: ${props => props.width || 280}px;
padding: 15px 0; padding: 15px 0;
border: 1px solid transparent; border: 1px solid transparent;

View File

@ -1,5 +1,5 @@
import { Link as SmartLink } from '@0x/react-shared';
import * as React from 'react'; import * as React from 'react';
import { Link as ReactRouterLink } from 'react-router-dom';
import styled from 'styled-components'; import styled from 'styled-components';
interface LinkInterface { interface LinkInterface {
@ -16,16 +16,19 @@ interface LinkInterface {
} }
export const Link = (props: LinkInterface) => { export const Link = (props: LinkInterface) => {
const { const { children, isNoArrow, href } = props;
children,
isNoArrow,
href,
} = props;
return ( return (
<StyledLink to={href} {...props}> <StyledLink to={href} {...props}>
{children} {children}
{!isNoArrow && <svg width="25" height="25" fill="none" xmlns="http://www.w3.org/2000/svg"><path d="M8.484 5.246l.023 1.411 8.147.053L4.817 18.547l.996.996L17.65 7.706l.052 8.146 1.411.024-.068-10.561-10.561-.069z" fill="currentColor"/></svg>} {!isNoArrow && (
<svg width="25" height="25" fill="none" xmlns="http://www.w3.org/2000/svg">
<path
d="M8.484 5.246l.023 1.411 8.147.053L4.817 18.547l.996.996L17.65 7.706l.052 8.146 1.411.024-.068-10.561-10.561-.069z"
fill="currentColor"
/>
</svg>
)}
</StyledLink> </StyledLink>
); );
}; };
@ -39,7 +42,10 @@ export const LinkWrap = styled.div`
} }
`; `;
const StyledLink = styled(ReactRouterLink)<LinkInterface>` const StyledLink =
styled(SmartLink) <
LinkInterface >
`
display: ${props => !props.isBlock && 'inline-flex'}; display: ${props => !props.isBlock && 'inline-flex'};
color: ${props => props.color || props.theme.linkColor}; color: ${props => props.color || props.theme.linkColor};
text-align: center; text-align: center;