Dynamic forms (eav)
Dynamic form instances (values) are saved in the individual tables according to the entity which they are linked to ⇒ which is their owner (e.g. the entity IdmIdentityFormValue, IdmRoleFormValue). Form values are not saved, if null value (by persistent type) is given ⇒ filled values are saved only.
FormService service for working with the extended attributes on the back-end. FormInstance utility is useful on BE - contains value transformation to maps by attributes etc.. Use this service in your custom module, benefits:- single autowired service for work with definitions, attributes and values
- cache
core:form-definition-cachefor loading form definitions (with attributes) is effective here.
EavForm.
FormService#saveAttribute), then event EAV\_SAVE is not published. Save all attributes (FormService#saveValues), if publishing event EAV_SAVE for owner is needed.
Dynamic form attribute supports data types (persistentType):
CHAR- one characterTEXT- strings (long text). Searching byTEXTis not supported, column is not indexed -SHORTTEXTusage is preferred (+ indexed).SHORTTEXT- strings (2000 chars). Indexed.INT- integerLONG- longDOUBLE- saved as bigdecimalBOOLEAN- true / false / nullDATE- date (without time)DATETIME- date with timeBYTEARRAY- byte[]UUID- uuid identifier. Indexed.ATTACHMENT- attachment (~binary file). Read more about attachments.CODELIST- referenced code list - persists items "code" into short text. Uses face type as code list code.ENUMERATIONreferenced frontend enumeration - persists items "code" into short text. Uses face type as enumeration name.
persistentType and confidential is possible only for attributes without persisted values ⇒ when attribute is not used for some values. Data migration, when attribute's persistentType or confidential is changed is not supported now.
with properties:
readonlymulti values- Is represented on the front-end by a textarea, where a line is a value (a new line separates the values). This property is supported for persistent typesCHAR,TEXT,INT,LONG,DOUBLEandUUID.confidential- .The values are stored in an confidential storage). Stored values of these attributes - substitute characters only - are loaded on the front-end. The value can only be changed and determined whether it is filled in. This property is supported for persistent typesCHAR,TEXT,INT,LONG,DOUBLE,UUID,BYTEARRAY.required- value validation, read more.unique- value validation, read more.min- value validation, read more.max- value validation, read more.regex- value validation, read more.validationMessage- custom message, when some validation fails, read more.
Dynamic form attributes can be rendered differently on frontend. Face type (faceType) property is used for choosing frontend renderer. The default renderer is chosen by persistent type (e.g. UUID → UUID).
Renderer is a frontend component, superclass component AbstractFormAttributeRenderer is used for all renderer implementations. Renderer is responsible for IdmFormValue ⇔ input value transformation.
Renderers are registered in module's component-descriptor.js as single component with attributes:
id- unique component identifiertype=form-value- static component type is used for allform-valuerendererpersistentType- which persistent type renderer supportsfaceType- renderer face type ⇒ key. Unique face type should be given (by persistent type). Its optionalpersistentTypeis used as default, when nofaceTypeis given.component- renderer implementation (AbstractFormAttributeRendererdescendant).labelKey- localization key ⇒ renderer name. Its optional,faceTypeis used, when nolabelKeyis given.
All component descriptor features are supported. Read tutorial, how to create custom form attribute renderer.
Custom configuration can be added to registered renderers (@since CzechIdM 10.8.0) - use AbstractFormAttributeRenderer on backend to define additional renderer properties.
Adding the support of extended attributes for a new entity
Backend
- Adding an interface implementation
FormableEntityto the new entity, - creating a manager implementation by inheriting
AbstractFormableServicefor the new entity, - creating the entity by inheriting
AbstractFormValue, which will represent the values of extended attributes for the new entity (owner), - creating a repository by inheriting
AbstractFormValueRepositoryfor the values of the extended attributes, - creating the manager by inheriting
AbstractFormValueServicefor the values of the extended attributes.
Frontend
- issuing a REST endpoint for saving the extended attributes from the FE - e.g.
IdmIdentityController- controller has to evaluate security to read / save form values by their owner (e.g. by identity), - creating a service and redux manager communicating with REST by inheriting
FormableEntityManager- e.g.IdentityManager, - using the component
EavFormfor filling in and sending the values of the extended attributes from the FE to the BE - e.g.IdentityEavcontent.
Agenda for working with forms
On the FE, there is an agenda of forms - their definition and attributes. Each definition can contain zero or more attributes. To maintain the integrity, an interface UnmodifiableEntity has been created, which allows adding a flag that the entity has been created by the system and cannot be modified (or some of its attributes) and deleted (this logic now needs to be implemented manually into the relevant controllers), for example in IdmFormAttributeController.
persistentType or confidential is changed is not supported now.
Localization
spinal-case or kebab-case on frontend).
Example form code eu.bcvsolutions.idm.acc.entity.SysSystem will be transformed into eu-bcvsolutions-idm-acc-entity-syssystem
Read more in tutorial
Validation
For form attribute values is possible to configure prepared validations. Validation are evaluated (on the backend), when form with extended attributes is saved and sent to backend. Simple validations as required, min, max are evaluated on frontend after value is changed.
Required
Value is required.
Unique
Value has to be unique.
BYTEARRAY and ATTACHMENT persistent types.
Min, Max
Value has to be greater than (lesser than) or equal given min (max) values. Real number (38,4) can be configured.
DOUBLE, INT, LONG persistent types.
Regex
Value has to match given regular expression (java pattern is used).
BYTEARRAY and ATTACHMENT persistent types.
- GUI:
^([01]?\d\d?|2[0-4]\d|25[0-5])\.([01]?\d\d?|2[0-4]\d|25[0-5])\.([01]?\d\d?|2[0-4]\d|25[0-5])\.([01]?\d\d?|2[0-4]\d|25[0-5])$ - java:
^([01]?\\d\\d?|2[0-4]\\d|25[0-5])\\.([01]?\\d\\d?|2[0-4]\\d|25[0-5])\\.([01]?\\d\\d?|2[0-4]\\d|25[0-5])\\.([01]?\\d\\d?|2[0-4]\\d|25[0-5])$
Validation message
Custom validation message. If message is not defined, then default message by invalid validation type will be shown.
core:validationError.invalid.unique). Parameters min, max, regex, unique, required is available for localization message.
Code lists
CodeListManager for creating and providing code lists and items through application on backend (e.g. available for scripts, which could be used in provisioning).
CodeListSelect and CodeListValue for rendering code lists on frontend.
environment.development.title will be localized.
Authorization policies support
Identity form values can be secured by authorization policies when some identity extended attributes have to be secured - see how in configure authorization policies.
Future development
- Form value data migration, when persistent type is changed.
- Attachment renderer: support multiple files, validation support (now is validation on input)
- Created deep copy, when form values are copied ⇒ attachment is linked to two form values and is removed, when the first one is deleted.
- #1874: Support unique validation for multivalued eav attributes
