507 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Vue
		
	
	
			
		
		
	
	
			507 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Vue
		
	
	
| import PropTypes from '../_util/vue-types';
 | ||
| import classNames from 'classnames';
 | ||
| import find from 'lodash/find';
 | ||
| import Row from '../grid/Row';
 | ||
| import Col, { ColProps } from '../grid/Col';
 | ||
| import warning from '../_util/warning';
 | ||
| import { FIELD_META_PROP, FIELD_DATA_PROP } from './constants';
 | ||
| import {
 | ||
|   initDefaultProps,
 | ||
|   getComponentFromProp,
 | ||
|   filterEmpty,
 | ||
|   getSlotOptions,
 | ||
|   isValidElement,
 | ||
|   getAllChildren,
 | ||
| } from '../_util/props-util';
 | ||
| import getTransitionProps from '../_util/getTransitionProps';
 | ||
| import BaseMixin from '../_util/BaseMixin';
 | ||
| import { cloneElement, cloneVNodes } from '../_util/vnode';
 | ||
| import Icon from '../icon';
 | ||
| import { ConfigConsumerProps } from '../config-provider';
 | ||
| 
 | ||
| function noop() {}
 | ||
| 
 | ||
| function intersperseSpace(list) {
 | ||
|   return list.reduce((current, item) => [...current, ' ', item], []).slice(1);
 | ||
| }
 | ||
| export const FormItemProps = {
 | ||
|   id: PropTypes.string,
 | ||
|   prefixCls: PropTypes.string,
 | ||
|   label: PropTypes.any,
 | ||
|   labelCol: PropTypes.shape(ColProps).loose,
 | ||
|   wrapperCol: PropTypes.shape(ColProps).loose,
 | ||
|   help: PropTypes.any,
 | ||
|   extra: PropTypes.any,
 | ||
|   validateStatus: PropTypes.oneOf(['', 'success', 'warning', 'error', 'validating']),
 | ||
|   hasFeedback: PropTypes.bool,
 | ||
|   required: PropTypes.bool,
 | ||
|   colon: PropTypes.bool,
 | ||
|   fieldDecoratorId: PropTypes.string,
 | ||
|   fieldDecoratorOptions: PropTypes.object,
 | ||
|   selfUpdate: PropTypes.bool,
 | ||
| };
 | ||
| function comeFromSlot(vnodes = [], itemVnode) {
 | ||
|   let isSlot = false;
 | ||
|   for (let i = 0, len = vnodes.length; i < len; i++) {
 | ||
|     const vnode = vnodes[i];
 | ||
|     if (vnode && (vnode === itemVnode || vnode.$vnode === itemVnode)) {
 | ||
|       isSlot = true;
 | ||
|     } else {
 | ||
|       const componentOptions =
 | ||
|         vnode.componentOptions || (vnode.$vnode && vnode.$vnode.componentOptions);
 | ||
|       const children = componentOptions ? componentOptions.children : vnode.$children;
 | ||
|       isSlot = comeFromSlot(children, itemVnode);
 | ||
|     }
 | ||
|     if (isSlot) {
 | ||
|       break;
 | ||
|     }
 | ||
|   }
 | ||
|   return isSlot;
 | ||
| }
 | ||
| 
 | ||
| export default {
 | ||
|   name: 'AFormItem',
 | ||
|   __ANT_FORM_ITEM: true,
 | ||
|   mixins: [BaseMixin],
 | ||
|   props: initDefaultProps(FormItemProps, {
 | ||
|     hasFeedback: false,
 | ||
|     colon: true,
 | ||
|   }),
 | ||
|   inject: {
 | ||
|     FormProps: { default: () => ({}) },
 | ||
|     decoratorFormProps: { default: () => ({}) },
 | ||
|     collectFormItemContext: { default: () => noop },
 | ||
|     configProvider: { default: () => ConfigConsumerProps },
 | ||
|   },
 | ||
|   data() {
 | ||
|     return { helpShow: false };
 | ||
|   },
 | ||
|   computed: {
 | ||
|     itemSelfUpdate() {
 | ||
|       return !!(this.selfUpdate === undefined ? this.FormProps.selfUpdate : this.selfUpdate);
 | ||
|     },
 | ||
|   },
 | ||
|   created() {
 | ||
|     this.collectContext();
 | ||
|   },
 | ||
|   beforeUpdate() {
 | ||
|     if (process.env.NODE_ENV !== 'production') {
 | ||
|       this.collectContext();
 | ||
|     }
 | ||
|   },
 | ||
|   beforeDestroy() {
 | ||
|     this.collectFormItemContext(this.$vnode.context, 'delete');
 | ||
|   },
 | ||
|   mounted() {
 | ||
|     const { help, validateStatus } = this.$props;
 | ||
|     warning(
 | ||
|       this.getControls(this.slotDefault, true).length <= 1 ||
 | ||
|         (help !== undefined || validateStatus !== undefined),
 | ||
|       '`Form.Item` cannot generate `validateStatus` and `help` automatically, ' +
 | ||
|         'while there are more than one `getFieldDecorator` in it.',
 | ||
|     );
 | ||
|     warning(
 | ||
|       !this.fieldDecoratorId,
 | ||
|       '`fieldDecoratorId` is deprecated. please use `v-decorator={id, options}` instead.',
 | ||
|     );
 | ||
|   },
 | ||
|   methods: {
 | ||
|     collectContext() {
 | ||
|       if (this.FormProps.form && this.FormProps.form.templateContext) {
 | ||
|         const { templateContext } = this.FormProps.form;
 | ||
|         const vnodes = Object.values(templateContext.$slots || {}).reduce((a, b) => {
 | ||
|           return [...a, ...b];
 | ||
|         }, []);
 | ||
|         const isSlot = comeFromSlot(vnodes, this.$vnode);
 | ||
|         warning(!isSlot, 'You can not set FormItem from slot, please use slot-scope instead slot');
 | ||
|         let isSlotScope = false;
 | ||
|         // čŋ䏿Ĩ夿æ¯åĻæ¯éčŋslot-scopeäŧ é
 | ||
|         if (!isSlot && this.$vnode.context !== templateContext) {
 | ||
|           isSlotScope = comeFromSlot(this.$vnode.context.$children, templateContext.$vnode);
 | ||
|         }
 | ||
|         if (!isSlotScope && !isSlot) {
 | ||
|           this.collectFormItemContext(this.$vnode.context);
 | ||
|         }
 | ||
|       }
 | ||
|     },
 | ||
|     getHelpMessage() {
 | ||
|       const help = getComponentFromProp(this, 'help');
 | ||
|       const onlyControl = this.getOnlyControl();
 | ||
|       if (help === undefined && onlyControl) {
 | ||
|         const errors = this.getField().errors;
 | ||
|         if (errors) {
 | ||
|           return intersperseSpace(
 | ||
|             errors.map((e, index) => {
 | ||
|               let node = null;
 | ||
|               if (isValidElement(e)) {
 | ||
|                 node = e;
 | ||
|               } else if (isValidElement(e.message)) {
 | ||
|                 node = e.message;
 | ||
|               }
 | ||
|               return node ? cloneElement(node, { key: index }) : e.message;
 | ||
|             }),
 | ||
|           );
 | ||
|         } else {
 | ||
|           return '';
 | ||
|         }
 | ||
|       }
 | ||
| 
 | ||
|       return help;
 | ||
|     },
 | ||
| 
 | ||
|     getControls(childrenArray = [], recursively) {
 | ||
|       let controls = [];
 | ||
|       for (let i = 0; i < childrenArray.length; i++) {
 | ||
|         if (!recursively && controls.length > 0) {
 | ||
|           break;
 | ||
|         }
 | ||
| 
 | ||
|         const child = childrenArray[i];
 | ||
|         if (!child.tag && child.text.trim() === '') {
 | ||
|           continue;
 | ||
|         }
 | ||
| 
 | ||
|         if (getSlotOptions(child).__ANT_FORM_ITEM) {
 | ||
|           continue;
 | ||
|         }
 | ||
|         const children = getAllChildren(child);
 | ||
|         const attrs = (child.data && child.data.attrs) || {};
 | ||
|         if (FIELD_META_PROP in attrs) {
 | ||
|           // And means FIELD_DATA_PROP in child.props, too.
 | ||
|           controls.push(child);
 | ||
|         } else if (children) {
 | ||
|           controls = controls.concat(this.getControls(children, recursively));
 | ||
|         }
 | ||
|       }
 | ||
|       return controls;
 | ||
|     },
 | ||
| 
 | ||
|     getOnlyControl() {
 | ||
|       const child = this.getControls(this.slotDefault, false)[0];
 | ||
|       return child !== undefined ? child : null;
 | ||
|     },
 | ||
| 
 | ||
|     getChildAttr(prop) {
 | ||
|       const child = this.getOnlyControl();
 | ||
|       let data = {};
 | ||
|       if (!child) {
 | ||
|         return undefined;
 | ||
|       }
 | ||
|       if (child.data) {
 | ||
|         data = child.data;
 | ||
|       } else if (child.$vnode && child.$vnode.data) {
 | ||
|         data = child.$vnode.data;
 | ||
|       }
 | ||
|       return data[prop] || data.attrs[prop];
 | ||
|     },
 | ||
| 
 | ||
|     getId() {
 | ||
|       return this.getChildAttr('id');
 | ||
|     },
 | ||
| 
 | ||
|     getMeta() {
 | ||
|       return this.getChildAttr(FIELD_META_PROP);
 | ||
|     },
 | ||
| 
 | ||
|     getField() {
 | ||
|       return this.getChildAttr(FIELD_DATA_PROP);
 | ||
|     },
 | ||
| 
 | ||
|     onHelpAnimEnd(_key, helpShow) {
 | ||
|       this.helpShow = helpShow;
 | ||
|       if (!helpShow) {
 | ||
|         this.$forceUpdate();
 | ||
|       }
 | ||
|     },
 | ||
| 
 | ||
|     renderHelp(prefixCls) {
 | ||
|       const help = this.getHelpMessage();
 | ||
|       const children = help ? (
 | ||
|         <div class={`${prefixCls}-explain`} key="help">
 | ||
|           {help}
 | ||
|         </div>
 | ||
|       ) : null;
 | ||
|       if (children) {
 | ||
|         this.helpShow = !!children;
 | ||
|       }
 | ||
|       const transitionProps = getTransitionProps('show-help', {
 | ||
|         afterEnter: () => this.onHelpAnimEnd('help', true),
 | ||
|         afterLeave: () => this.onHelpAnimEnd('help', false),
 | ||
|       });
 | ||
|       return (
 | ||
|         <transition {...transitionProps} key="help">
 | ||
|           {children}
 | ||
|         </transition>
 | ||
|       );
 | ||
|     },
 | ||
| 
 | ||
|     renderExtra(prefixCls) {
 | ||
|       const extra = getComponentFromProp(this, 'extra');
 | ||
|       return extra ? <div class={`${prefixCls}-extra`}>{extra}</div> : null;
 | ||
|     },
 | ||
| 
 | ||
|     getValidateStatus() {
 | ||
|       const onlyControl = this.getOnlyControl();
 | ||
|       if (!onlyControl) {
 | ||
|         return '';
 | ||
|       }
 | ||
|       const field = this.getField();
 | ||
|       if (field.validating) {
 | ||
|         return 'validating';
 | ||
|       }
 | ||
|       if (field.errors) {
 | ||
|         return 'error';
 | ||
|       }
 | ||
|       const fieldValue = 'value' in field ? field.value : this.getMeta().initialValue;
 | ||
|       if (fieldValue !== undefined && fieldValue !== null && fieldValue !== '') {
 | ||
|         return 'success';
 | ||
|       }
 | ||
|       return '';
 | ||
|     },
 | ||
| 
 | ||
|     renderValidateWrapper(prefixCls, c1, c2, c3) {
 | ||
|       const props = this.$props;
 | ||
|       const onlyControl = this.getOnlyControl;
 | ||
|       const validateStatus =
 | ||
|         props.validateStatus === undefined && onlyControl
 | ||
|           ? this.getValidateStatus()
 | ||
|           : props.validateStatus;
 | ||
| 
 | ||
|       let classes = `${prefixCls}-item-control`;
 | ||
|       if (validateStatus) {
 | ||
|         classes = classNames(`${prefixCls}-item-control`, {
 | ||
|           'has-feedback': props.hasFeedback || validateStatus === 'validating',
 | ||
|           'has-success': validateStatus === 'success',
 | ||
|           'has-warning': validateStatus === 'warning',
 | ||
|           'has-error': validateStatus === 'error',
 | ||
|           'is-validating': validateStatus === 'validating',
 | ||
|         });
 | ||
|       }
 | ||
|       let iconType = '';
 | ||
|       switch (validateStatus) {
 | ||
|         case 'success':
 | ||
|           iconType = 'check-circle';
 | ||
|           break;
 | ||
|         case 'warning':
 | ||
|           iconType = 'exclamation-circle';
 | ||
|           break;
 | ||
|         case 'error':
 | ||
|           iconType = 'close-circle';
 | ||
|           break;
 | ||
|         case 'validating':
 | ||
|           iconType = 'loading';
 | ||
|           break;
 | ||
|         default:
 | ||
|           iconType = '';
 | ||
|           break;
 | ||
|       }
 | ||
|       const icon =
 | ||
|         props.hasFeedback && iconType ? (
 | ||
|           <span class={`${prefixCls}-item-children-icon`}>
 | ||
|             <Icon type={iconType} theme={iconType === 'loading' ? 'outlined' : 'filled'} />
 | ||
|           </span>
 | ||
|         ) : null;
 | ||
|       return (
 | ||
|         <div class={classes}>
 | ||
|           <span class={`${prefixCls}-item-children`}>
 | ||
|             {c1}
 | ||
|             {icon}
 | ||
|           </span>
 | ||
|           {c2}
 | ||
|           {c3}
 | ||
|         </div>
 | ||
|       );
 | ||
|     },
 | ||
| 
 | ||
|     renderWrapper(prefixCls, children) {
 | ||
|       const { wrapperCol = {} } = this;
 | ||
|       const { class: cls, style, id, on, ...restProps } = wrapperCol;
 | ||
|       const className = classNames(`${prefixCls}-item-control-wrapper`, cls);
 | ||
|       const colProps = {
 | ||
|         props: restProps,
 | ||
|         class: className,
 | ||
|         key: 'wrapper',
 | ||
|         style,
 | ||
|         id,
 | ||
|         on,
 | ||
|       };
 | ||
|       return <Col {...colProps}>{children}</Col>;
 | ||
|     },
 | ||
| 
 | ||
|     isRequired() {
 | ||
|       const { required } = this;
 | ||
|       if (required !== undefined) {
 | ||
|         return required;
 | ||
|       }
 | ||
|       if (this.getOnlyControl()) {
 | ||
|         const meta = this.getMeta() || {};
 | ||
|         const validate = meta.validate || [];
 | ||
| 
 | ||
|         return validate
 | ||
|           .filter(item => !!item.rules)
 | ||
|           .some(item => {
 | ||
|             return item.rules.some(rule => rule.required);
 | ||
|           });
 | ||
|       }
 | ||
|       return false;
 | ||
|     },
 | ||
| 
 | ||
|     // Resolve duplicated ids bug between different forms
 | ||
|     // https://github.com/ant-design/ant-design/issues/7351
 | ||
|     onLabelClick(e) {
 | ||
|       const label = getComponentFromProp(this, 'label');
 | ||
|       const id = this.id || this.getId();
 | ||
|       if (!id) {
 | ||
|         return;
 | ||
|       }
 | ||
|       const formItemNode = this.$el;
 | ||
|       const control = formItemNode.querySelector(`[id="${id}"]`);
 | ||
|       if (control) {
 | ||
|         // Only prevent in default situation
 | ||
|         // Avoid preventing event in `label={<a href="xx">link</a>}``
 | ||
|         if (typeof label === 'string') {
 | ||
|           e.preventDefault();
 | ||
|         }
 | ||
|         if (control.focus) {
 | ||
|           control.focus();
 | ||
|         }
 | ||
|       }
 | ||
|     },
 | ||
| 
 | ||
|     renderLabel(prefixCls) {
 | ||
|       const { labelCol = {}, colon, id } = this;
 | ||
|       const label = getComponentFromProp(this, 'label');
 | ||
|       const required = this.isRequired();
 | ||
|       const {
 | ||
|         class: labelColClass,
 | ||
|         style: labelColStyle,
 | ||
|         id: labelColId,
 | ||
|         on,
 | ||
|         ...restProps
 | ||
|       } = labelCol;
 | ||
|       const labelColClassName = classNames(`${prefixCls}-item-label`, labelColClass);
 | ||
|       const labelClassName = classNames({
 | ||
|         [`${prefixCls}-item-required`]: required,
 | ||
|       });
 | ||
| 
 | ||
|       let labelChildren = label;
 | ||
|       // Keep label is original where there should have no colon
 | ||
|       const haveColon = colon && this.FormProps.layout !== 'vertical';
 | ||
|       // Remove duplicated user input colon
 | ||
|       if (haveColon && typeof label === 'string' && label.trim() !== '') {
 | ||
|         labelChildren = label.replace(/[īŧ|:]\s*$/, '');
 | ||
|       }
 | ||
|       const colProps = {
 | ||
|         props: restProps,
 | ||
|         class: labelColClassName,
 | ||
|         key: 'label',
 | ||
|         style: labelColStyle,
 | ||
|         id: labelColId,
 | ||
|         on,
 | ||
|       };
 | ||
| 
 | ||
|       return label ? (
 | ||
|         <Col {...colProps}>
 | ||
|           <label
 | ||
|             for={id || this.getId()}
 | ||
|             class={labelClassName}
 | ||
|             title={typeof label === 'string' ? label : ''}
 | ||
|             onClick={this.onLabelClick}
 | ||
|           >
 | ||
|             {labelChildren}
 | ||
|           </label>
 | ||
|         </Col>
 | ||
|       ) : null;
 | ||
|     },
 | ||
|     renderChildren(prefixCls) {
 | ||
|       return [
 | ||
|         this.renderLabel(prefixCls),
 | ||
|         this.renderWrapper(
 | ||
|           prefixCls,
 | ||
|           this.renderValidateWrapper(
 | ||
|             prefixCls,
 | ||
|             this.slotDefault,
 | ||
|             this.renderHelp(prefixCls),
 | ||
|             this.renderExtra(prefixCls),
 | ||
|           ),
 | ||
|         ),
 | ||
|       ];
 | ||
|     },
 | ||
|     renderFormItem() {
 | ||
|       const { prefixCls: customizePrefixCls, colon } = this.$props;
 | ||
|       const getPrefixCls = this.configProvider.getPrefixCls;
 | ||
|       const prefixCls = getPrefixCls('form', customizePrefixCls);
 | ||
|       const children = this.renderChildren(prefixCls);
 | ||
|       const itemClassName = {
 | ||
|         [`${prefixCls}-item`]: true,
 | ||
|         [`${prefixCls}-item-with-help`]: this.helpShow,
 | ||
|         [`${prefixCls}-item-no-colon`]: !colon,
 | ||
|       };
 | ||
| 
 | ||
|       return <Row class={classNames(itemClassName)}>{children}</Row>;
 | ||
|     },
 | ||
|     decoratorOption(vnode) {
 | ||
|       if (vnode.data && vnode.data.directives) {
 | ||
|         const directive = find(vnode.data.directives, ['name', 'decorator']);
 | ||
|         warning(
 | ||
|           !directive || (directive && Array.isArray(directive.value)),
 | ||
|           `Invalid directive: type check failed for directive "decorator". Expected Array, got ${typeof (directive
 | ||
|             ? directive.value
 | ||
|             : directive)}. At ${vnode.tag}.`,
 | ||
|         );
 | ||
|         return directive ? directive.value : null;
 | ||
|       } else {
 | ||
|         return null;
 | ||
|       }
 | ||
|     },
 | ||
|     decoratorChildren(vnodes) {
 | ||
|       const { FormProps } = this;
 | ||
|       const getFieldDecorator = FormProps.form.getFieldDecorator;
 | ||
|       for (let i = 0, len = vnodes.length; i < len; i++) {
 | ||
|         const vnode = vnodes[i];
 | ||
|         if (getSlotOptions(vnode).__ANT_FORM_ITEM) {
 | ||
|           break;
 | ||
|         }
 | ||
|         if (vnode.children) {
 | ||
|           vnode.children = this.decoratorChildren(cloneVNodes(vnode.children));
 | ||
|         } else if (vnode.componentOptions && vnode.componentOptions.children) {
 | ||
|           vnode.componentOptions.children = this.decoratorChildren(
 | ||
|             cloneVNodes(vnode.componentOptions.children),
 | ||
|           );
 | ||
|         }
 | ||
|         const option = this.decoratorOption(vnode);
 | ||
|         if (option && option[0]) {
 | ||
|           vnodes[i] = getFieldDecorator(option[0], option[1], this)(vnode);
 | ||
|         }
 | ||
|       }
 | ||
|       return vnodes;
 | ||
|     },
 | ||
|   },
 | ||
| 
 | ||
|   render() {
 | ||
|     const {
 | ||
|       $slots,
 | ||
|       decoratorFormProps,
 | ||
|       fieldDecoratorId,
 | ||
|       fieldDecoratorOptions = {},
 | ||
|       FormProps,
 | ||
|     } = this;
 | ||
|     let child = filterEmpty($slots.default || []);
 | ||
|     if (decoratorFormProps.form && fieldDecoratorId && child.length) {
 | ||
|       const getFieldDecorator = decoratorFormProps.form.getFieldDecorator;
 | ||
|       child[0] = getFieldDecorator(fieldDecoratorId, fieldDecoratorOptions, this)(child[0]);
 | ||
|       warning(
 | ||
|         !(child.length > 1),
 | ||
|         '`autoFormCreate` just `decorator` then first children. but you can use JSX to support multiple children',
 | ||
|       );
 | ||
|       this.slotDefault = child;
 | ||
|     } else if (FormProps.form) {
 | ||
|       child = cloneVNodes(child);
 | ||
|       this.slotDefault = this.decoratorChildren(child);
 | ||
|     } else {
 | ||
|       this.slotDefault = child;
 | ||
|     }
 | ||
|     return this.renderFormItem();
 | ||
|   },
 | ||
| };
 |