import React from 'react'; import { FormattedMessage, injectIntl } from 'react-intl'; import classNames from 'classnames'; import { UnorderedListOutlined } from '@ant-design/icons'; import { Select, Row, Col, Popover, Button } from 'antd'; import * as utils from '../../utils'; import packageJson from '../../../../../package.json'; import Logo from './Logo'; import SearchBox from './SearchBox'; import More from './More'; import Navigation from './Navigation'; import Github from './Github'; import SiteContext from '../SiteContext'; import { ping } from '../../utils'; import './index.less'; const RESPONSIVE_XS = 1120; const RESPONSIVE_SM = 1200; const { Option } = Select; const antdVersion: string = packageJson.version; let docsearch: any; if (typeof window !== 'undefined') { // eslint-disable-next-line global-require docsearch = require('docsearch.js'); } function initDocSearch(locale: string) { if (!docsearch) { return; } const lang = locale === 'zh-CN' ? 'cn' : 'en'; docsearch({ apiKey: '60ac2c1a7d26ab713757e4a081e133d0', indexName: 'ant_design', inputSelector: '#search-box input', algoliaOptions: { facetFilters: [`tags:${lang}`] }, transformData(hits: { url: string }[]) { hits.forEach(hit => { hit.url = hit.url.replace('ant.design', window.location.host); hit.url = hit.url.replace('https:', window.location.protocol); }); return hits; }, debug: false, // Set debug to true if you want to inspect the dropdown }); } export interface HeaderProps { intl: { locale: string; }; location: { pathname: string; query: any }; router: any; themeConfig: { docVersions: Record }; changeDirection: (direction: string) => void; } interface HeaderState { menuVisible: boolean; windowWidth: number; searching: boolean; showTechUIButton: boolean; } class Header extends React.Component { static contextType = SiteContext; pingTimer: NodeJS.Timeout; state = { menuVisible: false, windowWidth: 1400, searching: false, showTechUIButton: false, }; componentDidMount() { const { intl, router } = this.props; router.listen(this.handleHideMenu); initDocSearch(intl.locale); window.addEventListener('resize', this.onWindowResize); this.onWindowResize(); this.pingTimer = ping(status => { if (status !== 'timeout' && status !== 'error') { this.setState({ showTechUIButton: true, }); } }); } componentWillUnmount() { window.removeEventListener('resize', this.onWindowResize); clearTimeout(this.pingTimer); } onWindowResize = () => { this.setState({ windowWidth: window.innerWidth, }); }; onTriggerSearching = (searching: boolean) => { this.setState({ searching }); }; handleShowMenu = () => { this.setState({ menuVisible: true, }); }; handleHideMenu = () => { this.setState({ menuVisible: false, }); }; onDirectionChange = () => { const { changeDirection } = this.props; const { direction } = this.context; if (direction !== 'rtl') { changeDirection('rtl'); } else { changeDirection('ltr'); } }; getNextDirectionText = () => { const { direction } = this.context; if (direction !== 'rtl') { return 'RTL'; } return 'LTR'; }; getDropdownStyle = (): React.CSSProperties => { const { direction } = this.context; if (direction === 'rtl') { return { direction: 'ltr', textAlign: 'right', }; } return {}; }; onMenuVisibleChange = (visible: boolean) => { this.setState({ menuVisible: visible, }); }; handleVersionChange = (url: string) => { const currentUrl = window.location.href; const currentPathname = window.location.pathname; if (/overview/.test(currentPathname) && /0?[1-39][0-3]?x/.test(url)) { window.location.href = currentUrl .replace(window.location.origin, url) .replace(/\/components\/overview/, `/docs${/0(9|10)x/.test(url) ? '' : '/react'}/introduce`) .replace(/\/$/, ''); return; } window.location.href = currentUrl.replace(window.location.origin, url); }; onLangChange = () => { const { location: { pathname, query }, } = this.props; const currentProtocol = `${window.location.protocol}//`; const currentHref = window.location.href.substr(currentProtocol.length); if (utils.isLocalStorageNameSupported()) { localStorage.setItem('locale', utils.isZhCN(pathname) ? 'en-US' : 'zh-CN'); } window.location.href = currentProtocol + currentHref.replace( window.location.pathname, utils.getLocalizedPathname(pathname, !utils.isZhCN(pathname), query).pathname, ); }; render() { return ( {({ isMobile }) => { const { menuVisible, windowWidth, searching, showTechUIButton } = this.state; const { direction } = this.context; const { location, themeConfig, intl: { locale }, } = this.props; const docVersions: Record = { [antdVersion]: antdVersion, ...themeConfig.docVersions, }; const versionOptions = Object.keys(docVersions).map(version => ( )); const pathname = location.pathname.replace(/(^\/|\/$)/g, ''); const isHome = ['', 'index', 'index-cn'].includes(pathname); const isZhCN = locale === 'zh-CN'; const isRTL = direction === 'rtl'; let responsive: null | 'narrow' | 'crowded' = null; if (windowWidth < RESPONSIVE_XS) { responsive = 'crowded'; } else if (windowWidth < RESPONSIVE_SM) { responsive = 'narrow'; } const headerClassName = classNames({ clearfix: true, 'home-header': isHome, }); const sharedProps = { isZhCN, isRTL, }; const searchBox = ( ); const navigationNode = ( ); let menu: (React.ReactElement | null)[] = [ navigationNode, , , , , , ]; if (windowWidth < RESPONSIVE_XS) { menu = searching ? [] : [navigationNode]; } else if (windowWidth < RESPONSIVE_SM) { menu = searching ? [] : menu; } const colProps = isHome ? [{ flex: 'none' }, { flex: 'auto' }] : [ { xxl: 4, xl: 5, lg: 6, md: 6, sm: 24, xs: 24, }, { xxl: 20, xl: 19, lg: 18, md: 18, sm: 0, xs: 0, }, ]; return ( ); }} ); } } export default injectIntl(Header as any);