mirror of
https://github.com/ant-design/ant-design.git
synced 2024-11-25 11:40:04 +08:00
8f830ec366
React component state updates using setState may asynchronously update this.props and this.state, thus it is not safe to use either of the two when calculating the new state passed to setState, and instead the callback-based variant should be used instead.
104 lines
2.5 KiB
TypeScript
104 lines
2.5 KiB
TypeScript
import * as React from 'react';
|
|
import * as PropTypes from 'prop-types';
|
|
import classNames from 'classnames';
|
|
import { SiderProps } from './Sider';
|
|
|
|
export interface BasicProps extends React.HTMLAttributes<HTMLDivElement> {
|
|
prefixCls?: string;
|
|
hasSider?: boolean;
|
|
}
|
|
|
|
function generator(props: BasicProps) {
|
|
return (BasicComponent: React.ComponentClass<BasicProps>): any => {
|
|
return class Adapter extends React.Component<BasicProps, any> {
|
|
static Header: any;
|
|
static Footer: any;
|
|
static Content: any;
|
|
static Sider: any;
|
|
render() {
|
|
const { prefixCls } = props;
|
|
return <BasicComponent prefixCls={prefixCls} {...this.props} />;
|
|
}
|
|
};
|
|
};
|
|
}
|
|
|
|
class Basic extends React.Component<BasicProps, any> {
|
|
render() {
|
|
const { prefixCls, className, children, ...others } = this.props;
|
|
const divCls = classNames(className, prefixCls);
|
|
return (
|
|
<div className={divCls} {...others}>
|
|
{children}
|
|
</div>
|
|
);
|
|
}
|
|
}
|
|
|
|
interface BasicLayoutState {
|
|
siders: string[];
|
|
}
|
|
|
|
class BasicLayout extends React.Component<BasicProps, BasicLayoutState> {
|
|
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, ...others } = this.props;
|
|
const divCls = classNames(className, prefixCls, {
|
|
[`${prefixCls}-has-sider`]: hasSider || this.state.siders.length > 0,
|
|
});
|
|
return (
|
|
<div className={divCls} {...others}>
|
|
{children}
|
|
</div>
|
|
);
|
|
}
|
|
}
|
|
|
|
const Layout: React.ComponentClass<BasicProps> & {
|
|
Header: React.ComponentClass<BasicProps>;
|
|
Footer: React.ComponentClass<BasicProps>;
|
|
Content: React.ComponentClass<BasicProps>;
|
|
Sider: React.ComponentClass<SiderProps>;
|
|
} = generator({
|
|
prefixCls: 'ant-layout',
|
|
})(BasicLayout);
|
|
|
|
const Header = generator({
|
|
prefixCls: 'ant-layout-header',
|
|
})(Basic);
|
|
|
|
const Footer = generator({
|
|
prefixCls: 'ant-layout-footer',
|
|
})(Basic);
|
|
|
|
const Content = generator({
|
|
prefixCls: 'ant-layout-content',
|
|
})(Basic);
|
|
|
|
Layout.Header = Header;
|
|
Layout.Footer = Footer;
|
|
Layout.Content = Content;
|
|
|
|
export default Layout;
|