/* eslint-disable no-redeclare */ import type { ComponentType, FC, ReactElement } from 'react'; import React, { useContext } from 'react'; import type { CSSInterpolation } from '@ant-design/cssinjs'; import { token2CSSVar, useCSSVarRegister, useStyleRegister } from '@ant-design/cssinjs'; import classNames from 'classnames'; import { warning } from 'rc-util'; import { ConfigContext } from '../../config-provider/context'; import { genCommonStyle, genLinkStyle } from '../../style'; import type { ComponentTokenMap, GlobalToken, OverrideToken, UseComponentStyleResult, } from '../interface'; import useToken, { ignore, unitless } from '../useToken'; import genCalc from './calc'; import type AbstractCalculator from './calc/calculator'; import genMaxMin from './maxmin'; import statisticToken, { merge as mergeToken } from './statistic'; import useResetIconStyle from './useResetIconStyle'; export type OverrideTokenWithoutDerivative = ComponentTokenMap; export type OverrideComponent = keyof OverrideTokenWithoutDerivative; export type GlobalTokenWithComponent = GlobalToken & ComponentTokenMap[C]; type ComponentToken = Exclude; type ComponentTokenKey = keyof ComponentToken; export interface StyleInfo { hashId: string; prefixCls: string; rootPrefixCls: string; iconPrefixCls: string; } export type CSSUtil = { calc: (number: any) => AbstractCalculator; max: (...values: (number | string)[]) => number | string; min: (...values: (number | string)[]) => number | string; }; export type TokenWithCommonCls = T & { /** Wrap component class with `.` prefix */ componentCls: string; /** Origin prefix which do not have `.` prefix */ prefixCls: string; /** Wrap icon class with `.` prefix */ iconCls: string; /** Wrap ant prefixCls class with `.` prefix */ antCls: string; } & CSSUtil; export type FullToken = TokenWithCommonCls< GlobalTokenWithComponent >; export type GenStyleFn = ( token: FullToken, info: StyleInfo, ) => CSSInterpolation; export type GetDefaultToken = | null | OverrideTokenWithoutDerivative[C] | ((token: GlobalToken) => OverrideTokenWithoutDerivative[C]); export type FormatComponentToken = ( token: NonNullable, ) => NonNullable; const getDefaultComponentToken = ( component: C, token: GlobalToken, getDefaultToken: GetDefaultToken, ) => { if (typeof getDefaultToken === 'function') { return getDefaultToken(mergeToken(token, token[component] ?? {})); } return getDefaultToken ?? {}; }; const getComponentToken = ( component: C, token: GlobalToken, defaultToken: OverrideTokenWithoutDerivative[C], options?: { deprecatedTokens?: [ComponentTokenKey, ComponentTokenKey][]; format?: FormatComponentToken; }, ) => { const customToken = { ...(token[component] as ComponentToken) }; if (options?.deprecatedTokens) { const { deprecatedTokens } = options; deprecatedTokens.forEach(([oldTokenKey, newTokenKey]) => { if (process.env.NODE_ENV !== 'production') { warning( !customToken?.[oldTokenKey], `The token '${String(oldTokenKey)}' of ${component} had deprecated, use '${String( newTokenKey, )}' instead.`, ); } // Should wrap with `if` clause, or there will be `undefined` in object. if (customToken?.[oldTokenKey] || customToken?.[newTokenKey]) { customToken[newTokenKey] ??= customToken?.[oldTokenKey]; } }); } let mergedToken: any = { ...defaultToken, ...customToken }; if (options?.format) { mergedToken = options.format(mergedToken); } // Remove same value as global token to minimize size Object.keys(mergedToken).forEach((key) => { if (mergedToken[key] === token[key as keyof GlobalToken]) { delete mergedToken[key]; } }); return mergedToken; }; const getCompVarPrefix = (component: string, prefix?: string) => `${[ prefix, component.replace(/([A-Z]+)([A-Z][a-z]+)/g, '$1-$2').replace(/([a-z])([A-Z])/g, '$1-$2'), ] .filter(Boolean) .join('-')}`; export default function genComponentStyleHook( componentName: C | [C, string], styleFn: GenStyleFn, getDefaultToken?: | null | OverrideTokenWithoutDerivative[C] | ((token: GlobalToken) => OverrideTokenWithoutDerivative[C]), options: { resetStyle?: boolean; // Deprecated token key map [["oldTokenKey", "newTokenKey"], ["oldTokenKey", "newTokenKey"]] deprecatedTokens?: [ComponentTokenKey, ComponentTokenKey][]; /** * Only use component style in client side. Ignore in SSR. */ clientOnly?: boolean; /** * Set order of component style. Default is -999. */ order?: number; format?: FormatComponentToken; injectStyle?: boolean; } = {}, ) { const cells = (Array.isArray(componentName) ? componentName : [componentName, componentName]) as [ C, string, ]; const [component] = cells; const concatComponent = cells.join('-'); return (prefixCls: string): UseComponentStyleResult => { const [theme, realToken, hashId, token, cssVar] = useToken(); const { getPrefixCls, iconPrefixCls, csp } = useContext(ConfigContext); const rootPrefixCls = getPrefixCls(); const type = cssVar ? 'css' : 'js'; const calc = genCalc(type); const { max, min } = genMaxMin(type); // Shared config const sharedConfig: Omit[0], 'path'> = { theme, token, hashId, nonce: () => csp?.nonce!, clientOnly: options.clientOnly, // antd is always at top of styles order: options.order || -999, }; // Generate style for all a tags in antd component. useStyleRegister( { ...sharedConfig, clientOnly: false, path: ['Shared', rootPrefixCls] }, () => [ { // Link '&': genLinkStyle(token), }, ], ); // Generate style for icons useResetIconStyle(iconPrefixCls, csp); const wrapSSR = useStyleRegister( { ...sharedConfig, path: [concatComponent, prefixCls, iconPrefixCls] }, () => { if (options.injectStyle === false) { return []; } const { token: proxyToken, flush } = statisticToken(token); const defaultComponentToken = getDefaultComponentToken( component, realToken, getDefaultToken, ); const componentCls = `.${prefixCls}`; const componentToken = getComponentToken(component, realToken, defaultComponentToken, { deprecatedTokens: options.deprecatedTokens, format: options.format, }); if (cssVar) { Object.keys(defaultComponentToken).forEach((key) => { defaultComponentToken[key] = `var(${token2CSSVar( key, getCompVarPrefix(component, cssVar.prefix), )})`; }); } const mergedToken = mergeToken< TokenWithCommonCls> >( proxyToken, { componentCls, prefixCls, iconCls: `.${iconPrefixCls}`, antCls: `.${rootPrefixCls}`, calc, max, min, }, cssVar ? defaultComponentToken : componentToken, ); const styleInterpolation = styleFn(mergedToken as unknown as FullToken, { hashId, prefixCls, rootPrefixCls, iconPrefixCls, }); flush(component, componentToken); return [ options.resetStyle === false ? null : genCommonStyle(mergedToken, prefixCls), styleInterpolation, ]; }, ); return [wrapSSR, classNames(hashId, cssVar?.key)]; }; } export interface SubStyleComponentProps { prefixCls: string; } // Get from second argument type RestParameters = T extends [any, ...infer Rest] ? Rest : never; export const genSubStyleComponent: ( componentName: [C, string], ...args: RestParameters>> ) => ComponentType = (componentName, styleFn, getDefaultToken, options) => { const useStyle = genComponentStyleHook(componentName, styleFn, getDefaultToken, { resetStyle: false, // Sub Style should default after root one order: -998, ...options, }); const StyledComponent: ComponentType = ({ prefixCls, }: SubStyleComponentProps) => { useStyle(prefixCls); return null; }; if (process.env.NODE_ENV !== 'production') { StyledComponent.displayName = `SubStyle_${ Array.isArray(componentName) ? componentName.join('.') : componentName }`; } return StyledComponent; }; export type CSSVarRegisterProps = { rootCls: string; component: string; cssVar: { prefix?: string; key?: string; }; }; const genCSSVarRegister = ( component: C, getDefaultToken?: GetDefaultToken, options?: { unitless?: { [key in ComponentTokenKey]: boolean; }; deprecatedTokens?: [ComponentTokenKey, ComponentTokenKey][]; format?: FormatComponentToken; injectStyle?: boolean; }, ) => { function prefixToken(key: string) { return `${component}${key.slice(0, 1).toUpperCase()}${key.slice(1)}`; } const { unitless: originUnitless = {}, injectStyle = true } = options ?? {}; const compUnitless: any = { [prefixToken('zIndexPopup')]: true, }; Object.keys(originUnitless).forEach((key: keyof ComponentTokenKey) => { compUnitless[prefixToken(key)] = originUnitless[key]; }); const CSSVarRegister: FC = ({ rootCls, cssVar }) => { const [, realToken] = useToken(); useCSSVarRegister( { path: [component], prefix: cssVar.prefix, key: cssVar?.key!, unitless: { ...unitless, ...compUnitless, }, ignore, token: realToken, scope: rootCls, }, () => { const defaultToken = getDefaultComponentToken(component, realToken, getDefaultToken); const componentToken = getComponentToken(component, realToken, defaultToken, { format: options?.format, deprecatedTokens: options?.deprecatedTokens, }); Object.keys(defaultToken).forEach((key) => { componentToken[prefixToken(key)] = componentToken[key]; delete componentToken[key]; }); return componentToken; }, ); return null; }; const useCSSVar = (rootCls: string) => { const [, , , , cssVar] = useToken(); return (node: ReactElement): ReactElement => injectStyle && cssVar ? ( <> {node} ) : ( node ); }; return useCSSVar; }; export const genStyleHooks = ( component: C, styleFn: GenStyleFn, getDefaultToken?: GetDefaultToken, options?: { resetStyle?: boolean; deprecatedTokens?: [ComponentTokenKey, ComponentTokenKey][]; /** * Chance to format component token with user input. * Useful when need calculated token as css variables. */ format?: FormatComponentToken; /** * Component tokens that do not need unit. */ unitless?: { [key in ComponentTokenKey]: boolean; }; /** * Only use component style in client side. Ignore in SSR. */ clientOnly?: boolean; /** * Set order of component style. * @default -999 */ order?: number; /** * Whether generate styles * @default true */ injectStyle?: boolean; }, ) => { const useStyle = genComponentStyleHook(component, styleFn, getDefaultToken, options); const useCSSVar = genCSSVarRegister(component, getDefaultToken, options); return (prefixCls: string, rootCls: string = prefixCls) => { const [, hashId] = useStyle(prefixCls); const wrapCSSVar = useCSSVar(rootCls); return [wrapCSSVar, hashId] as const; }; };