// deps-lint-skip-all import { TinyColor } from '@ctrl/tinycolor'; import { CSSObject } from '@ant-design/cssinjs'; import { resetComponent, GenerateStyle, FullToken, genComponentStyleHook, mergeToken, } from '../../_util/theme'; interface SwitchToken extends FullToken<'Switch'> { switchMinWidth: number; switchHeight: number; switchDuration: string; switchColor: string; switchDisabledOpacity: number; switchInnerMarginMin: number; switchInnerMarginMax: number; switchPadding: number; switchPinSize: number; switchBg: string; switchShadowColor: string; switchMinWidthSM: number; switchHeightSM: number; switchInnerMarginMinSM: number; switchInnerMarginMaxSM: number; switchPinSizeSM: number; } const genSwitchSmallStyle: GenerateStyle = token => ({ [`&${token.componentCls}-small`]: { minWidth: token.switchMinWidthSM, height: token.switchHeightSM, lineHeight: `${token.switchHeightSM}px`, [`${token.componentCls}-inner`]: { marginInlineStart: token.switchInnerMarginMaxSM, marginInlineEnd: token.switchInnerMarginMinSM, fontSize: token.fontSizeSM, }, [`${token.componentCls}-handle`]: { width: token.switchPinSizeSM, height: token.switchPinSizeSM, }, [`${token.componentCls}-loading-icon`]: { top: (token.switchPinSizeSM - 9) / 2, // FIXME fontSize: 9, }, [`&${token.componentCls}-checked`]: { [`${token.componentCls}-inner`]: { marginInlineStart: token.switchInnerMarginMinSM, marginInlineEnd: token.switchInnerMarginMaxSM, }, [`${token.componentCls}-handle`]: { insetInlineStart: `calc(100% - ${token.switchPinSizeSM + token.switchPadding}px)`, }, }, }, }); const genSwitchLoadingStyle: GenerateStyle = token => ({ [`${token.componentCls}-loading-icon${token.iconCls}`]: { position: 'relative', top: (token.switchPinSize - token.fontSize) / 2, color: 'rgba(0, 0, 0, 0.65)', verticalAlign: 'top', }, [`&${token.componentCls}-checked ${token.componentCls}-loading-icon`]: { color: token.switchColor, }, }); const genSwitchHandleStyle: GenerateStyle = token => { const switchHandleCls = `${token.componentCls}-handle`; return { [switchHandleCls]: { position: 'absolute', top: token.switchPadding, insetInlineStart: token.switchPadding, width: token.switchPinSize, height: token.switchPinSize, transition: `all ${token.switchDuration} ease-in-out`, '&::before': { position: 'absolute', top: 0, insetInlineEnd: 0, bottom: 0, insetInlineStart: 0, backgroundColor: token.switchBg, borderRadius: token.switchPinSize / 2, // FIXME boxShadow: `0 2px 4px 0 ${token.switchShadowColor}`, transition: `all ${token.switchDuration} ease-in-out`, content: '""', }, }, [`&${token.componentCls}-checked ${switchHandleCls}`]: { insetInlineStart: `calc(100% - ${token.switchPinSize + token.switchPadding}px)`, }, [`&:not(${token.componentCls}-disabled):active`]: { [`${switchHandleCls}::before`]: { // FIXME insetInlineEnd: '-30%', insetInlineStart: 0, }, [`&${token.componentCls}-checked ${switchHandleCls}::before`]: { insetInlineEnd: 0, // FIXME insetInlineStart: '-30%', }, }, }; }; const genSwitchInnerStyle: GenerateStyle = token => { const switchInnerCls = `${token.componentCls}-inner`; return { [switchInnerCls]: { display: 'block', marginInlineEnd: token.switchInnerMarginMin, marginInlineStart: token.switchInnerMarginMax, color: token.colorTextLightSolid, fontSize: token.fontSizeSM, transition: `margin-inline-end ${token.switchDuration}, margin-inline-start ${token.switchDuration}`, }, [`&${token.componentCls}-checked ${switchInnerCls}`]: { marginInlineEnd: token.switchInnerMarginMax, marginInlineStart: token.switchInnerMarginMin, }, }; }; const genSwitchStyle = (token: SwitchToken): CSSObject => { const { componentCls } = token; return { [componentCls]: { ...resetComponent(token), position: 'relative', display: 'inline-block', boxSizing: 'border-box', minWidth: token.switchMinWidth, height: token.switchHeight, lineHeight: `${token.switchHeight}px`, verticalAlign: 'middle', backgroundImage: `linear-gradient(to right, ${token.colorTextDisabled}, ${token.colorTextDisabled}), linear-gradient(to right, #fff, #fff)`, border: '0', borderRadius: 100, cursor: 'pointer', transition: `all ${token.switchDuration}`, userSelect: 'none', '&:focus-visible': { outline: 0, boxShadow: `0 0 0 ${token.controlOutlineWidth}px ${token.colorDefaultOutline}`, }, [`&${token.componentCls}-checked:focus-visible`]: { boxShadow: `0 0 0 ${token.controlOutlineWidth}px ${token.colorPrimaryOutline}`, }, '&:focus:hover': { boxShadow: 'none', }, [`&${token.componentCls}-checked`]: { background: token.switchColor, }, [`&${token.componentCls}-loading, &${token.componentCls}-disabled`]: { cursor: 'not-allowed', opacity: token.switchDisabledOpacity, '*': { boxShadow: 'none', cursor: 'not-allowed', }, }, // inner style ...genSwitchInnerStyle(token), // handle style ...genSwitchHandleStyle(token), // loading style ...genSwitchLoadingStyle(token), // small style ...genSwitchSmallStyle(token), // rtl style [`&${token.componentCls}-rtl`]: { direction: 'rtl', }, }, }; }; // ============================== Export ============================== export default genComponentStyleHook('Switch', token => { const switchHeight = 22; const switchHeightSM = 16; const switchToken = mergeToken(token, { // FIXME: missing token switchMinWidth: 44, switchHeight, switchDuration: '0.2s', switchColor: token.colorPrimary, switchDisabledOpacity: 0.4, switchInnerMarginMin: Math.ceil(switchHeight * 0.3), switchInnerMarginMax: Math.ceil(switchHeight * 1.1), switchPadding: 2, switchPinSize: switchHeight - 4, switchBg: token.colorBgComponent, switchShadowColor: new TinyColor('#00230b').setAlpha(0.2).toRgbString(), switchMinWidthSM: 28, switchHeightSM, switchInnerMarginMinSM: Math.ceil(switchHeight * 0.3), switchInnerMarginMaxSM: Math.ceil(switchHeight * 1.1), switchPinSizeSM: switchHeightSM - 4, }); return [genSwitchStyle(switchToken)]; });