200 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			Vue
		
	
	
			
		
		
	
	
			200 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			Vue
		
	
	
| import PropTypes, { withUndefined } from '../_util/vue-types';
 | |
| import type { CSSProperties, ExtractPropTypes } from 'vue';
 | |
| import { defineComponent } from 'vue';
 | |
| import type { EventHandler } from '../_util/EventInterface';
 | |
| import classNames from '../_util/classNames';
 | |
| import type { VueNode } from '../_util/type';
 | |
| import { booleanType, stringType, functionType } from '../_util/type';
 | |
| import type { StepIconRender, Status } from './interface';
 | |
| import omit from '../_util/omit';
 | |
| function isString(str: any): str is string {
 | |
|   return typeof str === 'string';
 | |
| }
 | |
| function noop() {}
 | |
| 
 | |
| export const VcStepProps = () => ({
 | |
|   prefixCls: String,
 | |
|   itemWidth: String,
 | |
|   active: { type: Boolean, default: undefined },
 | |
|   disabled: { type: Boolean, default: undefined },
 | |
|   status: stringType<Status>(),
 | |
|   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: functionType(),
 | |
|   onStepClick: functionType<(next: number) => void>(),
 | |
|   stepIcon: functionType<StepIconRender>(),
 | |
|   itemRender: functionType<(stepItem: VueNode) => VueNode>(),
 | |
|   __legacy: booleanType(),
 | |
| });
 | |
| 
 | |
| export type VCStepProps = Partial<ExtractPropTypes<ReturnType<typeof VcStepProps>>>;
 | |
| export default defineComponent({
 | |
|   compatConfig: { MODE: 3 },
 | |
|   name: 'Step',
 | |
|   inheritAttrs: false,
 | |
|   props: VcStepProps(),
 | |
|   setup(props, { slots, emit, attrs }) {
 | |
|     const onItemClick: EventHandler = e => {
 | |
|       emit('click', e);
 | |
|       emit('stepClick', props.stepIndex);
 | |
|     };
 | |
|     // if (props.__legacy !== false) {
 | |
|     //   warning(
 | |
|     //     false,
 | |
|     //     'Steps',
 | |
|     //     'Step is deprecated, and not support inline type. Please use `items` directly. ',
 | |
|     //   );
 | |
|     // }
 | |
|     const renderIconNode = ({ icon, title, description }) => {
 | |
|       const {
 | |
|         prefixCls,
 | |
|         stepNumber,
 | |
|         status,
 | |
|         iconPrefix,
 | |
|         icons,
 | |
|         progressDot = slots.progressDot,
 | |
|         stepIcon = slots.stepIcon,
 | |
|       } = props;
 | |
| 
 | |
|       let iconNode;
 | |
|       const iconClassName = classNames(`${prefixCls}-icon`, `${iconPrefix}icon`, {
 | |
|         [`${iconPrefix}icon-${icon}`]: icon && isString(icon),
 | |
|         [`${iconPrefix}icon-check`]:
 | |
|           !icon && status === 'finish' && ((icons && !icons.finish) || !icons),
 | |
|         [`${iconPrefix}icon-cross`]:
 | |
|           !icon && status === 'error' && ((icons && !icons.error) || !icons),
 | |
|       });
 | |
|       const iconDot = <span class={`${prefixCls}-icon-dot`} />;
 | |
|       // `progressDot` enjoy the highest priority
 | |
|       if (progressDot) {
 | |
|         if (typeof progressDot === 'function') {
 | |
|           iconNode = (
 | |
|             <span class={`${prefixCls}-icon`}>
 | |
|               {progressDot({
 | |
|                 iconDot,
 | |
|                 index: stepNumber - 1,
 | |
|                 status,
 | |
|                 title,
 | |
|                 description,
 | |
|                 prefixCls,
 | |
|               })}
 | |
|             </span>
 | |
|           );
 | |
|         } else {
 | |
|           iconNode = <span class={`${prefixCls}-icon`}>{iconDot}</span>;
 | |
|         }
 | |
|       } else if (icon && !isString(icon)) {
 | |
|         iconNode = <span class={`${prefixCls}-icon`}>{icon}</span>;
 | |
|       } else if (icons && icons.finish && status === 'finish') {
 | |
|         iconNode = <span class={`${prefixCls}-icon`}>{icons.finish}</span>;
 | |
|       } else if (icons && icons.error && status === 'error') {
 | |
|         iconNode = <span class={`${prefixCls}-icon`}>{icons.error}</span>;
 | |
|       } else if (icon || status === 'finish' || status === 'error') {
 | |
|         iconNode = <span class={iconClassName} />;
 | |
|       } else {
 | |
|         iconNode = <span class={`${prefixCls}-icon`}>{stepNumber}</span>;
 | |
|       }
 | |
| 
 | |
|       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 mergedStatus = status || 'wait';
 | |
|       const classString = classNames(`${prefixCls}-item`, `${prefixCls}-item-${mergedStatus}`, {
 | |
|         [`${prefixCls}-item-custom`]: icon,
 | |
|         [`${prefixCls}-item-active`]: active,
 | |
|         [`${prefixCls}-item-disabled`]: disabled === true,
 | |
|       });
 | |
|       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;
 | |
|       }
 | |
|       const stepNode = (
 | |
|         <div
 | |
|           {...omit(attrs, ['__legacy'])}
 | |
|           class={[classString, attrs.class]}
 | |
|           style={[attrs.style as CSSProperties, stepItemStyle]}
 | |
|         >
 | |
|           <div {...accessibilityProps} class={`${prefixCls}-item-container`}>
 | |
|             <div class={`${prefixCls}-item-tail`}>{tailContent}</div>
 | |
|             <div class={`${prefixCls}-item-icon`}>
 | |
|               {renderIconNode({ icon, title, description })}
 | |
|             </div>
 | |
|             <div class={`${prefixCls}-item-content`}>
 | |
|               <div class={`${prefixCls}-item-title`}>
 | |
|                 {title}
 | |
|                 {subTitle && (
 | |
|                   <div
 | |
|                     title={typeof subTitle === 'string' ? subTitle : undefined}
 | |
|                     class={`${prefixCls}-item-subtitle`}
 | |
|                   >
 | |
|                     {subTitle}
 | |
|                   </div>
 | |
|                 )}
 | |
|               </div>
 | |
|               {description && <div class={`${prefixCls}-item-description`}>{description}</div>}
 | |
|             </div>
 | |
|           </div>
 | |
|         </div>
 | |
|       );
 | |
|       if (props.itemRender) {
 | |
|         return props.itemRender(stepNode);
 | |
|       }
 | |
|       return stepNode;
 | |
|     };
 | |
|   },
 | |
| });
 |