aboutsummaryrefslogblamecommitdiffstats
path: root/packages/website/ts/@next/components/header.tsx
blob: 368b1e665cd63b4ec30fbaefde5509c0908c0a0c (plain) (tree)
1
2
3
4
5
6
7
8
9
                       
                               
                                          
                                                              
                                                      
 
                                                    

                                                                                       
                                                          
                                                
                                                          
                                                         
                                                              
 
                       
                        
                           
                                 
                          

 



                   





                                        

 
                             




                         



                                
                                            
                           




                           
                                              








                                   
                                                 
                     
      

  
                                                       
                                      
                                                                    
 
                
                          

                                                
                                
                                      
 
                              
                                                           


                                                        
                              
                           
                               
 








                                                               
 
                                                                                 

                                                                                            

                           
     

 

                                            
                                                            
                           


                                          
                  
                                         







                                                         
                   


      

                                                

  
                           

















                                                                      

                                
                                             

                              

                    





                   
                                    

                                 
 
                             





                        
                           
                  

                                   
 
                               

                      

  
                                                       
                                            
                    



                                                              

                       
                                                             



                                                                    
                
 
                       







                                  

             
                                             
                                                                

                           
     





                                                                         
 
                               

                      

  
                                   
                                  
  
import _ from 'lodash';
import * as React from 'react';
import MediaQuery from 'react-responsive';
import { NavLink as ReactRouterLink } from 'react-router-dom';
import styled, { withTheme } from 'styled-components';

import { Button } from 'ts/@next/components/button';
import { DropdownDevelopers } from 'ts/@next/components/dropdowns/dropdown_developers';
import { DropdownProducts } from 'ts/@next/components/dropdowns/dropdown_products';
import { Hamburger } from 'ts/@next/components/hamburger';
import { Logo } from 'ts/@next/components/logo';
import { MobileNav } from 'ts/@next/components/mobileNav';
import { FlexWrap } from 'ts/@next/components/newLayout';
import { ThemeInterface } from 'ts/@next/components/siteWrap';

interface HeaderProps {
    location?: Location;
    isNavToggled?: boolean;
    toggleMobileNav?: () => void;
    theme: ThemeInterface;
}

interface NavItem {
    url?: string;
    id?: string;
    text?: string;
    dropdownWidth?: number;
    dropdownComponent?: React.ReactNode;
}

interface DropdownWrapInterface {
    width?: number;
}

const navItems: NavItem[] = [
    {
        id: 'why',
        url: '/next/why',
        text: 'Why 0x',
    },
    {
        id: 'products',
        url: '/next/0x-instant',
        text: 'Products',
        dropdownComponent: DropdownProducts,
        dropdownWidth: 280,
    },
    {
        id: 'developers',
        url: '#',
        text: 'Developers',
        dropdownComponent: DropdownDevelopers,
        dropdownWidth: 480,
    },
    {
        id: 'about',
        url: '/next/about/mission',
        text: 'About',
    },
    {
        id: 'blog',
        url: 'https://blog.0xproject.com/latest',
        text: 'Blog',
    },
];

class HeaderBase extends React.Component<HeaderProps> {
    public render(): React.ReactNode {
        const { isNavToggled, toggleMobileNav, theme } = this.props;

        return (
            <StyledHeader>
                <HeaderWrap>
                    <ReactRouterLink to="/next">
                        <Logo />
                    </ReactRouterLink>

                    <NavLinks>
                        {_.map(navItems, (link, index) => (
                            <NavItem
                                key={`navlink-${index}`}
                                link={link}
                            />
                        ))}
                    </NavLinks>

                    <MediaQuery minWidth={990}>
                        <TradeButton
                            bgColor={theme.headerButtonBg}
                            color="#ffffff"
                            href="https://0xproject.com/portal"
                        >
                            Trade on 0x
                        </TradeButton>
                    </MediaQuery>

                    <Hamburger isOpen={isNavToggled} onClick={toggleMobileNav} />
                    <MobileNav isToggled={isNavToggled} toggleMobileNav={toggleMobileNav} />
                </HeaderWrap>
            </StyledHeader>
        );
    }
}

export const Header = withTheme(HeaderBase);

const NavItem = (props: { link: NavItem; key: string }) => {
    const { link } = props;
    const Subnav = link.dropdownComponent;

    return (
        <LinkWrap>
            <StyledNavLink to={link.url}>
                {link.text}
            </StyledNavLink>

            {link.dropdownComponent &&
                <DropdownWrap width={link.dropdownWidth}>
                    <Subnav />
                </DropdownWrap>
            }
        </LinkWrap>
    );
};

const StyledHeader = styled.header<HeaderProps>`
    padding: 30px;
`;

const LinkWrap = styled.li`
    position: relative;

    a {
        display: block;
    }

    @media (min-width: 800px) {
        &:hover > div {
            display: block;
            visibility: visible;
            opacity: 1;
            transform: translate3d(0, 0, 0);
            transition: opacity 0.35s, transform 0.35s, visibility 0s;
        }
    }
`;

const StyledNavLink = styled(ReactRouterLink).attrs({
    activeStyle: { opacity: 1 },
})`
    color: ${props => props.theme.textColor};
    opacity: 0.5;
    transition: opacity 0.35s;
    padding: 15px 0;
    margin: 0 30px;

    &:hover {
        opacity: 1;
    }
`;

const HeaderWrap = styled(FlexWrap)`
  justify-content: space-between;
  align-items: center;

  @media (max-width: 800px) {
      padding-top: 0;
      display: flex;
      padding-bottom: 0;
  }
`;

const NavLinks = styled.ul`
    display: flex;
    align-items: center;
    justify-content: space-between;

    @media (max-width: 800px) {
        display: none;
    }
`;

const DropdownWrap = styled.div<DropdownWrapInterface>`
    width: ${props => props.width || 280}px;
    padding: 15px 0;
    border: 1px solid transparent;
    border-color: ${props => props.theme.dropdownBorderColor};
    background-color: ${props => props.theme.dropdownBg};
    color: ${props => props.theme.dropdownColor};
    position: absolute;
    top: 100%;
    left: calc(50% - ${props => (props.width || 280) / 2}px);
    visibility: hidden;
    opacity: 0;
    transform: translate3d(0, -10px, 0);
    transition: opacity 0.35s, transform 0.35s, visibility 0s 0.35s;
    z-index: 20;

    &:after, &:before {
        bottom: 100%;
        left: 50%;
        border: solid transparent;
        content: " ";
        height: 0;
        width: 0;
        position: absolute;
        pointer-events: none;
    }
    &:after {
        border-color: rgba(255, 255, 255, 0);
        border-bottom-color: ${props => props.theme.dropdownBg};
        border-width: 10px;
        margin-left: -10px;
    }
    &:before {
        border-color: rgba(255, 0, 0, 0);
        border-bottom-color: ${props => props.theme.dropdownBorderColor};
        border-width: 11px;
        margin-left: -11px;
    }

    @media (max-width: 768px) {
        display: none;
    }
`;

const TradeButton = styled(Button)`
    padding: 14px 22px !important;
`;