import * as React from 'react'; import CheckOutlined from '@ant-design/icons/CheckOutlined'; import CloseOutlined from '@ant-design/icons/CloseOutlined'; import classNames from 'classnames'; import RcSteps from 'rc-steps'; import type { ProgressDotRender, StepsProps as RcStepsProps, StepIconRender, } from 'rc-steps/lib/Steps'; import { ConfigContext } from '../config-provider'; import useSize from '../config-provider/hooks/useSize'; import useBreakpoint from '../grid/hooks/useBreakpoint'; import Progress from '../progress'; import Tooltip from '../tooltip'; import useStyle from './style'; import useLegacyItems from './useLegacyItems'; export interface StepProps { className?: string; description?: React.ReactNode; icon?: React.ReactNode; onClick?: React.MouseEventHandler<HTMLElement>; status?: 'wait' | 'process' | 'finish' | 'error'; disabled?: boolean; title?: React.ReactNode; subTitle?: React.ReactNode; style?: React.CSSProperties; } export interface StepsProps { type?: 'default' | 'navigation' | 'inline'; className?: string; rootClassName?: string; current?: number; direction?: 'horizontal' | 'vertical'; iconPrefix?: string; initial?: number; labelPlacement?: 'horizontal' | 'vertical'; prefixCls?: string; progressDot?: boolean | ProgressDotRender; responsive?: boolean; size?: 'default' | 'small'; status?: 'wait' | 'process' | 'finish' | 'error'; style?: React.CSSProperties; percent?: number; onChange?: (current: number) => void; children?: React.ReactNode; items?: StepProps[]; } type CompoundedComponent = React.FC<StepsProps> & { Step: typeof RcSteps.Step; }; const Steps: CompoundedComponent = (props) => { const { percent, size: customizeSize, className, rootClassName, direction, items, responsive = true, current = 0, children, style, ...restProps } = props; const { xs } = useBreakpoint(responsive); const { getPrefixCls, direction: rtlDirection, steps } = React.useContext(ConfigContext); const realDirectionValue = React.useMemo<RcStepsProps['direction']>( () => (responsive && xs ? 'vertical' : direction), [xs, direction], ); const size = useSize(customizeSize); const prefixCls = getPrefixCls('steps', props.prefixCls); const [wrapCSSVar, hashId, cssVarCls] = useStyle(prefixCls); const isInline = props.type === 'inline'; const iconPrefix = getPrefixCls('', props.iconPrefix); const mergedItems = useLegacyItems(items, children); const mergedPercent = isInline ? undefined : percent; const mergedStyle: React.CSSProperties = { ...steps?.style, ...style }; const stepsClassName = classNames( steps?.className, { [`${prefixCls}-rtl`]: rtlDirection === 'rtl', [`${prefixCls}-with-progress`]: mergedPercent !== undefined, }, className, rootClassName, hashId, cssVarCls, ); const icons = { finish: <CheckOutlined className={`${prefixCls}-finish-icon`} />, error: <CloseOutlined className={`${prefixCls}-error-icon`} />, }; const stepIconRender: StepIconRender = ({ node, status }) => { if (status === 'process' && mergedPercent !== undefined) { // currently it's hard-coded, since we can't easily read the actually width of icon const progressWidth = size === 'small' ? 32 : 40; // iconWithProgress return ( <div className={`${prefixCls}-progress-icon`}> <Progress type="circle" percent={mergedPercent} size={progressWidth} strokeWidth={4} format={() => null} /> {node} </div> ); } return node; }; const itemRender = (item: StepProps, stepItem: React.ReactNode) => item.description ? <Tooltip title={item.description}>{stepItem}</Tooltip> : stepItem; return wrapCSSVar( <RcSteps icons={icons} {...restProps} style={mergedStyle} current={current} size={size} items={mergedItems} itemRender={isInline ? itemRender : undefined} stepIcon={stepIconRender} direction={realDirectionValue} prefixCls={prefixCls} iconPrefix={iconPrefix} className={stepsClassName} />, ); }; Steps.Step = RcSteps.Step; if (process.env.NODE_ENV !== 'production') { Steps.displayName = 'Steps'; } export default Steps;