import * as React from 'react'; import * as PropTypes from 'prop-types'; import classNames from 'classnames'; import { SiderProps } from './Sider'; import { ConfigConsumer, ConfigConsumerProps } from '../config-provider'; export interface GeneratorProps { suffixCls: string; tagName: 'header' | 'footer' | 'main' | 'section'; } export interface BasicProps extends React.HTMLAttributes { prefixCls?: string; hasSider?: boolean; tagName: 'header' | 'footer' | 'main' | 'section'; } function generator({ suffixCls, tagName }: GeneratorProps) { return (BasicComponent: React.ComponentClass): any => { return class Adapter extends React.Component { static Header: any; static Footer: any; static Content: any; static Sider: any; renderComponent = ({ getPrefixCls }: ConfigConsumerProps) => { const { prefixCls: customizePrefixCls } = this.props; const prefixCls = getPrefixCls(suffixCls, customizePrefixCls); return ; }; render() { return {this.renderComponent}; } }; }; } class Basic extends React.Component { render() { const { prefixCls, className, children, tagName: CustomElement, ...others } = this.props; const classString = classNames(className, prefixCls); return ( {children} ); } } interface BasicLayoutState { siders: string[]; } class BasicLayout extends React.Component { static childContextTypes = { siderHook: PropTypes.object, }; state = { siders: [] }; getChildContext() { return { siderHook: { addSider: (id: string) => { this.setState(state => ({ siders: [...state.siders, id], })); }, removeSider: (id: string) => { this.setState(state => ({ siders: state.siders.filter(currentId => currentId !== id), })); }, }, }; } render() { const { prefixCls, className, children, hasSider, tagName: CustomElement, ...others } = this.props; const classString = classNames(className, prefixCls, { [`${prefixCls}-has-sider`]: hasSider || this.state.siders.length > 0, }); return ( {children} ); } } const Layout: React.ComponentClass & { Header: React.ComponentClass; Footer: React.ComponentClass; Content: React.ComponentClass; Sider: React.ComponentClass; } = generator({ suffixCls: 'layout', tagName: 'section', })(BasicLayout); const Header = generator({ suffixCls: 'layout-header', tagName: 'header', })(Basic); const Footer = generator({ suffixCls: 'layout-footer', tagName: 'footer', })(Basic); const Content = generator({ suffixCls: 'layout-content', tagName: 'main', })(Basic); Layout.Header = Header; Layout.Footer = Footer; Layout.Content = Content; export default Layout;