import classNames from 'classnames'; import isObject from 'lodash/isObject'; import isNumber from 'lodash/isNumber'; import isString from 'lodash/isString'; import { FormContextType, GenericObjectType, ObjectFieldTemplateProps, ObjectFieldTemplatePropertyType, RJSFSchema, StrictRJSFSchema, UiSchema, canExpand, getTemplate, getUiOptions, titleId, buttonId, } from '@rjsf/utils'; import { Col, Row, ConfigProvider } from 'antd'; import { useContext } from 'react'; /** The `ObjectFieldTemplate` is the template to use to render all the inner properties of an object along with the * title and description if available. If the object is expandable, then an `AddButton` is also rendered after all * the properties. * * @param props - The `ObjectFieldTemplateProps` for this component */ export default function ObjectFieldTemplate< T = any, S extends StrictRJSFSchema = RJSFSchema, F extends FormContextType = any, >(props: ObjectFieldTemplateProps) { const { disabled, formData, fieldPathId, onAddProperty, optionalDataControl, properties, readonly, required, registry, schema, title, uiSchema, } = props; const uiOptions = getUiOptions(uiSchema); const TitleFieldTemplate = getTemplate<'TitleFieldTemplate', T, S, F>('TitleFieldTemplate', registry, uiOptions); const { formContext } = registry; const showOptionalDataControlInTitle = !readonly && !disabled; // Button templates are not overridden in the uiSchema const { ButtonTemplates: { AddButton }, } = registry.templates; const { colSpan = 24, labelAlign = 'right', rowGutter = 24 } = formContext as GenericObjectType; const findSchema = (element: ObjectFieldTemplatePropertyType): S => element.content.props.schema; const findSchemaType = (element: ObjectFieldTemplatePropertyType) => findSchema(element).type; const findUiSchema = (element: ObjectFieldTemplatePropertyType): UiSchema | undefined => element.content.props.uiSchema; const findUiSchemaField = (element: ObjectFieldTemplatePropertyType) => getUiOptions(findUiSchema(element)).field; const findUiSchemaWidget = (element: ObjectFieldTemplatePropertyType) => getUiOptions(findUiSchema(element)).widget; const calculateColSpan = (element: ObjectFieldTemplatePropertyType) => { const type = findSchemaType(element); const field = findUiSchemaField(element); const widget = findUiSchemaWidget(element); const defaultColSpan = properties.length < 2 || // Single or no field in object. type === 'object' || type === 'array' || widget === 'textarea' ? 24 : 12; if (isObject(colSpan)) { const colSpanObj: GenericObjectType = colSpan; if (isString(widget)) { return colSpanObj[widget]; } if (isString(field)) { return colSpanObj[field]; } if (isString(type)) { return colSpanObj[type]; } } if (isNumber(colSpan)) { return colSpan; } return defaultColSpan; }; const { getPrefixCls } = useContext(ConfigProvider.ConfigContext); const prefixCls = getPrefixCls('form'); const labelClsBasic = `${prefixCls}-item-label`; const labelColClassName = classNames( labelClsBasic, labelAlign === 'left' && `${labelClsBasic}-left`, // labelCol.className, ); return (
{title && ( )} {!showOptionalDataControlInTitle ? {optionalDataControl} : undefined} {properties .filter((e) => !e.hidden) .map((element: ObjectFieldTemplatePropertyType) => ( {element.content} ))} {canExpand(schema, uiSchema, formData) && ( )}
); }