mirror of
https://github.com/ant-design/ant-design.git
synced 2025-01-22 17:13:27 +08:00
549 lines
14 KiB
TypeScript
549 lines
14 KiB
TypeScript
import { Keyframes } from '@ant-design/cssinjs';
|
|
import type { FullToken, GenerateStyle } from '../../theme/internal';
|
|
import { genComponentStyleHook, mergeToken } from '../../theme/internal';
|
|
import { genFocusOutline, resetComponent } from '../../style';
|
|
|
|
// ============================== Tokens ==============================
|
|
export interface ComponentToken {}
|
|
|
|
interface RadioToken extends FullToken<'Radio'> {
|
|
radioFocusShadow: string;
|
|
radioButtonFocusShadow: string;
|
|
|
|
radioSize: number;
|
|
radioTop: number;
|
|
radioDotSize: number;
|
|
radioDotDisabledSize: number;
|
|
radioCheckedColor: string;
|
|
radioDotDisabledColor: string;
|
|
radioSolidCheckedColor: string;
|
|
|
|
radioButtonBg: string;
|
|
radioButtonCheckedBg: string;
|
|
radioButtonColor: string;
|
|
radioButtonHoverColor: string;
|
|
radioButtonActiveColor: string;
|
|
radioButtonPaddingHorizontal: number;
|
|
radioDisabledButtonCheckedBg: string;
|
|
radioDisabledButtonCheckedColor: string;
|
|
radioWrapperMarginRight: number;
|
|
}
|
|
|
|
// ============================== Styles ==============================
|
|
const antRadioEffect = new Keyframes('antRadioEffect', {
|
|
'0%': { transform: 'scale(1)', opacity: 0.5 },
|
|
'100%': { transform: 'scale(1.6)', opacity: 0 },
|
|
});
|
|
|
|
// styles from RadioGroup only
|
|
const getGroupRadioStyle: GenerateStyle<RadioToken> = (token) => {
|
|
const { componentCls, antCls } = token;
|
|
const groupPrefixCls = `${componentCls}-group`;
|
|
|
|
return {
|
|
[groupPrefixCls]: {
|
|
...resetComponent(token),
|
|
display: 'inline-block',
|
|
fontSize: 0,
|
|
|
|
// RTL
|
|
'&&-rtl': {
|
|
direction: 'rtl',
|
|
},
|
|
|
|
[`${antCls}-badge ${antCls}-badge-count`]: {
|
|
zIndex: 1,
|
|
},
|
|
|
|
[`> ${antCls}-badge:not(:first-child) > ${antCls}-button-wrapper`]: {
|
|
borderInlineStart: 'none',
|
|
},
|
|
},
|
|
};
|
|
};
|
|
|
|
// Styles from radio-wrapper
|
|
const getRadioBasicStyle: GenerateStyle<RadioToken> = (token) => {
|
|
const {
|
|
componentCls,
|
|
radioWrapperMarginRight,
|
|
radioCheckedColor,
|
|
radioTop,
|
|
radioSize,
|
|
motionDurationSlow,
|
|
motionDurationMid,
|
|
motionEaseInOut,
|
|
motionEaseInOutCirc,
|
|
radioButtonBg,
|
|
colorBorder,
|
|
lineWidth,
|
|
radioDotSize,
|
|
colorBgContainerDisabled,
|
|
colorTextDisabled,
|
|
paddingXS,
|
|
radioDotDisabledColor,
|
|
lineType,
|
|
radioDotDisabledSize,
|
|
wireframe,
|
|
colorWhite,
|
|
} = token;
|
|
const radioInnerPrefixCls = `${componentCls}-inner`;
|
|
|
|
return {
|
|
[`${componentCls}-wrapper`]: {
|
|
...resetComponent(token),
|
|
position: 'relative',
|
|
display: 'inline-flex',
|
|
alignItems: 'baseline',
|
|
marginInlineStart: 0,
|
|
marginInlineEnd: radioWrapperMarginRight,
|
|
cursor: 'pointer',
|
|
|
|
// RTL
|
|
'&&-rtl': {
|
|
direction: 'rtl',
|
|
},
|
|
|
|
'&-disabled': {
|
|
cursor: 'not-allowed',
|
|
},
|
|
|
|
'&::after': {
|
|
display: 'inline-block',
|
|
width: 0,
|
|
overflow: 'hidden',
|
|
content: '"\\a0"',
|
|
},
|
|
|
|
// hashId 在 wrapper 上,只能铺平
|
|
[`${componentCls}-checked::after`]: {
|
|
position: 'absolute',
|
|
insetBlockStart: 0,
|
|
insetInlineStart: 0,
|
|
width: '100%',
|
|
height: '100%',
|
|
border: `${lineWidth}px ${lineType} ${radioCheckedColor}`,
|
|
borderRadius: '50%',
|
|
visibility: 'hidden',
|
|
animationName: antRadioEffect,
|
|
animationDuration: motionDurationSlow,
|
|
animationTimingFunction: motionEaseInOut,
|
|
animationFillMode: 'both',
|
|
content: '""',
|
|
},
|
|
|
|
[componentCls]: {
|
|
...resetComponent(token),
|
|
position: 'relative',
|
|
insetBlockStart: radioTop,
|
|
display: 'inline-block',
|
|
outline: 'none',
|
|
cursor: 'pointer',
|
|
},
|
|
|
|
[`${componentCls}-wrapper:hover &,
|
|
&:hover ${radioInnerPrefixCls}`]: {
|
|
borderColor: radioCheckedColor,
|
|
},
|
|
|
|
[`${componentCls}-input:focus-visible + ${radioInnerPrefixCls}`]: {
|
|
...genFocusOutline(token),
|
|
},
|
|
|
|
[`${componentCls}:hover::after, ${componentCls}-wrapper:hover &::after`]: {
|
|
visibility: 'visible',
|
|
},
|
|
|
|
[`${componentCls}-inner`]: {
|
|
'&::after': {
|
|
boxSizing: 'border-box',
|
|
position: 'absolute',
|
|
insetBlockStart: '50%',
|
|
insetInlineStart: '50%',
|
|
display: 'block',
|
|
width: radioSize,
|
|
height: radioSize,
|
|
marginBlockStart: radioSize / -2,
|
|
marginInlineStart: radioSize / -2,
|
|
backgroundColor: wireframe ? radioCheckedColor : colorWhite,
|
|
borderBlockStart: 0,
|
|
borderInlineStart: 0,
|
|
borderRadius: radioSize,
|
|
transform: 'scale(0)',
|
|
opacity: 0,
|
|
transition: `all ${motionDurationSlow} ${motionEaseInOutCirc}`,
|
|
content: '""',
|
|
},
|
|
|
|
boxSizing: 'border-box',
|
|
position: 'relative',
|
|
insetBlockStart: 0,
|
|
insetInlineStart: 0,
|
|
display: 'block',
|
|
width: radioSize,
|
|
height: radioSize,
|
|
backgroundColor: radioButtonBg,
|
|
borderColor: colorBorder,
|
|
borderStyle: 'solid',
|
|
borderWidth: lineWidth,
|
|
borderRadius: '50%',
|
|
transition: `all ${motionDurationMid}`,
|
|
},
|
|
|
|
[`${componentCls}-input`]: {
|
|
position: 'absolute',
|
|
insetBlockStart: 0,
|
|
insetInlineEnd: 0,
|
|
insetBlockEnd: 0,
|
|
insetInlineStart: 0,
|
|
zIndex: 1,
|
|
cursor: 'pointer',
|
|
opacity: 0,
|
|
},
|
|
|
|
// 选中状态
|
|
[`${componentCls}-checked`]: {
|
|
[radioInnerPrefixCls]: {
|
|
borderColor: radioCheckedColor,
|
|
backgroundColor: wireframe ? radioButtonBg : radioCheckedColor,
|
|
|
|
'&::after': {
|
|
transform: `scale(${radioDotSize / radioSize})`,
|
|
opacity: 1,
|
|
transition: `all ${motionDurationSlow} ${motionEaseInOutCirc}`,
|
|
},
|
|
},
|
|
},
|
|
|
|
[`${componentCls}-disabled`]: {
|
|
cursor: 'not-allowed',
|
|
|
|
[radioInnerPrefixCls]: {
|
|
backgroundColor: colorBgContainerDisabled,
|
|
borderColor: colorBorder,
|
|
cursor: 'not-allowed',
|
|
|
|
'&::after': {
|
|
backgroundColor: radioDotDisabledColor,
|
|
},
|
|
},
|
|
|
|
'&-input': {
|
|
cursor: 'not-allowed',
|
|
},
|
|
|
|
[`${componentCls}-disabled + span`]: {
|
|
color: colorTextDisabled,
|
|
cursor: 'not-allowed',
|
|
},
|
|
|
|
[`&${componentCls}-checked`]: {
|
|
[radioInnerPrefixCls]: {
|
|
'&::after': {
|
|
transform: `scale(${radioDotDisabledSize / radioSize})`,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
|
|
[`span${componentCls} + *`]: {
|
|
paddingInlineStart: paddingXS,
|
|
paddingInlineEnd: paddingXS,
|
|
},
|
|
},
|
|
};
|
|
};
|
|
|
|
// Styles from radio-button
|
|
const getRadioButtonStyle: GenerateStyle<RadioToken> = (token) => {
|
|
const {
|
|
radioButtonColor,
|
|
controlHeight,
|
|
componentCls,
|
|
lineWidth,
|
|
lineType,
|
|
colorBorder,
|
|
motionDurationSlow,
|
|
motionDurationMid,
|
|
radioButtonPaddingHorizontal,
|
|
fontSize,
|
|
radioButtonBg,
|
|
fontSizeLG,
|
|
controlHeightLG,
|
|
controlHeightSM,
|
|
paddingXS,
|
|
borderRadius,
|
|
borderRadiusSM,
|
|
borderRadiusLG,
|
|
radioCheckedColor,
|
|
radioButtonCheckedBg,
|
|
radioButtonHoverColor,
|
|
radioButtonActiveColor,
|
|
radioSolidCheckedColor,
|
|
colorTextDisabled,
|
|
colorBgContainerDisabled,
|
|
radioDisabledButtonCheckedColor,
|
|
radioDisabledButtonCheckedBg,
|
|
} = token;
|
|
return {
|
|
[`${componentCls}-button-wrapper`]: {
|
|
position: 'relative',
|
|
display: 'inline-block',
|
|
height: controlHeight,
|
|
margin: 0,
|
|
paddingInline: radioButtonPaddingHorizontal,
|
|
paddingBlock: 0,
|
|
color: radioButtonColor,
|
|
fontSize,
|
|
lineHeight: `${controlHeight - lineWidth * 2}px`,
|
|
background: radioButtonBg,
|
|
border: `${lineWidth}px ${lineType} ${colorBorder}`,
|
|
// strange align fix for chrome but works
|
|
// https://gw.alipayobjects.com/zos/rmsportal/VFTfKXJuogBAXcvfAUWJ.gif
|
|
borderBlockStartWidth: lineWidth + 0.02,
|
|
borderInlineStartWidth: 0,
|
|
borderInlineEndWidth: lineWidth,
|
|
cursor: 'pointer',
|
|
transition: [
|
|
`color ${motionDurationMid}`,
|
|
`background ${motionDurationMid}`,
|
|
`border-color ${motionDurationMid}`,
|
|
`box-shadow ${motionDurationMid}`,
|
|
].join(','),
|
|
|
|
a: {
|
|
color: radioButtonColor,
|
|
},
|
|
|
|
[`> ${componentCls}-button`]: {
|
|
position: 'absolute',
|
|
insetBlockStart: 0,
|
|
insetInlineStart: 0,
|
|
zIndex: -1,
|
|
width: '100%',
|
|
height: '100%',
|
|
},
|
|
|
|
'&:not(:first-child)': {
|
|
'&::before': {
|
|
position: 'absolute',
|
|
insetBlockStart: -lineWidth,
|
|
insetInlineStart: -lineWidth,
|
|
display: 'block',
|
|
boxSizing: 'content-box',
|
|
width: 1,
|
|
height: '100%',
|
|
paddingBlock: lineWidth,
|
|
paddingInline: 0,
|
|
backgroundColor: colorBorder,
|
|
transition: `background-color ${motionDurationSlow}`,
|
|
content: '""',
|
|
},
|
|
},
|
|
|
|
'&:first-child': {
|
|
borderInlineStart: `${lineWidth}px ${lineType} ${colorBorder}`,
|
|
borderStartStartRadius: borderRadius,
|
|
borderEndStartRadius: borderRadius,
|
|
},
|
|
|
|
'&:last-child': {
|
|
borderStartEndRadius: borderRadius,
|
|
borderEndEndRadius: borderRadius,
|
|
},
|
|
|
|
'&:first-child:last-child': {
|
|
borderRadius,
|
|
},
|
|
|
|
[`${componentCls}-group-large &`]: {
|
|
height: controlHeightLG,
|
|
fontSize: fontSizeLG,
|
|
lineHeight: `${controlHeightLG - lineWidth * 2}px`,
|
|
|
|
'&:first-child': {
|
|
borderStartStartRadius: borderRadiusLG,
|
|
borderEndStartRadius: borderRadiusLG,
|
|
},
|
|
|
|
'&:last-child': {
|
|
borderStartEndRadius: borderRadiusLG,
|
|
borderEndEndRadius: borderRadiusLG,
|
|
},
|
|
},
|
|
|
|
[`${componentCls}-group-small &`]: {
|
|
height: controlHeightSM,
|
|
paddingInline: paddingXS - lineWidth,
|
|
paddingBlock: 0,
|
|
lineHeight: `${controlHeightSM - lineWidth * 2}px`,
|
|
|
|
'&:first-child': {
|
|
borderStartStartRadius: borderRadiusSM,
|
|
borderEndStartRadius: borderRadiusSM,
|
|
},
|
|
|
|
'&:last-child': {
|
|
borderStartEndRadius: borderRadiusSM,
|
|
borderEndEndRadius: borderRadiusSM,
|
|
},
|
|
},
|
|
|
|
'&:hover': {
|
|
position: 'relative',
|
|
color: radioCheckedColor,
|
|
},
|
|
|
|
'&:has(:focus-visible)': {
|
|
...genFocusOutline(token),
|
|
},
|
|
|
|
[`${componentCls}-inner, input[type='checkbox'], input[type='radio']`]: {
|
|
width: 0,
|
|
height: 0,
|
|
opacity: 0,
|
|
pointerEvents: 'none',
|
|
},
|
|
|
|
'&-checked:not(&-disabled)': {
|
|
zIndex: 1,
|
|
color: radioCheckedColor,
|
|
background: radioButtonCheckedBg,
|
|
borderColor: radioCheckedColor,
|
|
|
|
'&::before': {
|
|
backgroundColor: radioCheckedColor,
|
|
},
|
|
|
|
'&:first-child': {
|
|
borderColor: radioCheckedColor,
|
|
},
|
|
|
|
'&:hover': {
|
|
color: radioButtonHoverColor,
|
|
borderColor: radioButtonHoverColor,
|
|
|
|
'&::before': {
|
|
backgroundColor: radioButtonHoverColor,
|
|
},
|
|
},
|
|
|
|
'&:active': {
|
|
color: radioButtonActiveColor,
|
|
borderColor: radioButtonActiveColor,
|
|
|
|
'&::before': {
|
|
backgroundColor: radioButtonActiveColor,
|
|
},
|
|
},
|
|
},
|
|
|
|
[`${componentCls}-group-solid &-checked:not(&-disabled)`]: {
|
|
color: radioSolidCheckedColor,
|
|
background: radioCheckedColor,
|
|
borderColor: radioCheckedColor,
|
|
|
|
'&:hover': {
|
|
color: radioSolidCheckedColor,
|
|
background: radioButtonHoverColor,
|
|
borderColor: radioButtonHoverColor,
|
|
},
|
|
|
|
'&:active': {
|
|
color: radioSolidCheckedColor,
|
|
background: radioButtonActiveColor,
|
|
borderColor: radioButtonActiveColor,
|
|
},
|
|
},
|
|
|
|
'&-disabled': {
|
|
color: colorTextDisabled,
|
|
backgroundColor: colorBgContainerDisabled,
|
|
borderColor: colorBorder,
|
|
cursor: 'not-allowed',
|
|
|
|
'&:first-child, &:hover': {
|
|
color: colorTextDisabled,
|
|
backgroundColor: colorBgContainerDisabled,
|
|
borderColor: colorBorder,
|
|
},
|
|
},
|
|
|
|
'&-disabled&-checked': {
|
|
color: radioDisabledButtonCheckedColor,
|
|
backgroundColor: radioDisabledButtonCheckedBg,
|
|
borderColor: colorBorder,
|
|
boxShadow: 'none',
|
|
},
|
|
},
|
|
};
|
|
};
|
|
|
|
// ============================== Export ==============================
|
|
export default genComponentStyleHook('Radio', (token) => {
|
|
const {
|
|
padding,
|
|
lineWidth,
|
|
controlItemBgActiveDisabled,
|
|
colorTextDisabled,
|
|
colorBgContainer,
|
|
fontSize,
|
|
lineHeight,
|
|
fontSizeLG,
|
|
controlOutline,
|
|
colorPrimaryHover,
|
|
colorPrimaryActive,
|
|
colorText,
|
|
colorPrimary,
|
|
marginXS,
|
|
controlOutlineWidth,
|
|
wireframe,
|
|
} = token;
|
|
|
|
// Radio
|
|
const radioFocusShadow = `0 0 0 ${controlOutlineWidth}px ${controlOutline}`;
|
|
const radioButtonFocusShadow = radioFocusShadow;
|
|
|
|
const radioSize = fontSizeLG;
|
|
const radioTop = (Math.round(fontSize * lineHeight) - radioSize) / 2;
|
|
const dotPadding = 4; // Fixed value
|
|
const radioDotDisabledSize = radioSize - dotPadding * 2;
|
|
const radioDotSize = wireframe ? radioDotDisabledSize : radioSize - (dotPadding + lineWidth) * 2;
|
|
const radioCheckedColor = colorPrimary;
|
|
|
|
// Radio buttons
|
|
const radioButtonColor = colorText;
|
|
const radioButtonHoverColor = colorPrimaryHover;
|
|
const radioButtonActiveColor = colorPrimaryActive;
|
|
const radioButtonPaddingHorizontal = padding - lineWidth;
|
|
const radioDisabledButtonCheckedColor = colorTextDisabled;
|
|
const radioWrapperMarginRight = marginXS;
|
|
|
|
const radioToken = mergeToken<RadioToken>(token, {
|
|
radioFocusShadow,
|
|
radioButtonFocusShadow,
|
|
radioSize,
|
|
radioTop,
|
|
radioDotSize,
|
|
radioDotDisabledSize,
|
|
radioCheckedColor,
|
|
radioDotDisabledColor: colorTextDisabled,
|
|
radioSolidCheckedColor: colorBgContainer,
|
|
radioButtonBg: colorBgContainer,
|
|
radioButtonCheckedBg: colorBgContainer,
|
|
radioButtonColor,
|
|
radioButtonHoverColor,
|
|
radioButtonActiveColor,
|
|
radioButtonPaddingHorizontal,
|
|
radioDisabledButtonCheckedBg: controlItemBgActiveDisabled,
|
|
radioDisabledButtonCheckedColor,
|
|
radioWrapperMarginRight,
|
|
});
|
|
|
|
return [
|
|
getGroupRadioStyle(radioToken),
|
|
getRadioBasicStyle(radioToken),
|
|
getRadioButtonStyle(radioToken),
|
|
];
|
|
});
|