import PropTypes, { withUndefined } from '../_util/vue-types'; import type { CSSProperties, PropType } from 'vue'; import { defineComponent } from 'vue'; import type { EventHandler } from '../_util/EventInterface'; function isString(str: any): str is string { return typeof str === 'string'; } function noop() {} export const VcStepProps = () => ({ prefixCls: String, wrapperStyle: { type: Object as PropType, default: undefined as CSSProperties }, itemWidth: String, active: { type: Boolean, default: undefined }, disabled: { type: Boolean, default: undefined }, status: String, iconPrefix: String, icon: PropTypes.any, adjustMarginRight: String, stepNumber: Number, stepIndex: Number, description: PropTypes.any, title: PropTypes.any, subTitle: PropTypes.any, progressDot: withUndefined(PropTypes.oneOfType([PropTypes.looseBool, PropTypes.func])), tailContent: PropTypes.any, icons: PropTypes.shape({ finish: PropTypes.any, error: PropTypes.any, }).loose, onClick: Function, onStepClick: Function, stepIcon: Function, }); export default defineComponent({ name: 'Step', props: VcStepProps(), slots: ['title', 'subTitle', 'description', 'tailContent', 'stepIcon', 'progressDot'], emits: ['click', 'stepClick'], setup(props, { slots, emit }) { const onItemClick: EventHandler = e => { emit('click', e); emit('stepClick', props.stepIndex); }; const renderIconNode = ({ icon, title, description }) => { const { prefixCls, stepNumber, status, iconPrefix, icons, progressDot = slots.progressDot, stepIcon = slots.stepIcon, } = props; let iconNode: any; const iconClassName = { [`${prefixCls}-icon`]: true, [`${iconPrefix}icon`]: true, [`${iconPrefix}icon-${icon}`]: icon && isString(icon), [`${iconPrefix}icon-check`]: !icon && status === 'finish' && icons && !icons.finish, [`${iconPrefix}icon-close`]: !icon && status === 'error' && icons && !icons.error, }; const iconDot = ; // `progressDot` enjoy the highest priority if (progressDot) { if (typeof progressDot === 'function') { iconNode = ( {progressDot({ iconDot, index: stepNumber - 1, status, title, description, prefixCls, })} ); } else { iconNode = {iconDot}; } } else if (icon && !isString(icon)) { iconNode = {icon}; } else if (icons && icons.finish && status === 'finish') { iconNode = {icons.finish}; } else if (icons && icons.error && status === 'error') { iconNode = {icons.error}; } else if (icon || status === 'finish' || status === 'error') { iconNode = ; } else { iconNode = {stepNumber}; } if (stepIcon) { iconNode = stepIcon({ index: stepNumber - 1, status, title, description, node: iconNode, }); } return iconNode; }; return () => { const { prefixCls, itemWidth, active, status = 'wait', tailContent, adjustMarginRight, disabled, title = slots.title?.(), description = slots.description?.(), subTitle = slots.subTitle?.(), icon = slots.icon?.(), onClick, onStepClick, } = props; const classString = { [`${prefixCls}-item`]: true, [`${prefixCls}-item-${status}`]: true, [`${prefixCls}-item-custom`]: icon, [`${prefixCls}-item-active`]: active, [`${prefixCls}-item-disabled`]: disabled === true, }; const stepProps = { class: classString, }; const stepItemStyle: CSSProperties = {}; if (itemWidth) { stepItemStyle.width = itemWidth; } if (adjustMarginRight) { stepItemStyle.marginRight = adjustMarginRight; } const accessibilityProps: { role?: string; tabindex?: number; onClick?: EventHandler; } = { onClick: onClick || noop, }; if (onStepClick && !disabled) { accessibilityProps.role = 'button'; accessibilityProps.tabindex = 0; accessibilityProps.onClick = onItemClick; } return (
{tailContent}
{renderIconNode({ icon, title, description })}
{title} {subTitle && (
{subTitle}
)}
{description &&
{description}
}
); }; }, });