feat: fix links in navbar and dropdown
This commit is contained in:
parent
9df0ae90bc
commit
a77e5a1a12
@ -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>
|
||||||
|
@ -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';
|
||||||
|
|
||||||
|
@ -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;
|
||||||
|
@ -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;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user