2017-11-30 21:38:48 +08:00
import React from 'react' ;
2016-05-27 11:48:08 +08:00
import ReactDOM from 'react-dom' ;
2017-04-12 04:49:08 +08:00
import PropTypes from 'prop-types' ;
2019-08-08 12:35:48 +08:00
import { IntlProvider } from 'react-intl' ;
2019-12-25 19:04:29 +08:00
import { presetPalettes , presetDarkPalettes } from '@ant-design/colors' ;
2020-01-02 17:54:11 +08:00
import themeSwitcher from 'theme-switcher' ;
2019-12-25 10:10:28 +08:00
import { setTwoToneColor } from '@ant-design/icons' ;
2019-11-09 17:42:41 +08:00
import { Helmet , HelmetProvider } from 'react-helmet-async' ;
2018-05-18 11:34:26 +08:00
import 'moment/locale/zh-cn' ;
2019-07-24 10:34:55 +08:00
import { ConfigProvider } from 'antd' ;
2019-02-26 16:16:23 +08:00
import LogRocket from 'logrocket' ;
import setupLogRocketReact from 'logrocket-react' ;
2019-08-06 15:02:05 +08:00
// eslint-disable-next-line import/no-unresolved
2019-08-05 15:37:00 +08:00
import zhCN from 'antd/es/locale/zh_CN' ;
2016-05-27 11:48:08 +08:00
import Header from './Header' ;
2020-02-27 10:53:30 +08:00
import SiteContext from './SiteContext' ;
2016-09-27 10:06:34 +08:00
import enLocale from '../../en-US' ;
import cnLocale from '../../zh-CN' ;
2016-10-18 11:18:25 +08:00
import * as utils from '../utils' ;
2016-05-27 11:48:08 +08:00
2018-09-26 11:13:19 +08:00
if ( typeof window !== 'undefined' && navigator . serviceWorker ) {
2018-12-07 16:17:45 +08:00
navigator . serviceWorker . getRegistrations ( ) . then ( registrations => {
2018-09-26 11:13:19 +08:00
registrations . forEach ( registration => registration . unregister ( ) ) ;
} ) ;
}
2016-11-21 14:39:15 +08:00
if ( typeof window !== 'undefined' ) {
2020-02-27 17:11:29 +08:00
// Redirect to `ant.design` if is not next version anymore
2020-02-29 00:03:19 +08:00
if ( location . hostname === 'next.ant.design' ) {
2020-02-27 17:11:29 +08:00
location . href = location . href . replace ( 'next.ant.design' , 'ant.design' ) ;
}
2019-08-06 15:02:05 +08:00
// eslint-disable-next-line global-require
2016-11-21 14:39:15 +08:00
require ( '../../static/style' ) ;
// Expose to iframe
window . react = React ;
window [ 'react-dom' ] = ReactDOM ;
2019-08-06 15:09:07 +08:00
// eslint-disable-next-line global-require
2016-11-21 14:39:15 +08:00
window . antd = require ( 'antd' ) ;
2019-08-13 14:07:17 +08:00
// eslint-disable-next-line global-require
window [ '@ant-design/icons' ] = require ( '@ant-design/icons' ) ;
2019-02-27 15:31:32 +08:00
// Error log statistic
2019-03-06 12:23:41 +08:00
window . addEventListener ( 'error' , function onError ( e ) {
2019-02-27 15:31:32 +08:00
// Ignore ResizeObserver error
if ( e . message === 'ResizeObserver loop limit exceeded' ) {
e . stopPropagation ( ) ;
e . stopImmediatePropagation ( ) ;
}
} ) ;
if ( process . env . NODE _ENV === 'production' ) {
LogRocket . init ( 'kpuw4z/ant-design' ) ;
setupLogRocketReact ( LogRocket ) ;
}
2016-11-21 14:39:15 +08:00
}
2016-05-27 11:48:08 +08:00
2020-02-27 10:53:30 +08:00
const RESPONSIVE _MOBILE = 768 ;
2019-12-23 17:41:55 +08:00
const SITE _THEME _STORE _KEY = 'site-theme' ;
2019-12-24 11:14:13 +08:00
// for dark.css timestamp to remove cache
const timestamp = new Date ( ) . getTime ( ) ;
2020-01-02 17:54:11 +08:00
const themeMap = {
dark : ` /dark.css? ${ timestamp } ` ,
} ;
const themeConfig = {
themeMap ,
} ;
const { switcher } = themeSwitcher ( themeConfig ) ;
2019-12-24 11:14:13 +08:00
2016-06-03 15:26:25 +08:00
export default class Layout extends React . Component {
static contextTypes = {
2017-04-12 04:49:08 +08:00
router : PropTypes . object . isRequired ,
2018-12-07 16:17:45 +08:00
} ;
2018-06-22 21:05:13 +08:00
2018-01-04 20:00:38 +08:00
static childContextTypes = {
2019-12-18 18:45:05 +08:00
theme : PropTypes . oneOf ( [ 'default' , 'dark' ] ) ,
setTheme : PropTypes . func ,
2020-01-02 19:10:16 +08:00
direction : PropTypes . string ,
2020-01-02 17:54:11 +08:00
setIframeTheme : PropTypes . func ,
2018-01-04 20:00:38 +08:00
} ;
2016-12-09 13:02:16 +08:00
constructor ( props ) {
super ( props ) ;
2017-10-09 13:23:20 +08:00
const { pathname } = props . location ;
2016-12-09 13:02:16 +08:00
const appLocale = utils . isZhCN ( pathname ) ? cnLocale : enLocale ;
2018-04-22 18:58:40 +08:00
2016-12-09 13:02:16 +08:00
this . state = {
appLocale ,
2019-12-23 17:41:55 +08:00
theme :
typeof localStorage !== 'undefined'
? localStorage . getItem ( SITE _THEME _STORE _KEY ) || 'default'
: 'default' ,
2019-12-18 18:45:05 +08:00
setTheme : this . setTheme ,
2020-01-02 19:10:16 +08:00
direction : 'ltr' ,
2020-01-02 17:54:11 +08:00
setIframeTheme : this . setIframeTheme ,
2016-12-09 13:02:16 +08:00
} ;
2020-01-02 19:10:16 +08:00
this . changeDirection = this . changeDirection . bind ( this ) ;
2016-12-09 13:02:16 +08:00
}
2016-11-09 19:55:14 +08:00
2018-11-28 15:00:03 +08:00
getChildContext ( ) {
2020-02-27 10:53:30 +08:00
const { theme , setTheme , direction , setIframeTheme } = this . state ;
return { theme , setTheme , direction , setIframeTheme } ;
2018-11-28 15:00:03 +08:00
}
2016-06-03 15:26:25 +08:00
componentDidMount ( ) {
2019-12-23 17:41:55 +08:00
const { theme } = this . state ;
2019-12-24 11:32:46 +08:00
const { location } = this . props ;
2018-06-22 21:05:13 +08:00
const { router } = this . context ;
2018-12-07 16:17:45 +08:00
router . listen ( loc => {
2018-05-21 23:27:26 +08:00
if ( typeof window . ga !== 'undefined' ) {
2016-06-03 15:26:25 +08:00
window . ga ( 'send' , 'pageview' , loc . pathname + loc . search ) ;
2018-05-21 23:27:26 +08:00
}
// eslint-disable-next-line
if ( typeof window . _hmt !== 'undefined' ) {
// eslint-disable-next-line
window . _hmt . push ( [ '_trackPageview' , loc . pathname + loc . search ] ) ;
}
2019-12-23 17:58:59 +08:00
const { pathname } = loc ;
const componentPage = /^\/?components/ . test ( pathname ) ;
// only component page can use `dark` theme
if ( ! componentPage ) {
this . setTheme ( 'default' , false ) ;
}
2018-05-21 23:27:26 +08:00
} ) ;
2019-12-24 11:32:46 +08:00
this . setTheme ( /^\/?components/ . test ( location . pathname ) ? theme : 'default' ) ;
2019-12-23 17:41:55 +08:00
2016-12-09 14:24:38 +08:00
const nprogressHiddenStyle = document . getElementById ( 'nprogress-style' ) ;
if ( nprogressHiddenStyle ) {
2016-07-26 17:40:08 +08:00
this . timer = setTimeout ( ( ) => {
2016-12-09 14:24:38 +08:00
nprogressHiddenStyle . parentNode . removeChild ( nprogressHiddenStyle ) ;
} , 0 ) ;
2016-06-23 21:10:02 +08:00
}
2018-01-04 20:00:38 +08:00
2020-02-27 10:53:30 +08:00
this . updateMobileMode ( ) ;
window . addEventListener ( 'resize' , this . updateMobileMode ) ;
2016-06-23 21:10:02 +08:00
}
componentWillUnmount ( ) {
clearTimeout ( this . timer ) ;
2020-02-27 10:53:30 +08:00
window . removeEventListener ( 'resize' , this . updateMobileMode ) ;
2016-06-03 15:26:25 +08:00
}
2020-02-27 10:53:30 +08:00
updateMobileMode = ( ) => {
const { isMobile } = this . state ;
const newIsMobile = window . innerWidth < RESPONSIVE _MOBILE ;
if ( isMobile !== newIsMobile ) {
this . setState ( {
isMobile : newIsMobile ,
} ) ;
}
} ;
2020-01-02 17:54:11 +08:00
setIframeTheme = ( iframeNode , theme ) => {
iframeNode . contentWindow . postMessage (
JSON . stringify ( {
action : 'change.theme' ,
data : {
themeConfig ,
theme ,
} ,
} ) ,
'*' ,
) ;
} ;
2019-12-23 17:58:59 +08:00
setTheme = ( theme , persist = true ) => {
2019-12-23 16:02:09 +08:00
if ( typeof window === 'undefined' ) {
return ;
}
2020-01-02 17:54:11 +08:00
switcher ( {
theme ,
useStorage : persist ,
} ) ;
const iframeNodes = document . querySelectorAll ( '.iframe-demo' ) ;
// loop element node
[ ] . forEach . call ( iframeNodes , iframeNode => {
this . setIframeTheme ( iframeNode , theme ) ;
} ) ;
2019-12-18 18:45:05 +08:00
this . setState ( {
theme ,
} ) ;
2019-12-25 10:10:28 +08:00
const iconTwoToneThemeMap = {
2019-12-25 19:04:29 +08:00
dark : [ presetDarkPalettes . blue . primary , '#111d2c' ] ,
2019-12-25 10:10:28 +08:00
default : presetPalettes . blue . primary ,
} ;
setTwoToneColor ( iconTwoToneThemeMap [ theme ] || iconTwoToneThemeMap . default ) ;
2019-12-18 18:45:05 +08:00
} ;
2020-01-02 19:10:16 +08:00
changeDirection ( direction ) {
this . setState ( {
direction ,
} ) ;
}
2016-06-03 15:26:25 +08:00
render ( ) {
2019-11-11 14:30:11 +08:00
const { children , helmetContext = { } , ... restProps } = this . props ;
2020-02-27 10:53:30 +08:00
const { appLocale , direction , isMobile } = this . state ;
2019-09-16 11:23:26 +08:00
const title =
appLocale . locale === 'zh-CN'
? 'Ant Design - 一套企业级 UI 设计语言和 React 组件库'
: 'Ant Design - A UI Design Language and React UI library' ;
const description =
appLocale . locale === 'zh-CN'
? '基于 Ant Design 设计体系的 React UI 组件库,用于研发企业级中后台产品。'
: 'An enterprise-class UI design language and React UI library with a set of high-quality React components, one of best React UI library for enterprises' ;
2020-01-02 19:10:16 +08:00
let pageWrapperClass = 'page-wrapper' ;
if ( direction === 'rtl' ) {
pageWrapperClass += ' page-wrapper-rtl' ;
}
2016-06-03 15:26:25 +08:00
return (
2020-02-27 10:53:30 +08:00
< SiteContext.Provider value = { { isMobile } } >
< HelmetProvider context = { helmetContext } >
< Helmet encodeSpecialCharacters = { false } >
< html lang = { appLocale . locale === 'zh-CN' ? 'zh' : 'en' } data - direction = { direction } / >
< title > { title } < / title >
< link
rel = "apple-touch-icon-precomposed"
sizes = "144x144"
href = "https://gw.alipayobjects.com/zos/antfincdn/UmVnt3t4T0/antd.png"
/ >
< meta name = "description" content = { description } / >
< meta property = "og:title" content = { title } / >
< meta property = "og:type" content = "website" / >
< meta
property = "og:image"
content = "https://gw.alipayobjects.com/zos/rmsportal/rlpTLlbMzTNYuZGGCVYM.png"
/ >
< / Helmet >
< IntlProvider
locale = { appLocale . locale }
messages = { appLocale . messages }
defaultLocale = "en-US"
>
< ConfigProvider
locale = { appLocale . locale === 'zh-CN' ? zhCN : null }
direction = { direction }
>
< div className = { pageWrapperClass } >
< Header { ...restProps } changeDirection = { this . changeDirection } / >
{ children }
< / div >
< / ConfigProvider >
< / IntlProvider >
< / HelmetProvider >
< / SiteContext.Provider >
2016-06-03 15:26:25 +08:00
) ;
}
}