import classNames from 'classnames'; import type { FormInstance } from 'rc-field-form'; import { Field, FieldContext, ListContext } from 'rc-field-form'; import type { FieldProps } from 'rc-field-form/lib/Field'; import type { Meta, NamePath } from 'rc-field-form/lib/interface'; import useState from 'rc-util/lib/hooks/useState'; import { supportRef } from 'rc-util/lib/ref'; import * as React from 'react'; import useFormItemStatus from '../hooks/useFormItemStatus'; import { ConfigContext } from '../../config-provider'; import { cloneElement, isValidElement } from '../../_util/reactNode'; import warning from '../../_util/warning'; import { FormContext, NoStyleItemContext } from '../context'; import type { FormItemInputProps } from '../FormItemInput'; import type { FormItemLabelProps, LabelTooltipType } from '../FormItemLabel'; import useFrameState from '../hooks/useFrameState'; import useItemRef from '../hooks/useItemRef'; import { getFieldId, toArray } from '../util'; import ItemHolder from './ItemHolder'; import useStyle from '../style'; const NAME_SPLIT = '__SPLIT__'; interface FieldError { errors: string[]; warnings: string[]; } const ValidateStatuses = ['success', 'warning', 'error', 'validating', ''] as const; export type ValidateStatus = typeof ValidateStatuses[number]; type RenderChildren = (form: FormInstance) => React.ReactNode; type RcFieldProps = Omit, 'children'>; type ChildrenType = RenderChildren | React.ReactNode; interface MemoInputProps { value: any; update: any; children: React.ReactNode; childProps: any[]; } const MemoInput = React.memo( ({ children }: MemoInputProps) => children as JSX.Element, (prev, next) => prev.value === next.value && prev.update === next.update && prev.childProps.length === next.childProps.length && prev.childProps.every((value, index) => value === next.childProps[index]), ); export interface FormItemProps extends Omit, FormItemInputProps, RcFieldProps { prefixCls?: string; noStyle?: boolean; style?: React.CSSProperties; className?: string; rootClassName?: string; children?: ChildrenType; id?: string; hasFeedback?: boolean; validateStatus?: ValidateStatus; required?: boolean; hidden?: boolean; initialValue?: any; messageVariables?: Record; tooltip?: LabelTooltipType; /** @deprecated No need anymore */ fieldKey?: React.Key | React.Key[]; } function hasValidName(name?: NamePath): Boolean { if (name === null) { warning(false, 'Form.Item', '`null` is passed as `name` property'); } return !(name === undefined || name === null); } function genEmptyMeta(): Meta { return { errors: [], warnings: [], touched: false, validating: false, name: [], validated: false, }; } function InternalFormItem(props: FormItemProps): React.ReactElement { const { name, noStyle, className, dependencies, prefixCls: customizePrefixCls, shouldUpdate, rules, children, required, label, messageVariables, trigger = 'onChange', validateTrigger, hidden, help, } = props; const { getPrefixCls } = React.useContext(ConfigContext); const { name: formName } = React.useContext(FormContext); const isRenderProps = typeof children === 'function'; const notifyParentMetaChange = React.useContext(NoStyleItemContext); const { validateTrigger: contextValidateTrigger } = React.useContext(FieldContext); const mergedValidateTrigger = validateTrigger !== undefined ? validateTrigger : contextValidateTrigger; const hasName = hasValidName(name); const prefixCls = getPrefixCls('form', customizePrefixCls); // Style const [wrapSSR, hashId] = useStyle(prefixCls); // ========================= MISC ========================= // Get `noStyle` required info const listContext = React.useContext(ListContext); const fieldKeyPathRef = React.useRef(); // ======================== Errors ======================== // >>>>> Collect sub field errors const [subFieldErrors, setSubFieldErrors] = useFrameState>({}); // >>>>> Current field errors const [meta, setMeta] = useState(() => genEmptyMeta()); const onMetaChange = (nextMeta: Meta & { destroy?: boolean }) => { // This keyInfo is not correct when field is removed // Since origin keyManager no longer keep the origin key anymore // Which means we need cache origin one and reuse when removed const keyInfo = listContext?.getKey(nextMeta.name); // Destroy will reset all the meta setMeta(nextMeta.destroy ? genEmptyMeta() : nextMeta, true); // Bump to parent since noStyle if (noStyle && help !== false && notifyParentMetaChange) { let namePath = nextMeta.name; if (!nextMeta.destroy) { if (keyInfo !== undefined) { const [fieldKey, restPath] = keyInfo; namePath = [fieldKey, ...restPath]; fieldKeyPathRef.current = namePath; } } else { // Use origin cache data namePath = fieldKeyPathRef.current || namePath; } notifyParentMetaChange(nextMeta, namePath); } }; // >>>>> Collect noStyle Field error to the top FormItem const onSubItemMetaChange = (subMeta: Meta & { destroy: boolean }, uniqueKeys: React.Key[]) => { // Only `noStyle` sub item will trigger setSubFieldErrors((prevSubFieldErrors) => { const clone = { ...prevSubFieldErrors, }; // name: ['user', 1] + key: [4] = ['user', 4] const mergedNamePath = [...subMeta.name.slice(0, -1), ...uniqueKeys]; const mergedNameKey = mergedNamePath.join(NAME_SPLIT); if (subMeta.destroy) { // Remove delete clone[mergedNameKey]; } else { // Update clone[mergedNameKey] = subMeta; } return clone; }); }; // >>>>> Get merged errors const [mergedErrors, mergedWarnings] = React.useMemo(() => { const errorList: string[] = [...meta.errors]; const warningList: string[] = [...meta.warnings]; Object.values(subFieldErrors).forEach((subFieldError) => { errorList.push(...(subFieldError.errors || [])); warningList.push(...(subFieldError.warnings || [])); }); return [errorList, warningList]; }, [subFieldErrors, meta.errors, meta.warnings]); // ===================== Children Ref ===================== const getItemRef = useItemRef(); // ======================== Render ======================== function renderLayout( baseChildren: React.ReactNode, fieldId?: string, isRequired?: boolean, ): React.ReactNode { if (noStyle && !hidden) { return baseChildren; } return ( {baseChildren} ); } if (!hasName && !isRenderProps && !dependencies) { return wrapSSR(renderLayout(children) as JSX.Element); } let variables: Record = {}; if (typeof label === 'string') { variables.label = label; } else if (name) { variables.label = String(name); } if (messageVariables) { variables = { ...variables, ...messageVariables }; } // >>>>> With Field return wrapSSR( {(control, renderMeta, context) => { const mergedName = toArray(name).length && renderMeta ? renderMeta.name : []; const fieldId = getFieldId(mergedName, formName); const isRequired = required !== undefined ? required : !!( rules && rules.some((rule) => { if (rule && typeof rule === 'object' && rule.required && !rule.warningOnly) { return true; } if (typeof rule === 'function') { const ruleEntity = rule(context); return ruleEntity && ruleEntity.required && !ruleEntity.warningOnly; } return false; }) ); // ======================= Children ======================= const mergedControl: typeof control = { ...control, }; let childNode: React.ReactNode = null; warning( !(shouldUpdate && dependencies), 'Form.Item', "`shouldUpdate` and `dependencies` shouldn't be used together. See https://u.ant.design/form-deps.", ); if (Array.isArray(children) && hasName) { warning( false, 'Form.Item', 'A `Form.Item` with a `name` prop must have a single child element. For information on how to render more complex form items, see https://u.ant.design/complex-form-item.', ); childNode = children; } else if (isRenderProps && (!(shouldUpdate || dependencies) || hasName)) { warning( !!(shouldUpdate || dependencies), 'Form.Item', 'A `Form.Item` with a render function must have either `shouldUpdate` or `dependencies`.', ); warning( !hasName, 'Form.Item', 'A `Form.Item` with a render function cannot be a field, and thus cannot have a `name` prop.', ); } else if (dependencies && !isRenderProps && !hasName) { warning( false, 'Form.Item', 'Must set `name` or use a render function when `dependencies` is set.', ); } else if (isValidElement(children)) { warning( children.props.defaultValue === undefined, 'Form.Item', '`defaultValue` will not work on controlled Field. You should use `initialValues` of Form instead.', ); const childProps = { ...children.props, ...mergedControl }; if (!childProps.id) { childProps.id = fieldId; } if (help || mergedErrors.length > 0 || mergedWarnings.length > 0 || props.extra) { const describedbyArr = []; if (help || mergedErrors.length > 0) { describedbyArr.push(`${fieldId}_help`); } if (props.extra) { describedbyArr.push(`${fieldId}_extra`); } childProps['aria-describedby'] = describedbyArr.join(' '); } if (mergedErrors.length > 0) { childProps['aria-invalid'] = 'true'; } if (isRequired) { childProps['aria-required'] = 'true'; } if (supportRef(children)) { childProps.ref = getItemRef(mergedName, children); } // We should keep user origin event handler const triggers = new Set([ ...toArray(trigger), ...toArray(mergedValidateTrigger), ]); triggers.forEach((eventName) => { childProps[eventName] = (...args: any[]) => { mergedControl[eventName]?.(...args); children.props[eventName]?.(...args); }; }); // List of props that need to be watched for changes -> if changes are detected in MemoInput -> rerender const watchingChildProps = [ childProps['aria-required'], childProps['aria-invalid'], childProps['aria-describedby'], ]; childNode = ( {cloneElement(children, childProps)} ); } else if (isRenderProps && (shouldUpdate || dependencies) && !hasName) { childNode = children(context); } else { warning( !mergedName.length, 'Form.Item', '`name` is only used for validate React element. If you are using Form.Item as layout display, please remove `name` instead.', ); childNode = children as React.ReactNode; } return renderLayout(childNode, fieldId, isRequired); }} , ); } type InternalFormItemType = typeof InternalFormItem; type CompoundedComponent = InternalFormItemType & { useStatus: typeof useFormItemStatus; }; const FormItem = InternalFormItem as CompoundedComponent; FormItem.useStatus = useFormItemStatus; export default FormItem;