import * as React from 'react'; 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; } export interface LayoutContextProps { siderHook: { addSider: (id: string) => void; removeSider: (id: string) => void; }; } export const LayoutContext = React.createContext({ siderHook: { addSider: () => null, removeSider: () => null, }, }); interface BasicPropsWithTagName extends BasicProps { tagName: 'header' | 'footer' | 'main' | 'section'; } function generator({ suffixCls, tagName }: GeneratorProps) { return (BasicComponent: 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}; } }; }; } const Basic = (props: BasicPropsWithTagName) => { const { prefixCls, className, children, tagName, ...others } = props; const classString = classNames(className, prefixCls); return React.createElement(tagName, { className: classString, ...others }, children); }; interface BasicLayoutState { siders: string[]; } class BasicLayout extends React.Component { state = { siders: [] }; getSiderHook() { return { 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: Tag, ...others } = this.props; const classString = classNames(className, prefixCls, { [`${prefixCls}-has-sider`]: typeof hasSider === 'boolean' ? 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;