2018-11-26 12:06:42 +08:00
|
|
|
import * as React from 'react';
|
2019-04-16 12:06:04 +08:00
|
|
|
import createReactContext, { Context } from '@ant-design/create-react-context';
|
2018-11-26 12:06:42 +08:00
|
|
|
|
2018-12-26 16:01:00 +08:00
|
|
|
import defaultRenderEmpty, { RenderEmptyHandler } from './renderEmpty';
|
|
|
|
|
|
|
|
export { RenderEmptyHandler };
|
|
|
|
|
2019-01-09 20:15:37 +08:00
|
|
|
export interface CSPConfig {
|
|
|
|
nonce?: string;
|
|
|
|
}
|
|
|
|
|
2018-12-05 19:12:18 +08:00
|
|
|
export interface ConfigConsumerProps {
|
2018-11-26 12:06:42 +08:00
|
|
|
getPopupContainer?: (triggerNode?: HTMLElement) => HTMLElement;
|
2018-12-05 19:12:18 +08:00
|
|
|
rootPrefixCls?: string;
|
|
|
|
getPrefixCls: (suffixCls: string, customizePrefixCls?: string) => string;
|
2018-12-26 16:01:00 +08:00
|
|
|
renderEmpty: RenderEmptyHandler;
|
2019-01-09 20:15:37 +08:00
|
|
|
csp?: CSPConfig;
|
2019-01-10 11:47:11 +08:00
|
|
|
autoInsertSpaceInButton?: boolean;
|
2018-11-26 12:06:42 +08:00
|
|
|
}
|
|
|
|
|
2019-02-19 11:42:05 +08:00
|
|
|
export const configConsumerProps = [
|
|
|
|
'getPopupContainer',
|
|
|
|
'rootPrefixCls',
|
|
|
|
'getPrefixCls',
|
|
|
|
'renderEmpty',
|
|
|
|
'csp',
|
|
|
|
'autoInsertSpaceInButton',
|
|
|
|
];
|
|
|
|
|
2019-03-16 09:48:21 +08:00
|
|
|
export interface ConfigProviderProps {
|
2018-12-05 19:12:18 +08:00
|
|
|
getPopupContainer?: (triggerNode?: HTMLElement) => HTMLElement;
|
|
|
|
prefixCls?: string;
|
|
|
|
children?: React.ReactNode;
|
2018-12-26 16:01:00 +08:00
|
|
|
renderEmpty?: RenderEmptyHandler;
|
2019-01-09 20:15:37 +08:00
|
|
|
csp?: CSPConfig;
|
2019-01-10 11:47:11 +08:00
|
|
|
autoInsertSpaceInButton?: boolean;
|
2018-12-05 19:12:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const ConfigContext: Context<ConfigConsumerProps | null> = createReactContext({
|
2018-12-26 16:01:00 +08:00
|
|
|
// We provide a default function for Context without provider
|
2018-12-05 19:12:18 +08:00
|
|
|
getPrefixCls: (suffixCls: string, customizePrefixCls?: string) => {
|
|
|
|
if (customizePrefixCls) return customizePrefixCls;
|
|
|
|
|
|
|
|
return `ant-${suffixCls}`;
|
|
|
|
},
|
2018-12-26 16:01:00 +08:00
|
|
|
|
|
|
|
renderEmpty: defaultRenderEmpty,
|
2018-12-05 19:12:18 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
export const ConfigConsumer = ConfigContext.Consumer;
|
2018-11-26 12:06:42 +08:00
|
|
|
|
2018-12-05 19:12:18 +08:00
|
|
|
class ConfigProvider extends React.Component<ConfigProviderProps> {
|
|
|
|
getPrefixCls = (suffixCls: string, customizePrefixCls?: string) => {
|
|
|
|
const { prefixCls = 'ant' } = this.props;
|
|
|
|
|
|
|
|
if (customizePrefixCls) return customizePrefixCls;
|
|
|
|
|
|
|
|
return suffixCls ? `${prefixCls}-${suffixCls}` : prefixCls;
|
2018-11-26 12:06:42 +08:00
|
|
|
};
|
|
|
|
|
2018-12-05 19:12:18 +08:00
|
|
|
renderProvider = (context: ConfigConsumerProps) => {
|
2019-01-10 11:47:11 +08:00
|
|
|
const { children, getPopupContainer, renderEmpty, csp, autoInsertSpaceInButton } = this.props;
|
2018-12-05 19:12:18 +08:00
|
|
|
|
|
|
|
const config: ConfigConsumerProps = {
|
|
|
|
...context,
|
|
|
|
getPrefixCls: this.getPrefixCls,
|
2019-01-09 20:15:37 +08:00
|
|
|
csp,
|
2019-01-10 11:47:11 +08:00
|
|
|
autoInsertSpaceInButton,
|
2018-12-05 19:12:18 +08:00
|
|
|
};
|
2018-11-26 12:06:42 +08:00
|
|
|
|
2018-12-26 16:01:00 +08:00
|
|
|
if (getPopupContainer) {
|
|
|
|
config.getPopupContainer = getPopupContainer;
|
|
|
|
}
|
|
|
|
if (renderEmpty) {
|
|
|
|
config.renderEmpty = renderEmpty;
|
|
|
|
}
|
|
|
|
|
2018-12-07 20:02:01 +08:00
|
|
|
return <ConfigContext.Provider value={config}>{children}</ConfigContext.Provider>;
|
|
|
|
};
|
2018-12-05 19:12:18 +08:00
|
|
|
|
|
|
|
render() {
|
2018-12-07 20:02:01 +08:00
|
|
|
return <ConfigConsumer>{this.renderProvider}</ConfigConsumer>;
|
2018-12-05 19:12:18 +08:00
|
|
|
}
|
|
|
|
}
|
2018-11-26 12:06:42 +08:00
|
|
|
|
2019-01-03 17:10:04 +08:00
|
|
|
// =========================== withConfigConsumer ===========================
|
|
|
|
// We need define many types here. So let's put in the block region
|
|
|
|
type IReactComponent<P = any> =
|
|
|
|
| React.StatelessComponent<P>
|
|
|
|
| React.ComponentClass<P>
|
|
|
|
| React.ClassicComponentClass<P>;
|
|
|
|
|
|
|
|
interface BasicExportProps {
|
|
|
|
prefixCls?: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface ConsumerConfig {
|
|
|
|
prefixCls: string;
|
|
|
|
}
|
|
|
|
|
2019-02-19 11:42:05 +08:00
|
|
|
interface ConstructorProps {
|
|
|
|
displayName?: string;
|
|
|
|
}
|
|
|
|
|
2019-01-03 17:10:04 +08:00
|
|
|
export function withConfigConsumer<ExportProps extends BasicExportProps>(config: ConsumerConfig) {
|
2019-02-19 11:42:05 +08:00
|
|
|
return function<ComponentDef>(Component: IReactComponent): React.SFC<ExportProps> & ComponentDef {
|
2019-01-03 17:10:04 +08:00
|
|
|
// Wrap with ConfigConsumer. Since we need compatible with react 15, be care when using ref methods
|
2019-02-19 11:42:05 +08:00
|
|
|
const SFC = ((props: ExportProps) => (
|
2019-01-03 17:10:04 +08:00
|
|
|
<ConfigConsumer>
|
|
|
|
{(configProps: ConfigConsumerProps) => {
|
|
|
|
const { prefixCls: basicPrefixCls } = config;
|
|
|
|
const { getPrefixCls } = configProps;
|
|
|
|
const { prefixCls: customizePrefixCls } = props;
|
|
|
|
const prefixCls = getPrefixCls(basicPrefixCls, customizePrefixCls);
|
|
|
|
return <Component {...configProps} {...props} prefixCls={prefixCls} />;
|
|
|
|
}}
|
|
|
|
</ConfigConsumer>
|
2019-01-09 20:38:09 +08:00
|
|
|
)) as React.SFC<ExportProps> & ComponentDef;
|
2019-02-19 11:42:05 +08:00
|
|
|
|
|
|
|
const cons: ConstructorProps = Component.constructor as ConstructorProps;
|
|
|
|
const name = (cons && cons.displayName) || Component.name || 'Component';
|
|
|
|
|
|
|
|
SFC.displayName = `withConfigConsumer(${name})`;
|
|
|
|
|
|
|
|
return SFC;
|
2019-01-03 17:10:04 +08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-12-07 20:02:01 +08:00
|
|
|
export default ConfigProvider;
|