import classNames from 'classnames'; import * as React from 'react'; import { ConfigContext } from '../config-provider'; import useStyle from './style'; export interface GeneratorProps { suffixCls: string; tagName: 'header' | 'footer' | 'main' | 'section'; displayName: string; } 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, displayName }: GeneratorProps) { return (BasicComponent: any) => { const Adapter = React.forwardRef((props, ref) => { const { getPrefixCls } = React.useContext(ConfigContext); const { prefixCls: customizePrefixCls } = props; const prefixCls = getPrefixCls(suffixCls, customizePrefixCls); return ; }); if (process.env.NODE_ENV !== 'production') { Adapter.displayName = displayName; } return Adapter; }; } const Basic = React.forwardRef((props, ref) => { const { prefixCls, className, children, tagName, ...others } = props; const classString = classNames(prefixCls, className); return React.createElement(tagName, { className: classString, ...others, ref }, children); }); const BasicLayout = React.forwardRef((props, ref) => { const { direction } = React.useContext(ConfigContext); const [siders, setSiders] = React.useState([]); const { prefixCls, className, children, hasSider, tagName: Tag, ...others } = props; const [wrapSSR, hashId] = useStyle(prefixCls as string); const classString = classNames( prefixCls, { [`${prefixCls}-has-sider`]: typeof hasSider === 'boolean' ? hasSider : siders.length > 0, [`${prefixCls}-rtl`]: direction === 'rtl', }, className, hashId, ); const contextValue = React.useMemo( () => ({ siderHook: { addSider: (id: string) => { setSiders(prev => [...prev, id]); }, removeSider: (id: string) => { setSiders(prev => prev.filter(currentId => currentId !== id)); }, }, }), [], ); return wrapSSR( {children} , ); }); const Layout = generator({ suffixCls: 'layout', tagName: 'section', displayName: 'Layout', })(BasicLayout); const Header = generator({ suffixCls: 'layout-header', tagName: 'header', displayName: 'Header', })(Basic); const Footer = generator({ suffixCls: 'layout-footer', tagName: 'footer', displayName: 'Footer', })(Basic); const Content = generator({ suffixCls: 'layout-content', tagName: 'main', displayName: 'Content', })(Basic); export { Header, Footer, Content }; export default Layout;