import { inject, provide, Transition } from 'vue'; import cloneDeep from 'lodash-es/cloneDeep'; import PropTypes from '../_util/vue-types'; import classNames from '../_util/classNames'; import getTransitionProps from '../_util/getTransitionProps'; import Row from '../grid/Row'; import Col, { ColProps } from '../grid/Col'; import hasProp, { initDefaultProps, findDOMNode, getComponent, getOptionProps, getEvents, isValidElement, getSlot, } from '../_util/props-util'; import BaseMixin from '../_util/BaseMixin'; import { defaultConfigProvider } from '../config-provider'; import { cloneElement } from '../_util/vnode'; import CheckCircleFilled from '@ant-design/icons-vue/CheckCircleFilled'; import ExclamationCircleFilled from '@ant-design/icons-vue/ExclamationCircleFilled'; import CloseCircleFilled from '@ant-design/icons-vue/CloseCircleFilled'; import LoadingOutlined from '@ant-design/icons-vue/LoadingOutlined'; import { validateRules } from './utils/validateUtil'; import { getNamePath } from './utils/valueUtil'; import { toArray } from './utils/typeUtil'; import { warning } from '../vc-util/warning'; import find from 'lodash-es/find'; const iconMap = { success: CheckCircleFilled, warning: ExclamationCircleFilled, error: CloseCircleFilled, validating: LoadingOutlined, }; function getPropByPath(obj, namePathList, strict) { let tempObj = obj; const keyArr = namePathList; let i = 0; try { for (let len = keyArr.length; i < len - 1; ++i) { if (!tempObj && !strict) break; let key = keyArr[i]; if (key in tempObj) { tempObj = tempObj[key]; } else { if (strict) { throw Error('please transfer a valid name path to form item!'); } break; } } if (strict && !tempObj) { throw Error('please transfer a valid name path to form item!'); } } catch (error) { console.error('please transfer a valid name path to form item!'); } return { o: tempObj, k: keyArr[i], v: tempObj ? tempObj[keyArr[i]] : undefined, }; } export const FormItemProps = { id: PropTypes.string, htmlFor: PropTypes.string, prefixCls: PropTypes.string, label: PropTypes.any, help: PropTypes.any, extra: PropTypes.any, labelCol: PropTypes.shape(ColProps).loose, wrapperCol: PropTypes.shape(ColProps).loose, hasFeedback: PropTypes.bool, colon: PropTypes.bool, labelAlign: PropTypes.oneOf(['left', 'right']), prop: PropTypes.oneOfType([Array, String, Number]), name: PropTypes.oneOfType([Array, String, Number]), rules: PropTypes.oneOfType([Array, Object]), autoLink: PropTypes.bool, required: PropTypes.bool, validateFirst: PropTypes.bool, validateStatus: PropTypes.oneOf(['', 'success', 'warning', 'error', 'validating']), validateTrigger: PropTypes.oneOfType([PropTypes.string, PropTypes.array]), }; export default { name: 'AFormItem', mixins: [BaseMixin], inheritAttrs: false, __ANT_NEW_FORM_ITEM: true, props: initDefaultProps(FormItemProps, { hasFeedback: false, autoLink: true, }), setup() { return { isFormItemChildren: inject('isFormItemChildren', false), configProvider: inject('configProvider', defaultConfigProvider), FormContext: inject('FormContext', {}), }; }, data() { warning(!hasProp(this, 'prop'), `\`prop\` is deprecated. Please use \`name\` instead.`); return { validateState: this.validateStatus, validateMessage: '', validateDisabled: false, validator: {}, helpShow: false, errors: [], }; }, computed: { fieldName() { return this.name || this.prop; }, namePath() { return getNamePath(this.fieldName); }, fieldId() { if (this.id) { return this.id; } else if (!this.namePath.length) { return undefined; } else { const formName = this.FormContext.name; const mergedId = this.namePath.join('_'); return formName ? `${formName}_${mergedId}` : mergedId; } }, fieldValue() { const model = this.FormContext.model; if (!model || !this.fieldName) { return; } return getPropByPath(model, this.namePath, true).v; }, isRequired() { let rules = this.getRules(); let isRequired = false; if (rules && rules.length) { rules.every(rule => { if (rule.required) { isRequired = true; return false; } return true; }); } return isRequired || this.required; }, mergedValidateTrigger() { let validateTrigger = this.validateTrigger !== undefined ? this.validateTrigger : this.FormContext.validateTrigger; validateTrigger = validateTrigger === undefined ? 'change' : validateTrigger; return toArray(validateTrigger); }, }, watch: { validateStatus(val) { this.validateState = val; }, }, created() { provide('isFormItemChildren', true); }, mounted() { if (this.fieldName) { const { addField } = this.FormContext; addField && addField(this); this.initialValue = cloneDeep(this.fieldValue); } }, beforeUnmount() { const { removeField } = this.FormContext; removeField && removeField(this); }, methods: { getNamePath() { const { fieldName } = this; const { prefixName = [] } = this.FormContext; return fieldName !== undefined ? [...prefixName, ...this.namePath] : []; }, validateRules(options) { const { validateFirst = false, messageVariables } = this.$props; const { triggerName } = options || {}; const namePath = this.getNamePath(); let filteredRules = this.getRules(); if (triggerName) { filteredRules = filteredRules.filter(rule => { const { trigger } = rule; if (!trigger && !this.mergedValidateTrigger.length) { return true; } const triggerList = toArray(trigger || this.mergedValidateTrigger); return triggerList.includes(triggerName); }); } if (!filteredRules.length) { return Promise.resolve(); } const promise = validateRules( namePath, this.fieldValue, filteredRules, options, validateFirst, messageVariables, ); this.validateState = 'validating'; this.errors = []; promise .catch(e => e) .then((errors = []) => { if (this.validateState === 'validating') { this.validateState = errors.length ? 'error' : 'success'; this.validateMessage = errors[0]; this.errors = errors; } }); return promise; }, getRules() { let formRules = this.FormContext.rules; const selfRules = this.rules; const requiredRule = this.required !== undefined ? { required: !!this.required, trigger: this.mergedValidateTrigger } : []; const prop = getPropByPath(formRules, this.namePath); formRules = formRules ? prop.o[prop.k] || prop.v : []; const rules = [].concat(selfRules || formRules || []); if (find(rules, rule => rule.required)) { return rules; } else { return rules.concat(requiredRule); } }, onFieldBlur() { this.validateRules({ triggerName: 'blur' }); }, onFieldChange() { if (this.validateDisabled) { this.validateDisabled = false; return; } this.validateRules({ triggerName: 'change' }); }, clearValidate() { this.validateState = ''; this.validateMessage = ''; this.validateDisabled = false; }, resetField() { this.validateState = ''; this.validateMessage = ''; const model = this.FormContext.model || {}; const value = this.fieldValue; const prop = getPropByPath(model, this.namePath, true); this.validateDisabled = true; if (Array.isArray(value)) { prop.o[prop.k] = [].concat(this.initialValue); } else { prop.o[prop.k] = this.initialValue; } // reset validateDisabled after onFieldChange triggered this.$nextTick(() => { this.validateDisabled = false; }); }, getHelpMessage() { const help = getComponent(this, 'help'); return this.validateMessage || help; }, onLabelClick() { const id = this.fieldId; if (!id) { return; } const formItemNode = findDOMNode(this); const control = formItemNode.querySelector(`[id="${id}"]`); if (control && control.focus) { control.focus(); } }, onHelpAnimEnd(_key, helpShow) { this.helpShow = helpShow; if (!helpShow) { this.$forceUpdate(); } }, renderHelp(prefixCls) { const help = this.getHelpMessage(); const children = help ? (