mirror of
https://github.com/ant-design/ant-design.git
synced 2024-12-01 14:59:35 +08:00
a434ec7614
* refactor: steps full token * refactor: progress dot * refactor: no hard code * refactor: token optimization
356 lines
11 KiB
TypeScript
356 lines
11 KiB
TypeScript
// deps-lint-skip-all
|
|
import type { CSSObject } from '@ant-design/cssinjs';
|
|
import type { FullToken, GenerateStyle } from '../../_util/theme';
|
|
import { genComponentStyleHook, mergeToken, resetComponent } from '../../_util/theme';
|
|
import genStepsCustomIconStyle from './custom-icon';
|
|
import genStepsLabelPlacementStyle from './label-placement';
|
|
import genStepsNavStyle from './nav';
|
|
import genStepsProgressStyle from './progress';
|
|
import genStepsProgressDotStyle from './progress-dot';
|
|
import genStepsRTLStyle from './rtl';
|
|
import genStepsSmallStyle from './small';
|
|
import genStepsVerticalStyle from './vertical';
|
|
|
|
export interface ComponentToken {
|
|
descriptionWidth: number;
|
|
}
|
|
|
|
export interface StepsToken extends FullToken<'Steps'> {
|
|
// Steps variable default.less
|
|
processTailColor: string;
|
|
stepsNavArrowColor: string;
|
|
stepsIconSize: number;
|
|
stepsIconCustomSize: number;
|
|
stepsIconCustomTop: number;
|
|
stepsIconCustomFontSize: number;
|
|
stepsIconTop: number;
|
|
stepsIconFontSize: number;
|
|
stepsTitleLineHeight: number;
|
|
stepsSmallIconSize: number;
|
|
stepsDotSize: number;
|
|
stepsCurrentDotSize: number;
|
|
stepsNavContentMaxWidth: string;
|
|
// Steps component less variable
|
|
processIconColor: string;
|
|
processTitleColor: string;
|
|
processDescriptionColor: string;
|
|
processIconTextColor: string;
|
|
waitIconColor: string;
|
|
waitTitleColor: string;
|
|
waitDescriptionColor: string;
|
|
waitTailColor: string;
|
|
finishIconColor: string;
|
|
finishTitleColor: string;
|
|
finishDescriptionColor: string;
|
|
finishTailColor: string;
|
|
errorIconColor: string;
|
|
errorTitleColor: string;
|
|
errorDescriptionColor: string;
|
|
errorTailColor: string;
|
|
stepsNavActiveColor: string;
|
|
stepsProgressSize: number;
|
|
}
|
|
|
|
enum StepItemStatusEnum {
|
|
wait = 'wait',
|
|
process = 'process',
|
|
finish = 'finish',
|
|
error = 'error',
|
|
}
|
|
|
|
const genStepsItemStatusStyle = (status: StepItemStatusEnum, token: StepsToken): CSSObject => {
|
|
const prefix = `${token.componentCls}-item`;
|
|
const iconColorKey: keyof StepsToken = `${status}IconColor`;
|
|
const titleColorKey: keyof StepsToken = `${status}TitleColor`;
|
|
const descriptionColorKey: keyof StepsToken = `${status}DescriptionColor`;
|
|
const tailColorKey: keyof StepsToken = `${status}TailColor`;
|
|
return {
|
|
[`${prefix}-${status} ${prefix}-icon`]: {
|
|
backgroundColor: token.colorBgComponent,
|
|
borderColor: token[iconColorKey],
|
|
[`> ${token.componentCls}-icon`]: {
|
|
color: token[iconColorKey],
|
|
[`${token.componentCls}-icon-dot`]: {
|
|
background: token[iconColorKey],
|
|
},
|
|
},
|
|
},
|
|
[`${prefix}-${status} > ${prefix}-container > ${prefix}-content > ${prefix}-title`]: {
|
|
color: token[titleColorKey],
|
|
|
|
'&::after': {
|
|
backgroundColor: token[tailColorKey],
|
|
},
|
|
},
|
|
[`${prefix}-${status} > ${prefix}-container > ${prefix}-content > ${prefix}-description`]: {
|
|
color: token[descriptionColorKey],
|
|
},
|
|
[`${prefix}-${status} > ${prefix}-container > ${prefix}-tail::after`]: {
|
|
backgroundColor: token[tailColorKey],
|
|
},
|
|
};
|
|
};
|
|
|
|
const genStepsItemStyle: GenerateStyle<StepsToken, CSSObject> = token => {
|
|
const { componentCls, motionDurationSlow } = token;
|
|
const stepsItemCls = `${componentCls}-item`; // .ant-steps-item
|
|
|
|
return {
|
|
[stepsItemCls]: {
|
|
position: 'relative',
|
|
display: 'inline-block',
|
|
flex: 1,
|
|
overflow: 'hidden',
|
|
verticalAlign: 'top',
|
|
'&:last-child': {
|
|
flex: 'none',
|
|
[`> ${stepsItemCls}-container > ${stepsItemCls}-tail, > ${stepsItemCls}-container > ${stepsItemCls}-content > ${stepsItemCls}-title::after`]:
|
|
{
|
|
display: 'none',
|
|
},
|
|
},
|
|
},
|
|
[`${stepsItemCls}-container`]: {
|
|
outline: 'none',
|
|
},
|
|
[`${stepsItemCls}-icon, ${stepsItemCls}-content`]: {
|
|
display: 'inline-block',
|
|
verticalAlign: 'top',
|
|
},
|
|
[`${stepsItemCls}-icon`]: {
|
|
width: token.stepsIconSize,
|
|
height: token.stepsIconSize,
|
|
marginTop: 0,
|
|
marginBottom: 0,
|
|
marginInlineStart: 0,
|
|
marginInlineEnd: token.marginXS,
|
|
fontSize: token.stepsIconFontSize,
|
|
fontFamily: token.fontFamily,
|
|
lineHeight: `${token.stepsIconSize}px`,
|
|
textAlign: 'center',
|
|
border: `${token.controlLineWidth}px ${token.controlLineType} ${token.colorTextDisabled}`,
|
|
borderRadius: token.stepsIconSize,
|
|
transition: `background-color ${motionDurationSlow}, border-color ${motionDurationSlow}`,
|
|
[`${componentCls}-icon`]: {
|
|
position: 'relative',
|
|
top: token.stepsIconTop,
|
|
color: token.colorPrimary,
|
|
lineHeight: 1,
|
|
},
|
|
},
|
|
[`${stepsItemCls}-tail`]: {
|
|
position: 'absolute',
|
|
top: token.marginSM,
|
|
insetInlineStart: 0,
|
|
width: '100%',
|
|
|
|
'&::after': {
|
|
display: 'inline-block',
|
|
width: '100%',
|
|
height: token.lineWidth,
|
|
background: token.colorSplit,
|
|
borderRadius: token.lineWidth,
|
|
transition: `background ${motionDurationSlow}`,
|
|
content: '""',
|
|
},
|
|
},
|
|
[`${stepsItemCls}-title`]: {
|
|
position: 'relative',
|
|
display: 'inline-block',
|
|
paddingInlineEnd: token.padding,
|
|
color: token.colorText,
|
|
fontSize: token.fontSizeLG,
|
|
lineHeight: `${token.stepsTitleLineHeight}px`,
|
|
|
|
'&::after': {
|
|
position: 'absolute',
|
|
top: token.stepsTitleLineHeight / 2,
|
|
insetInlineStart: '100%',
|
|
display: 'block',
|
|
width: 9999,
|
|
height: token.lineWidth,
|
|
background: token.processTailColor,
|
|
content: '""',
|
|
},
|
|
},
|
|
[`${stepsItemCls}-subtitle`]: {
|
|
display: 'inline',
|
|
marginInlineStart: token.marginXS,
|
|
color: token.colorTextSecondary,
|
|
fontWeight: 'normal',
|
|
fontSize: token.fontSizeBase,
|
|
},
|
|
[`${stepsItemCls}-description`]: {
|
|
color: token.colorTextSecondary,
|
|
fontSize: token.fontSizeBase,
|
|
},
|
|
...genStepsItemStatusStyle(StepItemStatusEnum.wait, token),
|
|
...genStepsItemStatusStyle(StepItemStatusEnum.process, token),
|
|
[`${stepsItemCls}-process > ${stepsItemCls}-container > ${stepsItemCls}-icon`]: {
|
|
background: token.processIconColor,
|
|
|
|
[`${componentCls}-icon`]: {
|
|
color: token.processIconTextColor,
|
|
},
|
|
},
|
|
[`${stepsItemCls}-process > ${stepsItemCls}-container > ${stepsItemCls}-title`]: {
|
|
fontWeight: token.fontWeightStrong,
|
|
},
|
|
...genStepsItemStatusStyle(StepItemStatusEnum.finish, token),
|
|
...genStepsItemStatusStyle(StepItemStatusEnum.error, token),
|
|
[`${stepsItemCls}${componentCls}-next-error > ${componentCls}-item-title::after`]: {
|
|
background: token.colorError,
|
|
},
|
|
[`${stepsItemCls}-disabled`]: {
|
|
cursor: 'not-allowed',
|
|
},
|
|
};
|
|
};
|
|
|
|
// ============================= Clickable ===========================
|
|
const genStepsClickableStyle: GenerateStyle<StepsToken, CSSObject> = token => {
|
|
const { componentCls, motionDurationSlow } = token;
|
|
|
|
return {
|
|
[`& ${componentCls}-item`]: {
|
|
[`&:not(${componentCls}-item-active)`]: {
|
|
[`& > ${componentCls}-item-container[role='button']`]: {
|
|
cursor: 'pointer',
|
|
[`${componentCls}-item`]: {
|
|
[`&-title, &-subtitle, &-description, &-icon ${componentCls}-icon`]: {
|
|
transition: `color ${motionDurationSlow}`,
|
|
},
|
|
},
|
|
|
|
'&:hover': {
|
|
[`${componentCls}-item`]: {
|
|
[`&-title, &-subtitle, &-description`]: {
|
|
color: token.colorPrimary,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
|
|
[`&:not(${componentCls}-item-process)`]: {
|
|
[`& > ${componentCls}-item-container[role='button']:hover`]: {
|
|
[`${componentCls}-item`]: {
|
|
'&-icon': {
|
|
borderColor: token.colorPrimary,
|
|
|
|
[`${componentCls}-icon`]: {
|
|
color: token.colorPrimary,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
[`&${componentCls}-horizontal:not(${componentCls}-label-vertical)`]: {
|
|
[`${componentCls}-item`]: {
|
|
paddingInlineStart: token.padding,
|
|
whiteSpace: 'nowrap',
|
|
|
|
'&:first-child': {
|
|
paddingInlineStart: 0,
|
|
},
|
|
[`&:last-child ${componentCls}-item-title`]: {
|
|
paddingInlineEnd: 0,
|
|
},
|
|
'&-tail': {
|
|
display: 'none',
|
|
},
|
|
'&-description': {
|
|
maxWidth: token.descriptionWidth,
|
|
whiteSpace: 'normal',
|
|
},
|
|
},
|
|
},
|
|
};
|
|
};
|
|
|
|
const genStepsStyle: GenerateStyle<StepsToken, CSSObject> = token => {
|
|
const { componentCls } = token; // .ant-steps
|
|
|
|
return {
|
|
[componentCls]: {
|
|
...resetComponent(token),
|
|
display: 'flex',
|
|
width: '100%',
|
|
fontSize: 0,
|
|
textAlign: 'initial',
|
|
// single Item
|
|
...genStepsItemStyle(token),
|
|
// Clickable
|
|
...genStepsClickableStyle(token),
|
|
// custom-icon
|
|
...genStepsCustomIconStyle(token),
|
|
// small
|
|
...genStepsSmallStyle(token),
|
|
// vertical
|
|
...genStepsVerticalStyle(token),
|
|
// label-placement
|
|
...genStepsLabelPlacementStyle(token),
|
|
// progress-dot
|
|
...genStepsProgressDotStyle(token),
|
|
// nav
|
|
...genStepsNavStyle(token),
|
|
// rtl
|
|
...genStepsRTLStyle(token),
|
|
// progress
|
|
...genStepsProgressStyle(token),
|
|
},
|
|
};
|
|
};
|
|
|
|
// ============================== Export ==============================
|
|
export default genComponentStyleHook(
|
|
'Steps',
|
|
token => {
|
|
const stepsIconSize = token.controlHeight;
|
|
const processTailColor = token.colorSplit;
|
|
const processIconColor = token.colorPrimary;
|
|
|
|
const stepsToken = mergeToken<StepsToken>(token, {
|
|
// Steps variable default.less
|
|
processTailColor,
|
|
stepsNavArrowColor: token.colorTextDisabled,
|
|
stepsIconSize,
|
|
stepsIconCustomSize: stepsIconSize,
|
|
stepsIconCustomTop: 0,
|
|
stepsIconCustomFontSize: token.fontSizeHeading3,
|
|
stepsIconTop: -0.5, // magic for ui experience
|
|
stepsIconFontSize: token.fontSizeLG,
|
|
stepsTitleLineHeight: token.controlHeight,
|
|
stepsSmallIconSize: token.fontSizeHeading3,
|
|
stepsDotSize: token.controlHeight / 4,
|
|
stepsCurrentDotSize: token.controlHeightLG / 4,
|
|
stepsNavContentMaxWidth: 'auto',
|
|
// Steps component less variable
|
|
processIconColor,
|
|
processTitleColor: token.colorText,
|
|
processDescriptionColor: token.colorText,
|
|
processIconTextColor: token.colorTextLightSolid,
|
|
waitIconColor: token.colorTextDisabled,
|
|
waitTitleColor: token.colorTextSecondary,
|
|
waitDescriptionColor: token.colorTextSecondary,
|
|
waitTailColor: processTailColor,
|
|
finishIconColor: processIconColor,
|
|
finishTitleColor: token.colorText,
|
|
finishDescriptionColor: token.colorTextSecondary,
|
|
finishTailColor: token.colorPrimary,
|
|
errorIconColor: token.colorError,
|
|
errorTitleColor: token.colorError,
|
|
errorDescriptionColor: token.colorError,
|
|
errorTailColor: processTailColor,
|
|
stepsNavActiveColor: token.colorPrimary,
|
|
stepsProgressSize: token.controlHeightLG,
|
|
});
|
|
|
|
return [genStepsStyle(stepsToken)];
|
|
},
|
|
{
|
|
descriptionWidth: 140,
|
|
},
|
|
);
|