@yuuvis/client-framework 0.6.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +62 -0
- package/actions/README.md +3 -0
- package/actions/index.d.ts +6 -0
- package/actions/lib/actions/copy-action/copy-action.d.ts +17 -0
- package/actions/lib/actions/cut-action/cut-action.d.ts +17 -0
- package/actions/lib/actions/delete-action/delete/delete.component.d.ts +22 -0
- package/actions/lib/actions/delete-action/delete-action.d.ts +17 -0
- package/actions/lib/actions/download-action/download-action.d.ts +17 -0
- package/actions/lib/actions.icon.d.ts +7 -0
- package/actions/lib/actions.interface.d.ts +86 -0
- package/actions/lib/actions.module.d.ts +8 -0
- package/actions/lib/actions.service.d.ts +38 -0
- package/actions/lib/components/contextmenu/contextmenu.component.d.ts +11 -0
- package/app-bar/README.md +3 -0
- package/app-bar/index.d.ts +1 -0
- package/app-bar/lib/app-bar.component.d.ts +47 -0
- package/clipboard/README.md +3 -0
- package/clipboard/index.d.ts +1 -0
- package/clipboard/lib/clipboard.component.d.ts +16 -0
- package/common/README.md +3 -0
- package/common/index.d.ts +5 -0
- package/common/lib/components/focus-indicator/focus-indicator.component.d.ts +7 -0
- package/common/lib/components/token-input/token-input.component.d.ts +55 -0
- package/common/lib/components/token-input/token-input.interface.d.ts +8 -0
- package/common/lib/directives/busy-overlay.directive.d.ts +25 -0
- package/common/lib/directives/click-double.directive.d.ts +17 -0
- package/common/lib/directives/container-size.directive.d.ts +30 -0
- package/common/lib/directives/drag-select.directive.d.ts +21 -0
- package/common/lib/directives/file-drop-zone/file-drop-zone.directive.d.ts +18 -0
- package/common/lib/directives/file-drop-zone/file-drop-zone.interface.d.ts +5 -0
- package/common/lib/directives/focus-within.directive.d.ts +36 -0
- package/common/lib/directives/index.d.ts +8 -0
- package/common/lib/directives/light-dismiss.directive.d.ts +19 -0
- package/common/lib/directives/longpress.directive.d.ts +13 -0
- package/common/lib/directives/noop-value-accessor.directive.d.ts +10 -0
- package/esm2022/actions/index.mjs +7 -0
- package/esm2022/actions/lib/actions/copy-action/copy-action.mjs +31 -0
- package/esm2022/actions/lib/actions/cut-action/cut-action.mjs +30 -0
- package/esm2022/actions/lib/actions/delete-action/delete/delete.component.mjs +67 -0
- package/esm2022/actions/lib/actions/delete-action/delete-action.mjs +44 -0
- package/esm2022/actions/lib/actions/download-action/download-action.mjs +33 -0
- package/esm2022/actions/lib/actions.icon.mjs +8 -0
- package/esm2022/actions/lib/actions.interface.mjs +19 -0
- package/esm2022/actions/lib/actions.module.mjs +16 -0
- package/esm2022/actions/lib/actions.service.mjs +89 -0
- package/esm2022/actions/lib/components/contextmenu/contextmenu.component.mjs +27 -0
- package/esm2022/actions/yuuvis-client-framework-actions.mjs +5 -0
- package/esm2022/app-bar/index.mjs +2 -0
- package/esm2022/app-bar/lib/app-bar.component.mjs +89 -0
- package/esm2022/app-bar/yuuvis-client-framework-app-bar.mjs +5 -0
- package/esm2022/clipboard/index.mjs +2 -0
- package/esm2022/clipboard/lib/clipboard.component.mjs +48 -0
- package/esm2022/clipboard/yuuvis-client-framework-clipboard.mjs +5 -0
- package/esm2022/common/index.mjs +6 -0
- package/esm2022/common/lib/components/focus-indicator/focus-indicator.component.mjs +41 -0
- package/esm2022/common/lib/components/token-input/token-input.component.mjs +324 -0
- package/esm2022/common/lib/components/token-input/token-input.interface.mjs +2 -0
- package/esm2022/common/lib/directives/busy-overlay.directive.mjs +88 -0
- package/esm2022/common/lib/directives/click-double.directive.mjs +61 -0
- package/esm2022/common/lib/directives/container-size.directive.mjs +56 -0
- package/esm2022/common/lib/directives/drag-select.directive.mjs +114 -0
- package/esm2022/common/lib/directives/file-drop-zone/file-drop-zone.directive.mjs +153 -0
- package/esm2022/common/lib/directives/file-drop-zone/file-drop-zone.interface.mjs +2 -0
- package/esm2022/common/lib/directives/focus-within.directive.mjs +81 -0
- package/esm2022/common/lib/directives/index.mjs +9 -0
- package/esm2022/common/lib/directives/light-dismiss.directive.mjs +44 -0
- package/esm2022/common/lib/directives/longpress.directive.mjs +36 -0
- package/esm2022/common/lib/directives/noop-value-accessor.directive.mjs +42 -0
- package/esm2022/common/yuuvis-client-framework-common.mjs +5 -0
- package/esm2022/forms/index.mjs +19 -0
- package/esm2022/forms/lib/elements/catalog/catalog.component.mjs +118 -0
- package/esm2022/forms/lib/elements/datetime/datetime.component.mjs +82 -0
- package/esm2022/forms/lib/elements/datetime-range/datetime-range.component.mjs +166 -0
- package/esm2022/forms/lib/elements/number/number.component.mjs +195 -0
- package/esm2022/forms/lib/elements/number-range/number-range.component.mjs +176 -0
- package/esm2022/forms/lib/elements/organization/organization.component.mjs +268 -0
- package/esm2022/forms/lib/elements/string/string.component.mjs +254 -0
- package/esm2022/forms/lib/form-input/form-input.component.mjs +88 -0
- package/esm2022/forms/lib/forms.module.mjs +58 -0
- package/esm2022/forms/yuuvis-client-framework-forms.mjs +5 -0
- package/esm2022/icons/index.mjs +4 -0
- package/esm2022/icons/lib/icon.service.mjs +59 -0
- package/esm2022/icons/lib/icons.mjs +31 -0
- package/esm2022/icons/lib/object-type-icon/object-type-icon.component.mjs +29 -0
- package/esm2022/icons/yuuvis-client-framework-icons.mjs +5 -0
- package/esm2022/index.mjs +2 -0
- package/esm2022/lib/yuuvis-client-framework.module.mjs +15 -0
- package/esm2022/list/index.mjs +3 -0
- package/esm2022/list/lib/list-item.directive.mjs +72 -0
- package/esm2022/list/lib/list.component.mjs +133 -0
- package/esm2022/list/yuuvis-client-framework-list.mjs +5 -0
- package/esm2022/metadata-form/index.mjs +4 -0
- package/esm2022/metadata-form/lib/metadata-default-templates/metadata-default-templates.component.mjs +31 -0
- package/esm2022/metadata-form/lib/metadata-form-element-registry.service.mjs +99 -0
- package/esm2022/metadata-form/lib/metadata-form-field/metadata-form-field.component.mjs +126 -0
- package/esm2022/metadata-form/lib/object-metadata-element-template.directive.mjs +52 -0
- package/esm2022/metadata-form/yuuvis-client-framework-metadata-form.mjs +5 -0
- package/esm2022/object-details/index.mjs +7 -0
- package/esm2022/object-details/lib/object-audit/object-audit.component.mjs +195 -0
- package/esm2022/object-details/lib/object-details-shell/object-details-shell.component.mjs +127 -0
- package/esm2022/object-details/lib/object-details.component.mjs +61 -0
- package/esm2022/object-details/lib/object-metadata/form-section-group.pipe.mjs +17 -0
- package/esm2022/object-details/lib/object-metadata/object-metadata.component.mjs +201 -0
- package/esm2022/object-details/lib/object-metadata/object-metadata.interface.mjs +2 -0
- package/esm2022/object-details/yuuvis-client-framework-object-details.mjs +5 -0
- package/esm2022/object-flavor/index.mjs +6 -0
- package/esm2022/object-flavor/lib/abstract-apply-create-flavor/abstract-apply-create-flavor.component.mjs +25 -0
- package/esm2022/object-flavor/lib/abstract-apply-object-flavor/abstract-apply-object-flavor.component.mjs +21 -0
- package/esm2022/object-flavor/lib/flavor-chip/flavor-chip.component.mjs +35 -0
- package/esm2022/object-flavor/lib/object-flavor/object-flavor.component.mjs +93 -0
- package/esm2022/object-flavor/lib/object-flavor-picker/object-flavor-picker.component.mjs +43 -0
- package/esm2022/object-flavor/yuuvis-client-framework-object-flavor.mjs +5 -0
- package/esm2022/object-form/index.mjs +5 -0
- package/esm2022/object-form/lib/form-scripting.api.interface.mjs +2 -0
- package/esm2022/object-form/lib/form-scripting.service.mjs +160 -0
- package/esm2022/object-form/lib/object-form-element/object-form-element.component.mjs +87 -0
- package/esm2022/object-form/lib/object-form-extension.interface.mjs +36 -0
- package/esm2022/object-form/lib/object-form-group/object-form-group.component.mjs +85 -0
- package/esm2022/object-form/lib/object-form-script/form-scripting-element-extension/form-scripting-element-extension.component.mjs +23 -0
- package/esm2022/object-form/lib/object-form-script/object-form-script.service.mjs +115 -0
- package/esm2022/object-form/lib/object-form-script/object-form-scripting-scope.mjs +251 -0
- package/esm2022/object-form/lib/object-form-translate.service.mjs +73 -0
- package/esm2022/object-form/lib/object-form.component.mjs +627 -0
- package/esm2022/object-form/lib/object-form.interface.mjs +9 -0
- package/esm2022/object-form/lib/object-form.model.mjs +20 -0
- package/esm2022/object-form/lib/object-form.service.mjs +127 -0
- package/esm2022/object-form/lib/object-form.utils.mjs +55 -0
- package/esm2022/object-form/lib/object-form.validation.mjs +48 -0
- package/esm2022/object-form/yuuvis-client-framework-object-form.mjs +5 -0
- package/esm2022/object-preview/index.mjs +3 -0
- package/esm2022/object-preview/lib/components/index.mjs +3 -0
- package/esm2022/object-preview/lib/components/object-email-preview/object-email-preview.component.mjs +45 -0
- package/esm2022/object-preview/lib/components/object-preview/object-preview.component.mjs +78 -0
- package/esm2022/object-preview/lib/services/object-preview.service.mjs +92 -0
- package/esm2022/object-preview/yuuvis-client-framework-object-preview.mjs +5 -0
- package/esm2022/object-summary/index.mjs +5 -0
- package/esm2022/object-summary/lib/multi-object-summary/multi-object-summary.component.mjs +33 -0
- package/esm2022/object-summary/lib/object-summary/object-summary.component.mjs +273 -0
- package/esm2022/object-summary/lib/object-summary-data/object-summary-data.component.mjs +80 -0
- package/esm2022/object-summary/lib/object-summary.module.mjs +15 -0
- package/esm2022/object-summary/yuuvis-client-framework-object-summary.mjs +5 -0
- package/esm2022/pagination/index.mjs +3 -0
- package/esm2022/pagination/lib/pagination.component.mjs +48 -0
- package/esm2022/pagination/lib/pagination.interface.mjs +2 -0
- package/esm2022/pagination/yuuvis-client-framework-pagination.mjs +5 -0
- package/esm2022/panel/index.mjs +2 -0
- package/esm2022/panel/lib/panel.component.mjs +20 -0
- package/esm2022/panel/yuuvis-client-framework-panel.mjs +5 -0
- package/esm2022/renderer/index.mjs +11 -0
- package/esm2022/renderer/lib/property-renderer/abstract.renderer.mjs +29 -0
- package/esm2022/renderer/lib/property-renderer/datetime.renderer.mjs +13 -0
- package/esm2022/renderer/lib/property-renderer/decimal.renderer.component.mjs +12 -0
- package/esm2022/renderer/lib/property-renderer/filesize.renderer.component.mjs +28 -0
- package/esm2022/renderer/lib/property-renderer/icon.renderer.component.mjs +23 -0
- package/esm2022/renderer/lib/property-renderer/integer.renderer.component.mjs +12 -0
- package/esm2022/renderer/lib/property-renderer/organization.renderer.mjs +19 -0
- package/esm2022/renderer/lib/property-renderer/string.renderer.component.mjs +12 -0
- package/esm2022/renderer/lib/property-renderer/unknown.renderer.mjs +12 -0
- package/esm2022/renderer/lib/renderer.directive.mjs +51 -0
- package/esm2022/renderer/lib/services/renderer/renderer.interface.mjs +2 -0
- package/esm2022/renderer/lib/services/renderer/renderer.service.mjs +84 -0
- package/esm2022/renderer/yuuvis-client-framework-renderer.mjs +5 -0
- package/esm2022/sequence-list/index.mjs +6 -0
- package/esm2022/sequence-list/lib/due-date-picker/due-date-picker.component.mjs +99 -0
- package/esm2022/sequence-list/lib/sequence-list-template-manage/sequence-list-template-manage.component.mjs +183 -0
- package/esm2022/sequence-list/lib/sequence-list-templates/sequence-list-templates.component.mjs +114 -0
- package/esm2022/sequence-list/lib/sequence-list.component.mjs +146 -0
- package/esm2022/sequence-list/lib/sequence-list.interface.mjs +2 -0
- package/esm2022/sequence-list/yuuvis-client-framework-sequence-list.mjs +5 -0
- package/esm2022/simple-search/index.mjs +3 -0
- package/esm2022/simple-search/lib/simple-search/simple-search.component.mjs +111 -0
- package/esm2022/simple-search/lib/simple-search/simple-search.interface.mjs +2 -0
- package/esm2022/simple-search/yuuvis-client-framework-simple-search.mjs +5 -0
- package/esm2022/tile-list/index.mjs +11 -0
- package/esm2022/tile-list/lib/tile/tile.component.mjs +53 -0
- package/esm2022/tile-list/lib/tile-config/action-select/action-select.component.mjs +22 -0
- package/esm2022/tile-list/lib/tile-config/icon-select/icon-select.component.mjs +33 -0
- package/esm2022/tile-list/lib/tile-config/property-select/property-select.component.mjs +91 -0
- package/esm2022/tile-list/lib/tile-config/tile-config-tile/tile-config-tile.component.mjs +66 -0
- package/esm2022/tile-list/lib/tile-config/tile-config-trigger/tile-config-trigger.component.mjs +46 -0
- package/esm2022/tile-list/lib/tile-config/tile-config.component.mjs +156 -0
- package/esm2022/tile-list/lib/tile-extension/directive/tile-extension.directive.mjs +37 -0
- package/esm2022/tile-list/lib/tile-extension/extensions/email.extension.mjs +42 -0
- package/esm2022/tile-list/lib/tile-extension/tile-extension.service.mjs +35 -0
- package/esm2022/tile-list/lib/tile-list/tile-list.component.mjs +510 -0
- package/esm2022/tile-list/lib/tile-list/tile-list.interface.mjs +2 -0
- package/esm2022/tile-list/yuuvis-client-framework-tile-list.mjs +5 -0
- package/esm2022/token-search/index.mjs +3 -0
- package/esm2022/token-search/token-search.component.mjs +78 -0
- package/esm2022/token-search/token-search.interface.mjs +2 -0
- package/esm2022/token-search/yuuvis-client-framework-token-search.mjs +5 -0
- package/esm2022/tree/index.mjs +3 -0
- package/esm2022/tree/lib/tree-node/tree-node.component.mjs +65 -0
- package/esm2022/tree/lib/tree.component.mjs +148 -0
- package/esm2022/tree/lib/tree.interface.mjs +2 -0
- package/esm2022/tree/lib/tree.service.mjs +95 -0
- package/esm2022/tree/yuuvis-client-framework-tree.mjs +5 -0
- package/esm2022/upload-progress/index.mjs +2 -0
- package/esm2022/upload-progress/lib/upload-progress/upload-progress-overlay/upload-progress-overlay.component.mjs +49 -0
- package/esm2022/upload-progress/lib/upload-progress/upload-progress.component.mjs +37 -0
- package/esm2022/upload-progress/yuuvis-client-framework-upload-progress.mjs +5 -0
- package/esm2022/user-avatar/index.mjs +3 -0
- package/esm2022/user-avatar/lib/user-avatar.component.mjs +69 -0
- package/esm2022/user-avatar/lib/user-avatar.module.mjs +24 -0
- package/esm2022/user-avatar/yuuvis-client-framework-user-avatar.mjs +5 -0
- package/esm2022/yuuvis-client-framework.mjs +5 -0
- package/fesm2022/yuuvis-client-framework-actions.mjs +333 -0
- package/fesm2022/yuuvis-client-framework-actions.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-app-bar.mjs +96 -0
- package/fesm2022/yuuvis-client-framework-app-bar.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-clipboard.mjs +55 -0
- package/fesm2022/yuuvis-client-framework-clipboard.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-common.mjs +1020 -0
- package/fesm2022/yuuvis-client-framework-common.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-forms.mjs +1355 -0
- package/fesm2022/yuuvis-client-framework-forms.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-icons.mjs +123 -0
- package/fesm2022/yuuvis-client-framework-icons.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-list.mjs +209 -0
- package/fesm2022/yuuvis-client-framework-list.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-metadata-form.mjs +302 -0
- package/fesm2022/yuuvis-client-framework-metadata-form.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-object-details.mjs +583 -0
- package/fesm2022/yuuvis-client-framework-object-details.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-object-flavor.mjs +200 -0
- package/fesm2022/yuuvis-client-framework-object-flavor.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-object-form.mjs +1664 -0
- package/fesm2022/yuuvis-client-framework-object-form.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-object-preview.mjs +213 -0
- package/fesm2022/yuuvis-client-framework-object-preview.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-object-summary.mjs +397 -0
- package/fesm2022/yuuvis-client-framework-object-summary.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-pagination.mjs +55 -0
- package/fesm2022/yuuvis-client-framework-pagination.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-panel.mjs +27 -0
- package/fesm2022/yuuvis-client-framework-panel.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-renderer.mjs +262 -0
- package/fesm2022/yuuvis-client-framework-renderer.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-sequence-list.mjs +519 -0
- package/fesm2022/yuuvis-client-framework-sequence-list.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-simple-search.mjs +118 -0
- package/fesm2022/yuuvis-client-framework-simple-search.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-tile-list.mjs +1036 -0
- package/fesm2022/yuuvis-client-framework-tile-list.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-token-search.mjs +85 -0
- package/fesm2022/yuuvis-client-framework-token-search.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-tree.mjs +307 -0
- package/fesm2022/yuuvis-client-framework-tree.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-upload-progress.mjs +84 -0
- package/fesm2022/yuuvis-client-framework-upload-progress.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework-user-avatar.mjs +96 -0
- package/fesm2022/yuuvis-client-framework-user-avatar.mjs.map +1 -0
- package/fesm2022/yuuvis-client-framework.mjs +22 -0
- package/fesm2022/yuuvis-client-framework.mjs.map +1 -0
- package/forms/README.md +3 -0
- package/forms/index.d.ts +9 -0
- package/forms/lib/elements/catalog/catalog.component.d.ts +68 -0
- package/forms/lib/elements/datetime/datetime.component.d.ts +32 -0
- package/forms/lib/elements/datetime-range/datetime-range.component.d.ts +51 -0
- package/forms/lib/elements/number/number.component.d.ts +77 -0
- package/forms/lib/elements/number-range/number-range.component.d.ts +66 -0
- package/forms/lib/elements/organization/organization.component.d.ts +93 -0
- package/forms/lib/elements/string/string.component.d.ts +104 -0
- package/forms/lib/form-input/form-input.component.d.ts +48 -0
- package/forms/lib/forms.module.d.ts +16 -0
- package/icons/README.md +5 -0
- package/icons/index.d.ts +3 -0
- package/icons/lib/icon.service.d.ts +38 -0
- package/icons/lib/icons.d.ts +1 -0
- package/icons/lib/object-type-icon/object-type-icon.component.d.ts +11 -0
- package/index.d.ts +1 -0
- package/lib/yuuvis-client-framework.module.d.ts +7 -0
- package/list/README.md +3 -0
- package/list/index.d.ts +2 -0
- package/list/lib/list-item.directive.d.ts +15 -0
- package/list/lib/list.component.d.ts +36 -0
- package/metadata-form/README.md +21 -0
- package/metadata-form/index.d.ts +3 -0
- package/metadata-form/lib/metadata-default-templates/metadata-default-templates.component.d.ts +5 -0
- package/metadata-form/lib/metadata-form-element-registry.service.d.ts +47 -0
- package/metadata-form/lib/metadata-form-field/metadata-form-field.component.d.ts +31 -0
- package/metadata-form/lib/object-metadata-element-template.directive.d.ts +27 -0
- package/object-details/README.md +28 -0
- package/object-details/index.d.ts +6 -0
- package/object-details/lib/object-audit/object-audit.component.d.ts +59 -0
- package/object-details/lib/object-details-shell/object-details-shell.component.d.ts +51 -0
- package/object-details/lib/object-details.component.d.ts +36 -0
- package/object-details/lib/object-metadata/form-section-group.pipe.d.ts +8 -0
- package/object-details/lib/object-metadata/object-metadata.component.d.ts +48 -0
- package/object-details/lib/object-metadata/object-metadata.interface.d.ts +26 -0
- package/object-flavor/README.md +3 -0
- package/object-flavor/index.d.ts +5 -0
- package/object-flavor/lib/abstract-apply-create-flavor/abstract-apply-create-flavor.component.d.ts +12 -0
- package/object-flavor/lib/abstract-apply-object-flavor/abstract-apply-object-flavor.component.d.ts +12 -0
- package/object-flavor/lib/flavor-chip/flavor-chip.component.d.ts +17 -0
- package/object-flavor/lib/object-flavor/object-flavor.component.d.ts +26 -0
- package/object-flavor/lib/object-flavor-picker/object-flavor-picker.component.d.ts +10 -0
- package/object-form/README.md +3 -0
- package/object-form/index.d.ts +4 -0
- package/object-form/lib/form-scripting.api.interface.d.ts +131 -0
- package/object-form/lib/form-scripting.service.d.ts +30 -0
- package/object-form/lib/object-form-element/object-form-element.component.d.ts +22 -0
- package/object-form/lib/object-form-extension.interface.d.ts +22 -0
- package/object-form/lib/object-form-group/object-form-group.component.d.ts +22 -0
- package/object-form/lib/object-form-script/form-scripting-element-extension/form-scripting-element-extension.component.d.ts +10 -0
- package/object-form/lib/object-form-script/object-form-script.service.d.ts +45 -0
- package/object-form/lib/object-form-script/object-form-scripting-scope.d.ts +50 -0
- package/object-form/lib/object-form-translate.service.d.ts +15 -0
- package/object-form/lib/object-form.component.d.ts +60 -0
- package/object-form/lib/object-form.interface.d.ts +113 -0
- package/object-form/lib/object-form.model.d.ts +18 -0
- package/object-form/lib/object-form.service.d.ts +39 -0
- package/object-form/lib/object-form.utils.d.ts +20 -0
- package/object-form/lib/object-form.validation.d.ts +21 -0
- package/object-preview/README.md +3 -0
- package/object-preview/index.d.ts +2 -0
- package/object-preview/lib/components/index.d.ts +2 -0
- package/object-preview/lib/components/object-email-preview/object-email-preview.component.d.ts +16 -0
- package/object-preview/lib/components/object-preview/object-preview.component.d.ts +13 -0
- package/object-preview/lib/services/object-preview.service.d.ts +37 -0
- package/object-summary/README.md +3 -0
- package/object-summary/index.d.ts +4 -0
- package/object-summary/lib/multi-object-summary/multi-object-summary.component.d.ts +7 -0
- package/object-summary/lib/object-summary/object-summary.component.d.ts +73 -0
- package/object-summary/lib/object-summary-data/object-summary-data.component.d.ts +11 -0
- package/object-summary/lib/object-summary.module.d.ts +7 -0
- package/package.json +173 -0
- package/pagination/README.md +3 -0
- package/pagination/index.d.ts +2 -0
- package/pagination/lib/pagination.component.d.ts +18 -0
- package/pagination/lib/pagination.interface.d.ts +5 -0
- package/panel/README.md +3 -0
- package/panel/index.d.ts +1 -0
- package/panel/lib/panel.component.d.ts +12 -0
- package/renderer/README.md +5 -0
- package/renderer/index.d.ts +10 -0
- package/renderer/lib/property-renderer/abstract.renderer.d.ts +14 -0
- package/renderer/lib/property-renderer/datetime.renderer.d.ts +6 -0
- package/renderer/lib/property-renderer/decimal.renderer.component.d.ts +6 -0
- package/renderer/lib/property-renderer/filesize.renderer.component.d.ts +7 -0
- package/renderer/lib/property-renderer/icon.renderer.component.d.ts +6 -0
- package/renderer/lib/property-renderer/integer.renderer.component.d.ts +6 -0
- package/renderer/lib/property-renderer/organization.renderer.d.ts +7 -0
- package/renderer/lib/property-renderer/string.renderer.component.d.ts +6 -0
- package/renderer/lib/property-renderer/unknown.renderer.d.ts +6 -0
- package/renderer/lib/renderer.directive.d.ts +17 -0
- package/renderer/lib/services/renderer/renderer.interface.d.ts +8 -0
- package/renderer/lib/services/renderer/renderer.service.d.ts +35 -0
- package/sequence-list/README.md +3 -0
- package/sequence-list/index.d.ts +5 -0
- package/sequence-list/lib/due-date-picker/due-date-picker.component.d.ts +28 -0
- package/sequence-list/lib/sequence-list-template-manage/sequence-list-template-manage.component.d.ts +52 -0
- package/sequence-list/lib/sequence-list-templates/sequence-list-templates.component.d.ts +36 -0
- package/sequence-list/lib/sequence-list.component.d.ts +43 -0
- package/sequence-list/lib/sequence-list.interface.d.ts +16 -0
- package/simple-search/README.md +3 -0
- package/simple-search/index.d.ts +2 -0
- package/simple-search/lib/simple-search/simple-search.component.d.ts +55 -0
- package/simple-search/lib/simple-search/simple-search.interface.d.ts +4 -0
- package/styles/client-framework.scss +27 -0
- package/tile-list/README.md +3 -0
- package/tile-list/index.d.ts +10 -0
- package/tile-list/lib/tile/tile.component.d.ts +22 -0
- package/tile-list/lib/tile-config/action-select/action-select.component.d.ts +13 -0
- package/tile-list/lib/tile-config/icon-select/icon-select.component.d.ts +11 -0
- package/tile-list/lib/tile-config/property-select/property-select.component.d.ts +25 -0
- package/tile-list/lib/tile-config/tile-config-tile/tile-config-tile.component.d.ts +20 -0
- package/tile-list/lib/tile-config/tile-config-trigger/tile-config-trigger.component.d.ts +17 -0
- package/tile-list/lib/tile-config/tile-config.component.d.ts +47 -0
- package/tile-list/lib/tile-extension/directive/tile-extension.directive.d.ts +11 -0
- package/tile-list/lib/tile-extension/extensions/email.extension.d.ts +10 -0
- package/tile-list/lib/tile-extension/tile-extension.service.d.ts +16 -0
- package/tile-list/lib/tile-list/tile-list.component.d.ts +116 -0
- package/tile-list/lib/tile-list/tile-list.interface.d.ts +18 -0
- package/token-search/README.md +3 -0
- package/token-search/index.d.ts +2 -0
- package/token-search/token-search.component.d.ts +22 -0
- package/token-search/token-search.interface.d.ts +4 -0
- package/tree/README.md +3 -0
- package/tree/index.d.ts +2 -0
- package/tree/lib/tree-node/tree-node.component.d.ts +21 -0
- package/tree/lib/tree.component.d.ts +53 -0
- package/tree/lib/tree.interface.d.ts +11 -0
- package/tree/lib/tree.service.d.ts +35 -0
- package/upload-progress/README.md +3 -0
- package/upload-progress/index.d.ts +1 -0
- package/upload-progress/lib/upload-progress/upload-progress-overlay/upload-progress-overlay.component.d.ts +17 -0
- package/upload-progress/lib/upload-progress/upload-progress.component.d.ts +16 -0
- package/user-avatar/README.md +3 -0
- package/user-avatar/index.d.ts +2 -0
- package/user-avatar/lib/user-avatar.component.d.ts +44 -0
- package/user-avatar/lib/user-avatar.module.d.ts +8 -0
|
@@ -0,0 +1,627 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { ChangeDetectorRef, Component, effect, ElementRef, inject, input, output, ViewEncapsulation } from '@angular/core';
|
|
3
|
+
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
4
|
+
import { ReactiveFormsModule, Validators } from '@angular/forms';
|
|
5
|
+
import { Situation, SystemService, UserService, Utils } from '@yuuvis/client-core';
|
|
6
|
+
import { debounceTime, of } from 'rxjs';
|
|
7
|
+
import { FormScriptingApiService } from './form-scripting.service';
|
|
8
|
+
import { ObjectFormGroupComponent } from './object-form-group/object-form-group.component';
|
|
9
|
+
import { ObjectFormScriptService } from './object-form-script/object-form-script.service';
|
|
10
|
+
import { ObjectFormScriptingScope } from './object-form-script/object-form-scripting-scope';
|
|
11
|
+
import { ObjectFormControlWrapper } from './object-form.interface';
|
|
12
|
+
import { ObjectFormControl, ObjectFormGroup } from './object-form.model';
|
|
13
|
+
import { ObjectFormService } from './object-form.service';
|
|
14
|
+
import { ObjectFormUtils } from './object-form.utils';
|
|
15
|
+
import { FormValidation } from './object-form.validation';
|
|
16
|
+
import * as i0 from "@angular/core";
|
|
17
|
+
import * as i1 from "@angular/forms";
|
|
18
|
+
export class ObjectFormComponent {
|
|
19
|
+
#elementRef;
|
|
20
|
+
#systemService;
|
|
21
|
+
#formScriptService;
|
|
22
|
+
#formHelperService;
|
|
23
|
+
#userService;
|
|
24
|
+
#cdRef;
|
|
25
|
+
#formScriptingApiService;
|
|
26
|
+
#skipTranslationsFor;
|
|
27
|
+
#namePrefix;
|
|
28
|
+
#formOptionsEffect;
|
|
29
|
+
#elementExtensionsEffect;
|
|
30
|
+
#scriptModel;
|
|
31
|
+
#scriptingScope;
|
|
32
|
+
#subscriptions;
|
|
33
|
+
// local store for all the form control references
|
|
34
|
+
#formControls;
|
|
35
|
+
#initialValidators;
|
|
36
|
+
constructor() {
|
|
37
|
+
this.#elementRef = inject(ElementRef);
|
|
38
|
+
this.#systemService = inject(SystemService);
|
|
39
|
+
this.#formScriptService = inject(ObjectFormScriptService);
|
|
40
|
+
this.#formHelperService = inject(ObjectFormService);
|
|
41
|
+
this.#userService = inject(UserService);
|
|
42
|
+
this.#cdRef = inject(ChangeDetectorRef);
|
|
43
|
+
this.#formScriptingApiService = inject(FormScriptingApiService);
|
|
44
|
+
this.#skipTranslationsFor = ['core', 'data'];
|
|
45
|
+
this.#namePrefix = 'fg_';
|
|
46
|
+
this.gCount = 0;
|
|
47
|
+
this.id = Utils.uuid();
|
|
48
|
+
this.formOptions = input();
|
|
49
|
+
this.#formOptionsEffect = effect(() => {
|
|
50
|
+
this.#init(this.formOptions());
|
|
51
|
+
});
|
|
52
|
+
this.inert = input();
|
|
53
|
+
this.elementExtensions = input([]);
|
|
54
|
+
this.#elementExtensionsEffect = effect(() => {
|
|
55
|
+
this.#formHelperService.setElementExtensions(this.elementExtensions());
|
|
56
|
+
});
|
|
57
|
+
/**
|
|
58
|
+
* There are special scenarios where forms are within a form themselves.
|
|
59
|
+
* Setting this property to true, will handle the current form in a
|
|
60
|
+
* slightly different way when it comes to form scripting.
|
|
61
|
+
*/
|
|
62
|
+
this.isInnerTableForm = input();
|
|
63
|
+
/**
|
|
64
|
+
* triggered when the forms state has been changed
|
|
65
|
+
*/
|
|
66
|
+
this.statusChanged = output();
|
|
67
|
+
/**
|
|
68
|
+
* handler to be executed after the form has been set up
|
|
69
|
+
*/
|
|
70
|
+
this.onFormReady = output();
|
|
71
|
+
this.#scriptModel = {};
|
|
72
|
+
this.#subscriptions = [];
|
|
73
|
+
// local store for all the form control references
|
|
74
|
+
this.#formControls = {};
|
|
75
|
+
this.#initialValidators = {};
|
|
76
|
+
/**
|
|
77
|
+
* This method will be called each time the script changes its internal model.
|
|
78
|
+
* It is used to transfer the script changes to the actual form model.
|
|
79
|
+
*
|
|
80
|
+
* To ensure the right context, we define an instance method as callback for the scripting scope
|
|
81
|
+
* @see: https://blog.johnnyreilly.com/2014/04/typescript-instance-methods.html
|
|
82
|
+
*
|
|
83
|
+
* @param formControlName
|
|
84
|
+
* @param change
|
|
85
|
+
*/
|
|
86
|
+
this.#onScriptingModelChanged = (formControlName, change, formId) => {
|
|
87
|
+
if (!this.form || (formId && this.id !== formId))
|
|
88
|
+
return;
|
|
89
|
+
// find the target control
|
|
90
|
+
const fc = this.#formControls[formControlName];
|
|
91
|
+
if (fc) {
|
|
92
|
+
// change only allowed properties
|
|
93
|
+
switch (change.name) {
|
|
94
|
+
case 'value': {
|
|
95
|
+
if (Array.isArray(change.newValue)) {
|
|
96
|
+
this.#processArrayValueChange(fc, change);
|
|
97
|
+
}
|
|
98
|
+
else {
|
|
99
|
+
fc._eoFormElement.value = change.newValue;
|
|
100
|
+
if (fc.value !== change.newValue) {
|
|
101
|
+
fc.patchValue(change.newValue);
|
|
102
|
+
fc.updateValueAndValidity();
|
|
103
|
+
fc.markAsDirty();
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
this.form.markAsDirty();
|
|
107
|
+
break;
|
|
108
|
+
}
|
|
109
|
+
case 'required': {
|
|
110
|
+
fc._eoFormElement.required = change.newValue;
|
|
111
|
+
// apply new validators
|
|
112
|
+
// @see: https://scotch.io/tutorials/how-to-implement-conditional-validation-in-angular-2-model-driven-forms
|
|
113
|
+
fc.setValidators(Validators.compose(this.#getValidators(fc._eoFormElement).concat([this.#initialValidators[fc._eoFormElement.name]])));
|
|
114
|
+
// need to mark form control as touched because otherwise form validation will not show
|
|
115
|
+
// error messages
|
|
116
|
+
fc.markAsTouched();
|
|
117
|
+
fc.updateValueAndValidity();
|
|
118
|
+
break;
|
|
119
|
+
}
|
|
120
|
+
case 'readonly': {
|
|
121
|
+
fc._eoFormElement.readonly = change.newValue;
|
|
122
|
+
if (change.newValue === true) {
|
|
123
|
+
fc.disable();
|
|
124
|
+
}
|
|
125
|
+
else {
|
|
126
|
+
fc.enable();
|
|
127
|
+
}
|
|
128
|
+
break;
|
|
129
|
+
}
|
|
130
|
+
case 'error': {
|
|
131
|
+
fc._eoFormElement.error = change.newValue;
|
|
132
|
+
fc.markAsTouched();
|
|
133
|
+
fc.updateValueAndValidity();
|
|
134
|
+
break;
|
|
135
|
+
}
|
|
136
|
+
// new onrowedit function was applied by the script
|
|
137
|
+
case 'onrowedit': {
|
|
138
|
+
fc._eoFormElement.onrowedit = change.newValue;
|
|
139
|
+
break;
|
|
140
|
+
}
|
|
141
|
+
// new onchange function was applied by the script
|
|
142
|
+
case 'onchange': {
|
|
143
|
+
fc._eoFormElement.onchange = change.newValue;
|
|
144
|
+
break;
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
this.#cdRef.detectChanges();
|
|
148
|
+
}
|
|
149
|
+
};
|
|
150
|
+
this.#formScriptingApiService.formId = this.id;
|
|
151
|
+
this.#formScriptingApiService.api.events
|
|
152
|
+
.on(FormScriptingApiService.EVENT_MODEL_CHANGED)
|
|
153
|
+
.pipe(takeUntilDestroyed())
|
|
154
|
+
.subscribe((event) => event.data && this.#onScriptingModelChanged(event.data.formControlName, event.data.change, event.data.formId));
|
|
155
|
+
this.#formScriptingApiService.formId = this.id;
|
|
156
|
+
}
|
|
157
|
+
// initialize the form based on the provided form options
|
|
158
|
+
#init(fo) {
|
|
159
|
+
this.form = undefined;
|
|
160
|
+
if (!fo)
|
|
161
|
+
return;
|
|
162
|
+
setTimeout(() => {
|
|
163
|
+
this.#initialValidators = {};
|
|
164
|
+
this.unsubscribeAll();
|
|
165
|
+
this.#buildReactiveForm(fo);
|
|
166
|
+
}, 0);
|
|
167
|
+
}
|
|
168
|
+
#initScriptingScope(formOptions, dataFormModel) {
|
|
169
|
+
const { data, actions, objects, context } = (formOptions || {});
|
|
170
|
+
if (this.#scriptingScope) {
|
|
171
|
+
this.#scriptingScope.setModel(this.#scriptModel);
|
|
172
|
+
/** provide access to actions (used inside of BPM-Forms) */
|
|
173
|
+
this.#scriptingScope.actions = actions;
|
|
174
|
+
/** provide access to additional objects (used for example in BPM-Start-Forms to
|
|
175
|
+
* add data of DMS-Objects to start the process for)
|
|
176
|
+
*/
|
|
177
|
+
this.#scriptingScope.objects = objects || [];
|
|
178
|
+
this.#scriptingScope.context = context || { id: '', title: '', objectTypeId: '' };
|
|
179
|
+
/** provide readonly access to initial form data (which may also contain values that
|
|
180
|
+
* are not rendered as form elements (invisible values))
|
|
181
|
+
*/
|
|
182
|
+
this.#scriptingScope.data = data;
|
|
183
|
+
/** by default, scripting scopes are applied to forms. But table elements create their own scope
|
|
184
|
+
* for editing rows. Being one of those inner forms should not run the form script again, but
|
|
185
|
+
* instead just provide the observing abilities of the scripting scope.
|
|
186
|
+
*/
|
|
187
|
+
if (!this.isInnerTableForm() && dataFormModel) {
|
|
188
|
+
const scriptName = dataFormModel.name + '_' + dataFormModel.situation;
|
|
189
|
+
console.debug('executing form script ' + scriptName);
|
|
190
|
+
this.#formScriptService.runFormScript(this.#scriptingScope, dataFormModel.script, scriptName);
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
focusForm() {
|
|
195
|
+
this.#elementRef.nativeElement.querySelector('input').focus();
|
|
196
|
+
}
|
|
197
|
+
getGroup(id) {
|
|
198
|
+
return this.form?.get(id);
|
|
199
|
+
}
|
|
200
|
+
setFormData(data) {
|
|
201
|
+
const fo = this.formOptions();
|
|
202
|
+
if (!fo)
|
|
203
|
+
return;
|
|
204
|
+
fo.data = data;
|
|
205
|
+
setTimeout(() => this.#init(fo), 0);
|
|
206
|
+
}
|
|
207
|
+
patchValue(data) {
|
|
208
|
+
Object.keys(data).forEach((key) => {
|
|
209
|
+
this.#onScriptingModelChanged(key, {
|
|
210
|
+
name: 'value',
|
|
211
|
+
newValue: data[key]
|
|
212
|
+
});
|
|
213
|
+
});
|
|
214
|
+
}
|
|
215
|
+
/**
|
|
216
|
+
* Extracts the values from the form model. Each form value is represented by one
|
|
217
|
+
* property on the result object holding the fields value. The keys (properties) are the `name`
|
|
218
|
+
* properties of the form element.
|
|
219
|
+
*
|
|
220
|
+
* How values are extracted is influenced by the forms situation.
|
|
221
|
+
*
|
|
222
|
+
* @return object of key value pairs
|
|
223
|
+
*/
|
|
224
|
+
getFormData() {
|
|
225
|
+
return this.#formToData();
|
|
226
|
+
}
|
|
227
|
+
getFormElements() {
|
|
228
|
+
const formElements = {};
|
|
229
|
+
Object.keys(this.#formControls).forEach((k) => {
|
|
230
|
+
formElements[k] = this.#formControls[k].__eoFormElement;
|
|
231
|
+
});
|
|
232
|
+
return formElements;
|
|
233
|
+
}
|
|
234
|
+
setFormPristine() {
|
|
235
|
+
if (!this.form)
|
|
236
|
+
return;
|
|
237
|
+
this.form.markAsPristine();
|
|
238
|
+
}
|
|
239
|
+
// reset the form to its initial state
|
|
240
|
+
resetForm() {
|
|
241
|
+
const fo = this.formOptions();
|
|
242
|
+
if (!fo)
|
|
243
|
+
return;
|
|
244
|
+
this.setFormData(fo.data);
|
|
245
|
+
this.#emitFormChangedEvent();
|
|
246
|
+
}
|
|
247
|
+
/**
|
|
248
|
+
* Returns the observed model that was passed to the current form script running. If there is
|
|
249
|
+
* no form script, this method will return NULL.
|
|
250
|
+
* @returns
|
|
251
|
+
*/
|
|
252
|
+
getObservedScriptModel() {
|
|
253
|
+
return this.#scriptingScope ? this.#scriptingScope.getModel() : null;
|
|
254
|
+
}
|
|
255
|
+
// Create a reactive form from the enaio form model
|
|
256
|
+
#buildReactiveForm(formOptions) {
|
|
257
|
+
this.#scriptingScope = undefined;
|
|
258
|
+
this.#scriptModel = {};
|
|
259
|
+
this.#formControls = {};
|
|
260
|
+
const formModel = this.#dataToForm(formOptions.formModel, formOptions.data);
|
|
261
|
+
if (!formModel) {
|
|
262
|
+
return;
|
|
263
|
+
}
|
|
264
|
+
// if a script is available, we'll init the form scripting for the
|
|
265
|
+
// current form
|
|
266
|
+
if (this.isInnerTableForm() || (formModel.script && formModel.script.length > 0)) {
|
|
267
|
+
console.debug('adding form scripting scope');
|
|
268
|
+
this.#scriptingScope = new ObjectFormScriptingScope(formModel.situation, this.#onScriptingModelChanged, this.#formScriptingApiService.getApi(), this.isInnerTableForm());
|
|
269
|
+
this.#scriptingScope.objectId = this.formOptions().objectId;
|
|
270
|
+
}
|
|
271
|
+
const form = new ObjectFormGroup({});
|
|
272
|
+
if (formModel?.elements[0]?.elements) {
|
|
273
|
+
this.#addFormControl(form, formModel.elements[0], 'core');
|
|
274
|
+
}
|
|
275
|
+
if (formModel?.elements[1]?.elements) {
|
|
276
|
+
this.#addFormControl(form, formModel.elements[1], 'data');
|
|
277
|
+
}
|
|
278
|
+
this.form = form;
|
|
279
|
+
this.#formScriptingApiService.registerActiveForm(this);
|
|
280
|
+
setTimeout(() => {
|
|
281
|
+
if (!this.form)
|
|
282
|
+
return;
|
|
283
|
+
this.initValidators(this.form);
|
|
284
|
+
this.#subscriptions.push(this.form.valueChanges.pipe(debounceTime(500)).subscribe(() => this.#emitFormChangedEvent()));
|
|
285
|
+
this.#initScriptingScope(formOptions, formModel);
|
|
286
|
+
this.onFormReady.emit();
|
|
287
|
+
this.#emitFormChangedEvent(false);
|
|
288
|
+
}, 300);
|
|
289
|
+
}
|
|
290
|
+
initValidators(form) {
|
|
291
|
+
if (this.form) {
|
|
292
|
+
for (const key of Object.keys(form.controls)) {
|
|
293
|
+
const control = form.controls[key];
|
|
294
|
+
if (control.controls) {
|
|
295
|
+
this.initValidators(control);
|
|
296
|
+
}
|
|
297
|
+
else {
|
|
298
|
+
this.#initialValidators[control._eoFormElement.name] = control.validator;
|
|
299
|
+
control.setValidators(Validators.compose(this.#getValidators(control._eoFormElement).concat([this.#initialValidators[control._eoFormElement.name]])));
|
|
300
|
+
control.updateValueAndValidity();
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
#emitFormChangedEvent(compare = true) {
|
|
306
|
+
// check if indexdata has been changed
|
|
307
|
+
const currentFormData = this.getFormData();
|
|
308
|
+
const idxChange = compare ? JSON.stringify(this.formData) !== JSON.stringify(currentFormData) : false;
|
|
309
|
+
this.formData = currentFormData;
|
|
310
|
+
if (this.form) {
|
|
311
|
+
this.statusChanged.emit({
|
|
312
|
+
invalid: this.form.invalid,
|
|
313
|
+
dirty: this.form.dirty,
|
|
314
|
+
data: this.formData,
|
|
315
|
+
indexdataChanged: idxChange
|
|
316
|
+
});
|
|
317
|
+
}
|
|
318
|
+
}
|
|
319
|
+
/**
|
|
320
|
+
* This method will be called each time the script changes its internal model.
|
|
321
|
+
* It is used to transfer the script changes to the actual form model.
|
|
322
|
+
*
|
|
323
|
+
* To ensure the right context, we define an instance method as callback for the scripting scope
|
|
324
|
+
* @see: https://blog.johnnyreilly.com/2014/04/typescript-instance-methods.html
|
|
325
|
+
*
|
|
326
|
+
* @param formControlName
|
|
327
|
+
* @param change
|
|
328
|
+
*/
|
|
329
|
+
#onScriptingModelChanged;
|
|
330
|
+
#processArrayValueChange(fc, change) {
|
|
331
|
+
const newVal = change.newValue;
|
|
332
|
+
const targetType = fc._eoFormElement.type;
|
|
333
|
+
// for some types we have to ensure that meta data are provided as well
|
|
334
|
+
switch (targetType) {
|
|
335
|
+
case 'ORGANIZATION': {
|
|
336
|
+
this.#getDataMeta(fc._eoFormElement, newVal).subscribe((m) => {
|
|
337
|
+
fc._eoFormElement.dataMeta = m;
|
|
338
|
+
});
|
|
339
|
+
break;
|
|
340
|
+
}
|
|
341
|
+
case 'TABLE': {
|
|
342
|
+
const dataToBeProcessed = {};
|
|
343
|
+
fc._eoFormElement.elements.forEach((e) => {
|
|
344
|
+
if (e.type === 'ORGANIZATION' || e.type === 'CODESYSTEM') {
|
|
345
|
+
dataToBeProcessed[e.name] = e;
|
|
346
|
+
}
|
|
347
|
+
});
|
|
348
|
+
if (Object.keys(dataToBeProcessed).length) {
|
|
349
|
+
newVal.forEach((rowData) => {
|
|
350
|
+
Object.keys(rowData).forEach((key) => {
|
|
351
|
+
if (dataToBeProcessed[key]) {
|
|
352
|
+
this.#getDataMeta(dataToBeProcessed[key], rowData[key]).subscribe((m) => {
|
|
353
|
+
if (m) {
|
|
354
|
+
rowData[key + '_meta'] = m;
|
|
355
|
+
}
|
|
356
|
+
else {
|
|
357
|
+
delete rowData[key + '_meta'];
|
|
358
|
+
}
|
|
359
|
+
this.#updateArrayValue(fc, newVal);
|
|
360
|
+
});
|
|
361
|
+
}
|
|
362
|
+
});
|
|
363
|
+
});
|
|
364
|
+
}
|
|
365
|
+
break;
|
|
366
|
+
}
|
|
367
|
+
}
|
|
368
|
+
this.#updateArrayValue(fc, newVal);
|
|
369
|
+
}
|
|
370
|
+
#updateArrayValue(fc, newValue) {
|
|
371
|
+
fc._eoFormElement.value = [].concat(newValue);
|
|
372
|
+
fc.patchValue([].concat(newValue));
|
|
373
|
+
fc.updateValueAndValidity();
|
|
374
|
+
fc.markAsDirty();
|
|
375
|
+
}
|
|
376
|
+
#getDataMeta(formElement, newValue) {
|
|
377
|
+
if (newValue) {
|
|
378
|
+
switch (formElement.type) {
|
|
379
|
+
case 'ORGANIZATION': {
|
|
380
|
+
return this.#userService.getUserById(newValue);
|
|
381
|
+
}
|
|
382
|
+
}
|
|
383
|
+
}
|
|
384
|
+
return of(null);
|
|
385
|
+
}
|
|
386
|
+
#patchFormValue(formValue) {
|
|
387
|
+
let value;
|
|
388
|
+
if (Array.isArray(formValue)) {
|
|
389
|
+
value = [];
|
|
390
|
+
// copy by value for arrays of objects (e.g. table data)
|
|
391
|
+
formValue.forEach((o) => value.push(JSON.parse(JSON.stringify(o))));
|
|
392
|
+
}
|
|
393
|
+
else {
|
|
394
|
+
value = formValue;
|
|
395
|
+
}
|
|
396
|
+
return value;
|
|
397
|
+
}
|
|
398
|
+
/**
|
|
399
|
+
* Recursive method adding a new FormControl (group or control) to a parent form group
|
|
400
|
+
*
|
|
401
|
+
* @param parentGroup - the parent group to add the control to
|
|
402
|
+
* @param formElement - the enaio form model element to create the child control from
|
|
403
|
+
* @param [useName] - use this name instead of the one from the model
|
|
404
|
+
*/
|
|
405
|
+
#addFormControl(parentGroup, formElement, useName) {
|
|
406
|
+
const fo = this.formOptions();
|
|
407
|
+
if (!fo)
|
|
408
|
+
return;
|
|
409
|
+
let ctrl;
|
|
410
|
+
let name;
|
|
411
|
+
// add a form group
|
|
412
|
+
if (formElement.type === 'o2mGroup' || formElement.type === 'o2mGroupStack') {
|
|
413
|
+
// do not add groups that are empty
|
|
414
|
+
const isRootGroup = formElement.name && (formElement.name !== 'data' || formElement.name !== 'core');
|
|
415
|
+
if (!isRootGroup && (!formElement.elements || formElement.elements.length === 0)) {
|
|
416
|
+
console.error('Found empty form group', formElement);
|
|
417
|
+
return;
|
|
418
|
+
}
|
|
419
|
+
ctrl = new ObjectFormGroup({});
|
|
420
|
+
ctrl._eoFormGroup = {
|
|
421
|
+
layout: formElement.layout,
|
|
422
|
+
type: formElement.type
|
|
423
|
+
};
|
|
424
|
+
if (formElement.name) {
|
|
425
|
+
if (this.#skipTranslationsFor.includes(formElement.name)) {
|
|
426
|
+
ctrl._eoFormGroup.label = formElement.name;
|
|
427
|
+
}
|
|
428
|
+
else if (!formElement.label) {
|
|
429
|
+
// no label, so try to translate the technical name
|
|
430
|
+
ctrl._eoFormGroup.label = this.#systemService.getLocalizedResource(`${formElement.name}_label`) || formElement.name;
|
|
431
|
+
}
|
|
432
|
+
else {
|
|
433
|
+
// there is a label so we take this one
|
|
434
|
+
ctrl._eoFormGroup.label = formElement.label;
|
|
435
|
+
}
|
|
436
|
+
}
|
|
437
|
+
if (useName === 'core' || useName === 'data') {
|
|
438
|
+
ctrl._eoFormGroup.label = useName;
|
|
439
|
+
}
|
|
440
|
+
for (const e of formElement.elements) {
|
|
441
|
+
this.#addFormControl(ctrl, e);
|
|
442
|
+
}
|
|
443
|
+
name = useName || 'fg' + this.gCount++;
|
|
444
|
+
}
|
|
445
|
+
else {
|
|
446
|
+
// add form control
|
|
447
|
+
// To be able to integrate recursive form controls into the main form,
|
|
448
|
+
// we have to wrap them in a form group
|
|
449
|
+
ctrl = new ObjectFormControlWrapper({});
|
|
450
|
+
ctrl._eoFormControlWrapper = {
|
|
451
|
+
// the name of the wrapped FormControl
|
|
452
|
+
controlName: formElement.name,
|
|
453
|
+
situation: fo.formModel.situation
|
|
454
|
+
};
|
|
455
|
+
// do not set a reference as the form controls value
|
|
456
|
+
// otherwise we could not reset the form
|
|
457
|
+
const value = formElement?.value
|
|
458
|
+
? this.#patchFormValue(formElement?.value)
|
|
459
|
+
: !Utils.isEmpty(formElement?.defaultvalue) && fo.formModel.situation === Situation.CREATE
|
|
460
|
+
? this.#patchFormValue(formElement?.defaultvalue)
|
|
461
|
+
: formElement?.value;
|
|
462
|
+
// create the actual form control
|
|
463
|
+
const controlDisabled = fo.disabled || !!formElement.readonly;
|
|
464
|
+
const formControl = new ObjectFormControl({
|
|
465
|
+
value,
|
|
466
|
+
disabled: controlDisabled
|
|
467
|
+
});
|
|
468
|
+
// if form element has a 'label' property it will be shown as is ...
|
|
469
|
+
if (!formElement.label) {
|
|
470
|
+
// ... if there is no label but a 'labelKey' property we are going to localize that key ...
|
|
471
|
+
if (formElement.labelkey) {
|
|
472
|
+
formElement.label = this.#systemService.getLocalizedResource(`${formElement.labelkey}_label`) || formElement.labelkey;
|
|
473
|
+
formElement.description = this.#systemService.getLocalizedResource(`${formElement.labelkey}_description`);
|
|
474
|
+
}
|
|
475
|
+
else {
|
|
476
|
+
// ... no 'label' and also no 'labelKey' means that we are showing technical name instead
|
|
477
|
+
formElement.label = formElement.name;
|
|
478
|
+
}
|
|
479
|
+
}
|
|
480
|
+
formElement.readonly = controlDisabled;
|
|
481
|
+
// we are using an internal type to distinguish between the components
|
|
482
|
+
// to be used to render certain form elements
|
|
483
|
+
formElement._internalType = this.#systemService.getInternalFormElementType(formElement.type, formElement.classifications);
|
|
484
|
+
formControl._eoFormElement = formElement;
|
|
485
|
+
this.#formControls[formElement.name] = formControl;
|
|
486
|
+
if (formElement.type === 'TABLE') {
|
|
487
|
+
// Add 'onrowedit' property even if it's not set.
|
|
488
|
+
// This is required because otherwise mobX-Observer would not recognize
|
|
489
|
+
// changes to this property applied by the form script
|
|
490
|
+
if (!Object.keys(formControl._eoFormElement).includes('onrowedit')) {
|
|
491
|
+
formControl._eoFormElement.onrowedit = null;
|
|
492
|
+
}
|
|
493
|
+
if (this.#scriptingScope && formControl._eoFormElement.value) {
|
|
494
|
+
// having a scripting scope and table rows means that we need to set empty
|
|
495
|
+
// columns to NULL, because otherwise mobX won't be able to track those values
|
|
496
|
+
const valueFields = formControl._eoFormElement.elements.map((e) => e.name);
|
|
497
|
+
formControl._eoFormElement.value.forEach((rowValue) => {
|
|
498
|
+
valueFields.forEach((valueField) => {
|
|
499
|
+
if (!Object.keys(rowValue).includes(valueField)) {
|
|
500
|
+
rowValue[valueField] = null;
|
|
501
|
+
}
|
|
502
|
+
});
|
|
503
|
+
});
|
|
504
|
+
// update form controls value as well to reflect the observed value
|
|
505
|
+
formControl.patchValue(formControl._eoFormElement.value, {
|
|
506
|
+
onlySelf: true,
|
|
507
|
+
emitEvent: false,
|
|
508
|
+
emitModelToViewChange: false,
|
|
509
|
+
emitViewToModelChange: false
|
|
510
|
+
});
|
|
511
|
+
}
|
|
512
|
+
}
|
|
513
|
+
if (formElement._internalType === 'string:organization') {
|
|
514
|
+
formControl._eoFormElement.setFilter = (filterObject) => {
|
|
515
|
+
formControl._eoFormElement.filter = filterObject;
|
|
516
|
+
};
|
|
517
|
+
}
|
|
518
|
+
ObjectFormUtils.updateFormElement(formElement);
|
|
519
|
+
// remove empty descriptions
|
|
520
|
+
const desc = formControl._eoFormElement.description;
|
|
521
|
+
if (desc && desc.trim().length === 0) {
|
|
522
|
+
formControl._eoFormElement.description = null;
|
|
523
|
+
}
|
|
524
|
+
// add the form element to the script model that will be injected into
|
|
525
|
+
// the forms scripting scope later on
|
|
526
|
+
this.#scriptModel[formElement.name] = formControl._eoFormElement;
|
|
527
|
+
// apply change listener to the form control, that will trigger
|
|
528
|
+
// the form elements onChange listener
|
|
529
|
+
const controlWatch = ctrl.valueChanges.pipe(debounceTime(500));
|
|
530
|
+
this.#subscriptions.push(controlWatch.subscribe((v) => {
|
|
531
|
+
if (this.#scriptingScope) {
|
|
532
|
+
this.#scriptingScope.modelChanged(v);
|
|
533
|
+
}
|
|
534
|
+
}));
|
|
535
|
+
ctrl.addControl(formElement.name, formControl);
|
|
536
|
+
name = this.#namePrefix + formElement.name;
|
|
537
|
+
}
|
|
538
|
+
parentGroup.addControl(name, ctrl);
|
|
539
|
+
}
|
|
540
|
+
/**
|
|
541
|
+
* Build validators for the given form element to be attached to
|
|
542
|
+
* a reactive formControl.
|
|
543
|
+
*
|
|
544
|
+
* @param formElement - form element object
|
|
545
|
+
*/
|
|
546
|
+
#getValidators(formElement) {
|
|
547
|
+
const fo = this.formOptions();
|
|
548
|
+
const elmValidators = FormValidation.getValidators(formElement, fo?.formModel.situation);
|
|
549
|
+
// add custom validator for script enabled forms
|
|
550
|
+
if (this.#scriptingScope) {
|
|
551
|
+
elmValidators.push(FormValidation.customScriptingValidation);
|
|
552
|
+
}
|
|
553
|
+
return elmValidators;
|
|
554
|
+
}
|
|
555
|
+
/**
|
|
556
|
+
* Extract the values from the form
|
|
557
|
+
*/
|
|
558
|
+
#formToData() {
|
|
559
|
+
const fo = this.formOptions();
|
|
560
|
+
return fo && this.form ? this.#formHelperService.extractFormData(this.form, fo.formModel.situation, fo.data, this.isInnerTableForm()) : {};
|
|
561
|
+
}
|
|
562
|
+
/**
|
|
563
|
+
* Merge data into a form model.
|
|
564
|
+
*
|
|
565
|
+
* @param model - form model
|
|
566
|
+
* @param data - data object or array of SearchFilter objects in case of a search form
|
|
567
|
+
*/
|
|
568
|
+
#dataToForm(model, data) {
|
|
569
|
+
if (model && data) {
|
|
570
|
+
this.#setElementValues(model.elements, data);
|
|
571
|
+
}
|
|
572
|
+
return model;
|
|
573
|
+
}
|
|
574
|
+
// recursive method for adding values to model elements
|
|
575
|
+
#setElementValues(elements, data) {
|
|
576
|
+
elements?.forEach((element) => {
|
|
577
|
+
if (this.#hasValue(data, element)) {
|
|
578
|
+
element.value = this.#getValue(data, element);
|
|
579
|
+
}
|
|
580
|
+
else {
|
|
581
|
+
delete element.value;
|
|
582
|
+
}
|
|
583
|
+
if (element.type !== 'TABLE' && element.elements && element.elements.length > 0) {
|
|
584
|
+
this.#setElementValues(element.elements, data);
|
|
585
|
+
}
|
|
586
|
+
});
|
|
587
|
+
}
|
|
588
|
+
#hasValue(data, element) {
|
|
589
|
+
// differ between array of SearchFilters and a form data object
|
|
590
|
+
if (Array.isArray(data)) {
|
|
591
|
+
return !!data.find((filter) => filter.property === element.id);
|
|
592
|
+
}
|
|
593
|
+
else {
|
|
594
|
+
return Object.keys(data).includes(element.name);
|
|
595
|
+
}
|
|
596
|
+
}
|
|
597
|
+
#getValue(data, element) {
|
|
598
|
+
return data[element.name];
|
|
599
|
+
}
|
|
600
|
+
ngOnDestroy() {
|
|
601
|
+
this.unsubscribeAll();
|
|
602
|
+
this.#formScriptingApiService.unregisterActiveForm(this);
|
|
603
|
+
}
|
|
604
|
+
ngAfterViewInit() {
|
|
605
|
+
this.#cdRef.detectChanges();
|
|
606
|
+
}
|
|
607
|
+
// unsubscribe from all value change listeners for the current form
|
|
608
|
+
// to avoid memory leaks. This method will also be called every time
|
|
609
|
+
// a new form is rendered to get rid of the old form element subscriptions
|
|
610
|
+
unsubscribeAll() {
|
|
611
|
+
if (this.#subscriptions.length) {
|
|
612
|
+
console.debug('unsubscribed from ' + this.#subscriptions.length + ' value change listeners.');
|
|
613
|
+
this.#subscriptions.forEach((s) => s.unsubscribe());
|
|
614
|
+
this.#subscriptions = [];
|
|
615
|
+
}
|
|
616
|
+
}
|
|
617
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ObjectFormComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
618
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: ObjectFormComponent, isStandalone: true, selector: "yuv-object-form", inputs: { formOptions: { classPropertyName: "formOptions", publicName: "formOptions", isSignal: true, isRequired: false, transformFunction: null }, inert: { classPropertyName: "inert", publicName: "inert", isSignal: true, isRequired: false, transformFunction: null }, elementExtensions: { classPropertyName: "elementExtensions", publicName: "elementExtensions", isSignal: true, isRequired: false, transformFunction: null }, isInnerTableForm: { classPropertyName: "isInnerTableForm", publicName: "isInnerTableForm", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { statusChanged: "statusChanged", onFormReady: "onFormReady" }, host: { properties: { "attr.inert": "inert() ? true : null" }, classAttribute: "yuv-object-form" }, providers: [ObjectFormService, ObjectFormScriptService, FormScriptingApiService], ngImport: i0, template: "@let fo = formOptions();\n@if (fo && form) {\n <div class=\"yuv-object-form\" [formGroup]=\"form\">\n <div class=\"form-element-wrap core\">\n <yuv-object-form-group [group]=\"getGroup('core')\" [situation]=\"fo.formModel.situation\"></yuv-object-form-group>\n </div>\n <div class=\"form-element-wrap data\">\n <yuv-object-form-group [group]=\"getGroup('data')\" [situation]=\"fo.formModel.situation\"></yuv-object-form-group>\n </div>\n </div>\n}\n", styles: [".yuv-object-form>.core{padding:calc(var(--app-pane-padding) / 2)}.yuv-object-form>.data>yuv-object-form-group>.form-element.o2mGroupStack{background:transparent;border:0}.yuv-object-form>.data>yuv-object-form-group>.form-element.o2mGroupStack>yuv-object-form-group>.o2mGroup{background-color:var(--panel-background-lightgrey);border-top:1px solid rgba(0,0,0,.1);padding:var(--app-pane-padding)}.yuv-object-form>.data>yuv-object-form-group>.form-element.o2mGroupStack>p-tabview>.p-tabview>.p-tabview-panels{background-color:var(--panel-background-lightgrey)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: ObjectFormGroupComponent, selector: "yuv-object-form-group", inputs: ["situation", "group", "noGroupLabels"] }], encapsulation: i0.ViewEncapsulation.None }); }
|
|
619
|
+
}
|
|
620
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ObjectFormComponent, decorators: [{
|
|
621
|
+
type: Component,
|
|
622
|
+
args: [{ selector: 'yuv-object-form', standalone: true, imports: [CommonModule, ReactiveFormsModule, ObjectFormGroupComponent], providers: [ObjectFormService, ObjectFormScriptService, FormScriptingApiService], encapsulation: ViewEncapsulation.None, host: {
|
|
623
|
+
class: 'yuv-object-form',
|
|
624
|
+
'[attr.inert]': 'inert() ? true : null'
|
|
625
|
+
}, template: "@let fo = formOptions();\n@if (fo && form) {\n <div class=\"yuv-object-form\" [formGroup]=\"form\">\n <div class=\"form-element-wrap core\">\n <yuv-object-form-group [group]=\"getGroup('core')\" [situation]=\"fo.formModel.situation\"></yuv-object-form-group>\n </div>\n <div class=\"form-element-wrap data\">\n <yuv-object-form-group [group]=\"getGroup('data')\" [situation]=\"fo.formModel.situation\"></yuv-object-form-group>\n </div>\n </div>\n}\n", styles: [".yuv-object-form>.core{padding:calc(var(--app-pane-padding) / 2)}.yuv-object-form>.data>yuv-object-form-group>.form-element.o2mGroupStack{background:transparent;border:0}.yuv-object-form>.data>yuv-object-form-group>.form-element.o2mGroupStack>yuv-object-form-group>.o2mGroup{background-color:var(--panel-background-lightgrey);border-top:1px solid rgba(0,0,0,.1);padding:var(--app-pane-padding)}.yuv-object-form>.data>yuv-object-form-group>.form-element.o2mGroupStack>p-tabview>.p-tabview>.p-tabview-panels{background-color:var(--panel-background-lightgrey)}\n"] }]
|
|
626
|
+
}], ctorParameters: () => [] });
|
|
627
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"object-form.component.js","sourceRoot":"","sources":["../../../../../../../libs/yuuvis/client-framework/object-form/src/lib/object-form.component.ts","../../../../../../../libs/yuuvis/client-framework/object-form/src/lib/object-form.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAiB,iBAAiB,EAAE,SAAS,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,EAAE,KAAK,EAAa,MAAM,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AACrJ,OAAO,EAAE,kBAAkB,EAAE,MAAM,4BAA4B,CAAC;AAChE,OAAO,EAAE,mBAAmB,EAAe,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAC9E,OAAO,EAAE,SAAS,EAAE,aAAa,EAAE,WAAW,EAAE,KAAK,EAAE,MAAM,qBAAqB,CAAC;AACnF,OAAO,EAAE,YAAY,EAAc,EAAE,EAAgB,MAAM,MAAM,CAAC;AAClE,OAAO,EAAE,uBAAuB,EAAE,MAAM,0BAA0B,CAAC;AAEnE,OAAO,EAAE,wBAAwB,EAAE,MAAM,iDAAiD,CAAC;AAC3F,OAAO,EAAE,uBAAuB,EAAE,MAAM,iDAAiD,CAAC;AAC1F,OAAO,EAAE,wBAAwB,EAAE,MAAM,kDAAkD,CAAC;AAC5F,OAAO,EAA0B,wBAAwB,EAA4C,MAAM,yBAAyB,CAAC;AACrI,OAAO,EAAE,iBAAiB,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACzE,OAAO,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAC;AAC1D,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,cAAc,EAAE,MAAM,0BAA0B,CAAC;;;AAe1D,MAAM,OAAO,mBAAmB;IAC9B,WAAW,CAAsB;IACjC,cAAc,CAAyB;IACvC,kBAAkB,CAAmC;IACrD,kBAAkB,CAA6B;IAC/C,YAAY,CAAuB;IACnC,MAAM,CAA6B;IACnC,wBAAwB,CAAmC;IAE3D,oBAAoB,CAAoB;IACxC,WAAW,CAAS;IAOpB,kBAAkB,CAEf;IAIH,wBAAwB,CAErB;IAyBH,YAAY,CAAW;IACvB,eAAe,CAA4B;IAC3C,cAAc,CAAsB;IAEpC,kDAAkD;IAClD,aAAa,CAAW;IACxB,kBAAkB,CAAW;IAE7B;QAzDA,gBAAW,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;QACjC,mBAAc,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;QACvC,uBAAkB,GAAG,MAAM,CAAC,uBAAuB,CAAC,CAAC;QACrD,uBAAkB,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;QAC/C,iBAAY,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QACnC,WAAM,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;QACnC,6BAAwB,GAAG,MAAM,CAAC,uBAAuB,CAAC,CAAC;QAE3D,yBAAoB,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;QACxC,gBAAW,GAAG,KAAK,CAAC;QACpB,WAAM,GAAG,CAAC,CAAC;QAEX,OAAE,GAAG,KAAK,CAAC,IAAI,EAAE,CAAC;QAGlB,gBAAW,GAAG,KAAK,EAAqB,CAAC;QACzC,uBAAkB,GAAG,MAAM,CAAC,GAAG,EAAE;YAC/B,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC;QACjC,CAAC,CAAC,CAAC;QAEH,UAAK,GAAG,KAAK,EAAW,CAAC;QACzB,sBAAiB,GAAG,KAAK,CAAgC,EAAE,CAAC,CAAC;QAC7D,6BAAwB,GAAG,MAAM,CAAC,GAAG,EAAE;YACrC,IAAI,CAAC,kBAAkB,CAAC,oBAAoB,CAAC,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAAC;QACzE,CAAC,CAAC,CAAC;QAEH;;;;WAIG;QACH,qBAAgB,GAAG,KAAK,EAAW,CAAC;QAEpC;;WAEG;QACH,kBAAa,GAAG,MAAM,EAA0B,CAAC;QAEjD;;WAEG;QACH,gBAAW,GAAG,MAAM,EAAE,CAAC;QAQvB,iBAAY,GAAQ,EAAE,CAAC;QAEvB,mBAAc,GAAmB,EAAE,CAAC;QAEpC,kDAAkD;QAClD,kBAAa,GAAQ,EAAE,CAAC;QACxB,uBAAkB,GAAQ,EAAE,CAAC;QAkM7B;;;;;;;;;WASG;QACH,6BAAwB,GAAG,CAAC,eAAuB,EAAE,MAA6B,EAAE,MAAe,EAAE,EAAE;YACrG,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,EAAE,KAAK,MAAM,CAAC;gBAAE,OAAO;YACzD,0BAA0B;YAC1B,MAAM,EAAE,GAAsB,IAAI,CAAC,aAAa,CAAC,eAAe,CAAsB,CAAC;YACvF,IAAI,EAAE,EAAE,CAAC;gBACP,iCAAiC;gBACjC,QAAQ,MAAM,CAAC,IAAI,EAAE,CAAC;oBACpB,KAAK,OAAO,CAAC,CAAC,CAAC;wBACb,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,EAAE,CAAC;4BACnC,IAAI,CAAC,wBAAwB,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;wBAC5C,CAAC;6BAAM,CAAC;4BACN,EAAE,CAAC,cAAc,CAAC,KAAK,GAAG,MAAM,CAAC,QAAQ,CAAC;4BAC1C,IAAI,EAAE,CAAC,KAAK,KAAK,MAAM,CAAC,QAAQ,EAAE,CAAC;gCACjC,EAAE,CAAC,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;gCAC/B,EAAE,CAAC,sBAAsB,EAAE,CAAC;gCAC5B,EAAE,CAAC,WAAW,EAAE,CAAC;4BACnB,CAAC;wBACH,CAAC;wBACD,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;wBACxB,MAAM;oBACR,CAAC;oBACD,KAAK,UAAU,CAAC,CAAC,CAAC;wBAChB,EAAE,CAAC,cAAc,CAAC,QAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC;wBAC7C,uBAAuB;wBACvB,4GAA4G;wBAC5G,EAAE,CAAC,aAAa,CAAC,UAAU,CAAC,OAAO,CAAC,IAAI,CAAC,cAAc,CAAC,EAAE,CAAC,cAAc,CAAC,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,kBAAkB,CAAC,EAAE,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;wBACvI,uFAAuF;wBACvF,iBAAiB;wBACjB,EAAE,CAAC,aAAa,EAAE,CAAC;wBACnB,EAAE,CAAC,sBAAsB,EAAE,CAAC;wBAC5B,MAAM;oBACR,CAAC;oBACD,KAAK,UAAU,CAAC,CAAC,CAAC;wBAChB,EAAE,CAAC,cAAc,CAAC,QAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC;wBAC7C,IAAI,MAAM,CAAC,QAAQ,KAAK,IAAI,EAAE,CAAC;4BAC7B,EAAE,CAAC,OAAO,EAAE,CAAC;wBACf,CAAC;6BAAM,CAAC;4BACN,EAAE,CAAC,MAAM,EAAE,CAAC;wBACd,CAAC;wBACD,MAAM;oBACR,CAAC;oBACD,KAAK,OAAO,CAAC,CAAC,CAAC;wBACb,EAAE,CAAC,cAAc,CAAC,KAAK,GAAG,MAAM,CAAC,QAAQ,CAAC;wBAC1C,EAAE,CAAC,aAAa,EAAE,CAAC;wBACnB,EAAE,CAAC,sBAAsB,EAAE,CAAC;wBAC5B,MAAM;oBACR,CAAC;oBACD,mDAAmD;oBACnD,KAAK,WAAW,CAAC,CAAC,CAAC;wBACjB,EAAE,CAAC,cAAc,CAAC,SAAS,GAAG,MAAM,CAAC,QAAQ,CAAC;wBAC9C,MAAM;oBACR,CAAC;oBACD,kDAAkD;oBAClD,KAAK,UAAU,CAAC,CAAC,CAAC;wBAChB,EAAE,CAAC,cAAc,CAAC,QAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC;wBAC7C,MAAM;oBACR,CAAC;gBACH,CAAC;gBACD,IAAI,CAAC,MAAM,CAAC,aAAa,EAAE,CAAC;YAC9B,CAAC;QACH,CAAC,CAAC;QArQA,IAAI,CAAC,wBAAwB,CAAC,MAAM,GAAG,IAAI,CAAC,EAAE,CAAC;QAC/C,IAAI,CAAC,wBAAwB,CAAC,GAAG,CAAC,MAAM;aACrC,EAAE,CAAC,uBAAuB,CAAC,mBAAmB,CAAC;aAC/C,IAAI,CAAC,kBAAkB,EAAE,CAAC;aAC1B,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,IAAI,IAAI,IAAI,CAAC,wBAAwB,CAAC,KAAK,CAAC,IAAI,CAAC,eAAe,EAAE,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC;QAEvI,IAAI,CAAC,wBAAwB,CAAC,MAAM,GAAG,IAAI,CAAC,EAAE,CAAC;IACjD,CAAC;IAED,yDAAyD;IACzD,KAAK,CAAC,EAAiC;QACrC,IAAI,CAAC,IAAI,GAAG,SAAS,CAAC;QACtB,IAAI,CAAC,EAAE;YAAE,OAAO;QAChB,UAAU,CAAC,GAAG,EAAE;YACd,IAAI,CAAC,kBAAkB,GAAG,EAAE,CAAC;YAC7B,IAAI,CAAC,cAAc,EAAE,CAAC;YACtB,IAAI,CAAC,kBAAkB,CAAC,EAAE,CAAC,CAAC;QAC9B,CAAC,EAAE,CAAC,CAAC,CAAC;IACR,CAAC;IAED,mBAAmB,CAAC,WAA8B,EAAE,aAAkB;QACpE,MAAM,EAAE,IAAI,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,CAAC,WAAW,IAAI,EAAE,CAAsB,CAAC;QAErF,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;YACzB,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YACjD,2DAA2D;YAC3D,IAAI,CAAC,eAAe,CAAC,OAAO,GAAG,OAAO,CAAC;YACvC;;eAEG;YACH,IAAI,CAAC,eAAe,CAAC,OAAO,GAAG,OAAO,IAAI,EAAE,CAAC;YAC7C,IAAI,CAAC,eAAe,CAAC,OAAO,GAAG,OAAO,IAAI,EAAE,EAAE,EAAE,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,YAAY,EAAE,EAAE,EAAE,CAAC;YAClF;;eAEG;YACH,IAAI,CAAC,eAAe,CAAC,IAAI,GAAG,IAAI,CAAC;YAEjC;;;eAGG;YACH,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,IAAI,aAAa,EAAE,CAAC;gBAC9C,MAAM,UAAU,GAAG,aAAa,CAAC,IAAI,GAAG,GAAG,GAAG,aAAa,CAAC,SAAS,CAAC;gBACtE,OAAO,CAAC,KAAK,CAAC,wBAAwB,GAAG,UAAU,CAAC,CAAC;gBACrD,IAAI,CAAC,kBAAkB,CAAC,aAAa,CAAC,IAAI,CAAC,eAAe,EAAE,aAAa,CAAC,MAAM,EAAE,UAAU,CAAC,CAAC;YAChG,CAAC;QACH,CAAC;IACH,CAAC;IAED,SAAS;QACP,IAAI,CAAC,WAAW,CAAC,aAAa,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,CAAC;IAChE,CAAC;IAED,QAAQ,CAAC,EAAU;QACjB,OAAO,IAAI,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAoB,CAAC;IAC/C,CAAC;IAED,WAAW,CAAC,IAA6B;QACvC,MAAM,EAAE,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAC9B,IAAI,CAAC,EAAE;YAAE,OAAO;QAChB,EAAE,CAAC,IAAI,GAAG,IAAI,CAAC;QACf,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;IACtC,CAAC;IAED,UAAU,CAAC,IAA6B;QACtC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE;YAChC,IAAI,CAAC,wBAAwB,CAAC,GAAG,EAAE;gBACjC,IAAI,EAAE,OAAO;gBACb,QAAQ,EAAE,IAAI,CAAC,GAAG,CAAC;aACpB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;;;;OAQG;IACH,WAAW;QACT,OAAO,IAAI,CAAC,WAAW,EAAE,CAAC;IAC5B,CAAC;IAED,eAAe;QACb,MAAM,YAAY,GAAQ,EAAE,CAAC;QAC7B,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;YAC5C,YAAY,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,eAAe,CAAC;QAC1D,CAAC,CAAC,CAAC;QACH,OAAO,YAAY,CAAC;IACtB,CAAC;IAED,eAAe;QACb,IAAI,CAAC,IAAI,CAAC,IAAI;YAAE,OAAO;QACvB,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC;IAC7B,CAAC;IAED,sCAAsC;IACtC,SAAS;QACP,MAAM,EAAE,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAC9B,IAAI,CAAC,EAAE;YAAE,OAAO;QAChB,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC;QAC1B,IAAI,CAAC,qBAAqB,EAAE,CAAC;IAC/B,CAAC;IAED;;;;OAIG;IACH,sBAAsB;QACpB,OAAO,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;IACvE,CAAC;IAED,mDAAmD;IACnD,kBAAkB,CAAC,WAA8B;QAC/C,IAAI,CAAC,eAAe,GAAG,SAAS,CAAC;QACjC,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;QACvB,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;QACxB,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,SAAS,EAAE,WAAW,CAAC,IAAI,CAAC,CAAC;QAE5E,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,OAAO;QACT,CAAC;QACD,kEAAkE;QAClE,eAAe;QACf,IAAI,IAAI,CAAC,gBAAgB,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,IAAI,SAAS,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,EAAE,CAAC;YACjF,OAAO,CAAC,KAAK,CAAC,6BAA6B,CAAC,CAAC;YAC7C,IAAI,CAAC,eAAe,GAAG,IAAI,wBAAwB,CACjD,SAAS,CAAC,SAAS,EACnB,IAAI,CAAC,wBAAwB,EAC7B,IAAI,CAAC,wBAAwB,CAAC,MAAM,EAAE,EACtC,IAAI,CAAC,gBAAgB,EAAE,CACxB,CAAC;YACF,IAAI,CAAC,eAAe,CAAC,QAAQ,GAAG,IAAI,CAAC,WAAW,EAAG,CAAC,QAAQ,CAAC;QAC/D,CAAC;QAED,MAAM,IAAI,GAAG,IAAI,eAAe,CAAC,EAAE,CAAC,CAAC;QACrC,IAAI,SAAS,EAAE,QAAQ,CAAC,CAAC,CAAC,EAAE,QAAQ,EAAE,CAAC;YACrC,IAAI,CAAC,eAAe,CAAC,IAAI,EAAE,SAAS,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC;QAC5D,CAAC;QACD,IAAI,SAAS,EAAE,QAAQ,CAAC,CAAC,CAAC,EAAE,QAAQ,EAAE,CAAC;YACrC,IAAI,CAAC,eAAe,CAAC,IAAI,EAAE,SAAS,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC;QAC5D,CAAC;QACD,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,wBAAwB,CAAC,kBAAkB,CAAC,IAAI,CAAC,CAAC;QAEvD,UAAU,CAAC,GAAG,EAAE;YACd,IAAI,CAAC,IAAI,CAAC,IAAI;gBAAE,OAAO;YACvB,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC/B,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,qBAAqB,EAAE,CAAC,CAAC,CAAC;YAEvH,IAAI,CAAC,mBAAmB,CAAC,WAAW,EAAE,SAAS,CAAC,CAAC;YAEjD,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC;YACxB,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC,CAAC;QACpC,CAAC,EAAE,GAAG,CAAC,CAAC;IACV,CAAC;IAED,cAAc,CAAC,IAAS;QACtB,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC;YACd,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,EAAE,CAAC;gBAC7C,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;gBACnC,IAAI,OAAO,CAAC,QAAQ,EAAE,CAAC;oBACrB,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC;gBAC/B,CAAC;qBAAM,CAAC;oBACN,IAAI,CAAC,kBAAkB,CAAC,OAAO,CAAC,cAAc,CAAC,IAAI,CAAC,GAAG,OAAO,CAAC,SAAS,CAAC;oBACzE,OAAO,CAAC,aAAa,CAAC,UAAU,CAAC,OAAO,CAAC,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,kBAAkB,CAAC,OAAO,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;oBACtJ,OAAO,CAAC,sBAAsB,EAAE,CAAC;gBACnC,CAAC;YACH,CAAC;QACH,CAAC;IACH,CAAC;IAED,qBAAqB,CAAC,OAAO,GAAG,IAAI;QAClC,sCAAsC;QACtC,MAAM,eAAe,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAC3C,MAAM,SAAS,GAAG,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,IAAI,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;QACtG,IAAI,CAAC,QAAQ,GAAG,eAAe,CAAC;QAChC,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC;YACd,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC;gBACtB,OAAO,EAAE,IAAI,CAAC,IAAI,CAAC,OAAO;gBAC1B,KAAK,EAAE,IAAI,CAAC,IAAI,CAAC,KAAK;gBACtB,IAAI,EAAE,IAAI,CAAC,QAAQ;gBACnB,gBAAgB,EAAE,SAAS;aAC5B,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED;;;;;;;;;OASG;IACH,wBAAwB,CA4DtB;IAEF,wBAAwB,CAAC,EAAO,EAAE,MAAW;QAC3C,MAAM,MAAM,GAAG,MAAM,CAAC,QAAQ,CAAC;QAC/B,MAAM,UAAU,GAAG,EAAE,CAAC,cAAc,CAAC,IAAI,CAAC;QAC1C,uEAAuE;QACvE,QAAQ,UAAU,EAAE,CAAC;YACnB,KAAK,cAAc,CAAC,CAAC,CAAC;gBACpB,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,cAAc,EAAE,MAAM,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;oBAC3D,EAAE,CAAC,cAAc,CAAC,QAAQ,GAAG,CAAC,CAAC;gBACjC,CAAC,CAAC,CAAC;gBACH,MAAM;YACR,CAAC;YACD,KAAK,OAAO,CAAC,CAAC,CAAC;gBACb,MAAM,iBAAiB,GAA4B,EAAE,CAAC;gBACtD,EAAE,CAAC,cAAc,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAM,EAAE,EAAE;oBAC5C,IAAI,CAAC,CAAC,IAAI,KAAK,cAAc,IAAI,CAAC,CAAC,IAAI,KAAK,YAAY,EAAE,CAAC;wBACzD,iBAAiB,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;oBAChC,CAAC;gBACH,CAAC,CAAC,CAAC;gBACH,IAAI,MAAM,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,MAAM,EAAE,CAAC;oBAC1C,MAAM,CAAC,OAAO,CAAC,CAAC,OAAY,EAAE,EAAE;wBAC9B,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE;4BACnC,IAAI,iBAAiB,CAAC,GAAG,CAAC,EAAE,CAAC;gCAC3B,IAAI,CAAC,YAAY,CAAC,iBAAiB,CAAC,GAAG,CAAC,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;oCACtE,IAAI,CAAC,EAAE,CAAC;wCACN,OAAO,CAAC,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC;oCAC7B,CAAC;yCAAM,CAAC;wCACN,OAAO,OAAO,CAAC,GAAG,GAAG,OAAO,CAAC,CAAC;oCAChC,CAAC;oCACD,IAAI,CAAC,iBAAiB,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;gCACrC,CAAC,CAAC,CAAC;4BACL,CAAC;wBACH,CAAC,CAAC,CAAC;oBACL,CAAC,CAAC,CAAC;gBACL,CAAC;gBACD,MAAM;YACR,CAAC;QACH,CAAC;QACD,IAAI,CAAC,iBAAiB,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;IACrC,CAAC;IAED,iBAAiB,CAAC,EAAO,EAAE,QAAa;QACtC,EAAE,CAAC,cAAc,CAAC,KAAK,GAAG,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;QAC9C,EAAE,CAAC,UAAU,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC;QACnC,EAAE,CAAC,sBAAsB,EAAE,CAAC;QAC5B,EAAE,CAAC,WAAW,EAAE,CAAC;IACnB,CAAC;IAED,YAAY,CAAC,WAAgB,EAAE,QAAa;QAC1C,IAAI,QAAQ,EAAE,CAAC;YACb,QAAQ,WAAW,CAAC,IAAI,EAAE,CAAC;gBACzB,KAAK,cAAc,CAAC,CAAC,CAAC;oBACpB,OAAO,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;gBACjD,CAAC;YACH,CAAC;QACH,CAAC;QACD,OAAO,EAAE,CAAC,IAAI,CAAC,CAAC;IAClB,CAAC;IAED,eAAe,CAAC,SAA4B;QAC1C,IAAI,KAAU,CAAC;QACf,IAAI,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;YAC7B,KAAK,GAAG,EAAE,CAAC;YACX,wDAAwD;YACxD,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACtE,CAAC;aAAM,CAAC;YACN,KAAK,GAAG,SAAS,CAAC;QACpB,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;;;;;OAMG;IACH,eAAe,CAAC,WAA4B,EAAE,WAAgB,EAAE,OAAgB;QAC9E,MAAM,EAAE,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAC9B,IAAI,CAAC,EAAE;YAAE,OAAO;QAEhB,IAAI,IAAI,CAAC;QACT,IAAI,IAAI,CAAC;QAET,mBAAmB;QACnB,IAAI,WAAW,CAAC,IAAI,KAAK,UAAU,IAAI,WAAW,CAAC,IAAI,KAAK,eAAe,EAAE,CAAC;YAC5E,mCAAmC;YACnC,MAAM,WAAW,GAAG,WAAW,CAAC,IAAI,IAAI,CAAC,WAAW,CAAC,IAAI,KAAK,MAAM,IAAI,WAAW,CAAC,IAAI,KAAK,MAAM,CAAC,CAAC;YACrG,IAAI,CAAC,WAAW,IAAI,CAAC,CAAC,WAAW,CAAC,QAAQ,IAAI,WAAW,CAAC,QAAQ,CAAC,MAAM,KAAK,CAAC,CAAC,EAAE,CAAC;gBACjF,OAAO,CAAC,KAAK,CAAC,wBAAwB,EAAE,WAAW,CAAC,CAAC;gBACrD,OAAO;YACT,CAAC;YAED,IAAI,GAAG,IAAI,eAAe,CAAC,EAAE,CAAC,CAAC;YAC/B,IAAI,CAAC,YAAY,GAAG;gBAClB,MAAM,EAAE,WAAW,CAAC,MAAM;gBAC1B,IAAI,EAAE,WAAW,CAAC,IAAI;aACvB,CAAC;YAEF,IAAI,WAAW,CAAC,IAAI,EAAE,CAAC;gBACrB,IAAI,IAAI,CAAC,oBAAoB,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC;oBACzD,IAAI,CAAC,YAAY,CAAC,KAAK,GAAG,WAAW,CAAC,IAAI,CAAC;gBAC7C,CAAC;qBAAM,IAAI,CAAC,WAAW,CAAC,KAAK,EAAE,CAAC;oBAC9B,mDAAmD;oBACnD,IAAI,CAAC,YAAY,CAAC,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC,oBAAoB,CAAC,GAAG,WAAW,CAAC,IAAI,QAAQ,CAAC,IAAI,WAAW,CAAC,IAAI,CAAC;gBACtH,CAAC;qBAAM,CAAC;oBACN,uCAAuC;oBACvC,IAAI,CAAC,YAAY,CAAC,KAAK,GAAG,WAAW,CAAC,KAAK,CAAC;gBAC9C,CAAC;YACH,CAAC;YAED,IAAI,OAAO,KAAK,MAAM,IAAI,OAAO,KAAK,MAAM,EAAE,CAAC;gBAC7C,IAAI,CAAC,YAAY,CAAC,KAAK,GAAG,OAAO,CAAC;YACpC,CAAC;YAED,KAAK,MAAM,CAAC,IAAI,WAAW,CAAC,QAAQ,EAAE,CAAC;gBACrC,IAAI,CAAC,eAAe,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;YAChC,CAAC;YACD,IAAI,GAAG,OAAO,IAAI,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC;QACzC,CAAC;aAAM,CAAC;YACN,mBAAmB;YACnB,sEAAsE;YACtE,uCAAuC;YACvC,IAAI,GAAG,IAAI,wBAAwB,CAAC,EAAE,CAAC,CAAC;YACxC,IAAI,CAAC,qBAAqB,GAAG;gBAC3B,sCAAsC;gBACtC,WAAW,EAAE,WAAW,CAAC,IAAI;gBAC7B,SAAS,EAAE,EAAE,CAAC,SAAS,CAAC,SAAS;aAClC,CAAC;YAEF,oDAAoD;YACpD,wCAAwC;YACxC,MAAM,KAAK,GAAG,WAAW,EAAE,KAAK;gBAC9B,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,WAAW,EAAE,KAAK,CAAC;gBAC1C,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,WAAW,EAAE,YAAY,CAAC,IAAI,EAAE,CAAC,SAAS,CAAC,SAAS,KAAK,SAAS,CAAC,MAAM;oBACxF,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,WAAW,EAAE,YAAY,CAAC;oBACjD,CAAC,CAAC,WAAW,EAAE,KAAK,CAAC;YAEzB,iCAAiC;YACjC,MAAM,eAAe,GAAG,EAAE,CAAC,QAAQ,IAAI,CAAC,CAAC,WAAW,CAAC,QAAQ,CAAC;YAC9D,MAAM,WAAW,GAAG,IAAI,iBAAiB,CAAC;gBACxC,KAAK;gBACL,QAAQ,EAAE,eAAe;aAC1B,CAAC,CAAC;YAEH,oEAAoE;YACpE,IAAI,CAAC,WAAW,CAAC,KAAK,EAAE,CAAC;gBACvB,2FAA2F;gBAC3F,IAAI,WAAW,CAAC,QAAQ,EAAE,CAAC;oBACzB,WAAW,CAAC,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC,oBAAoB,CAAC,GAAG,WAAW,CAAC,QAAQ,QAAQ,CAAC,IAAI,WAAW,CAAC,QAAQ,CAAC;oBACtH,WAAW,CAAC,WAAW,GAAG,IAAI,CAAC,cAAc,CAAC,oBAAoB,CAAC,GAAG,WAAW,CAAC,QAAQ,cAAc,CAAC,CAAC;gBAC5G,CAAC;qBAAM,CAAC;oBACN,yFAAyF;oBACzF,WAAW,CAAC,KAAK,GAAG,WAAW,CAAC,IAAI,CAAC;gBACvC,CAAC;YACH,CAAC;YACD,WAAW,CAAC,QAAQ,GAAG,eAAe,CAAC;YACvC,sEAAsE;YACtE,6CAA6C;YAC7C,WAAW,CAAC,aAAa,GAAG,IAAI,CAAC,cAAc,CAAC,0BAA0B,CAAC,WAAW,CAAC,IAAI,EAAE,WAAW,CAAC,eAAe,CAAC,CAAC;YAE1H,WAAW,CAAC,cAAc,GAAG,WAAW,CAAC;YACzC,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,IAAI,CAAC,GAAG,WAAW,CAAC;YAEnD,IAAI,WAAW,CAAC,IAAI,KAAK,OAAO,EAAE,CAAC;gBACjC,iDAAiD;gBACjD,uEAAuE;gBACvE,sDAAsD;gBACtD,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,cAAc,CAAC,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE,CAAC;oBACnE,WAAW,CAAC,cAAc,CAAC,SAAS,GAAG,IAAI,CAAC;gBAC9C,CAAC;gBAED,IAAI,IAAI,CAAC,eAAe,IAAI,WAAW,CAAC,cAAc,CAAC,KAAK,EAAE,CAAC;oBAC7D,0EAA0E;oBAC1E,8EAA8E;oBAC9E,MAAM,WAAW,GAAG,WAAW,CAAC,cAAc,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;oBAChF,WAAW,CAAC,cAAc,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,QAAa,EAAE,EAAE;wBACzD,WAAW,CAAC,OAAO,CAAC,CAAC,UAAkB,EAAE,EAAE;4BACzC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE,CAAC;gCAChD,QAAQ,CAAC,UAAU,CAAC,GAAG,IAAI,CAAC;4BAC9B,CAAC;wBACH,CAAC,CAAC,CAAC;oBACL,CAAC,CAAC,CAAC;oBACH,mEAAmE;oBACnE,WAAW,CAAC,UAAU,CAAC,WAAW,CAAC,cAAc,CAAC,KAAK,EAAE;wBACvD,QAAQ,EAAE,IAAI;wBACd,SAAS,EAAE,KAAK;wBAChB,qBAAqB,EAAE,KAAK;wBAC5B,qBAAqB,EAAE,KAAK;qBAC7B,CAAC,CAAC;gBACL,CAAC;YACH,CAAC;YAED,IAAI,WAAW,CAAC,aAAa,KAAK,qBAAqB,EAAE,CAAC;gBACxD,WAAW,CAAC,cAAc,CAAC,SAAS,GAAG,CAAC,YAAiB,EAAE,EAAE;oBAC3D,WAAW,CAAC,cAAc,CAAC,MAAM,GAAG,YAAY,CAAC;gBACnD,CAAC,CAAC;YACJ,CAAC;YAED,eAAe,CAAC,iBAAiB,CAAC,WAAW,CAAC,CAAC;YAE/C,4BAA4B;YAC5B,MAAM,IAAI,GAAG,WAAW,CAAC,cAAc,CAAC,WAAW,CAAC;YACpD,IAAI,IAAI,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACrC,WAAW,CAAC,cAAc,CAAC,WAAW,GAAG,IAAI,CAAC;YAChD,CAAC;YAED,sEAAsE;YACtE,qCAAqC;YACrC,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,IAAI,CAAC,GAAG,WAAW,CAAC,cAAc,CAAC;YAEjE,+DAA+D;YAC/D,sCAAsC;YACtC,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC;YAC/D,IAAI,CAAC,cAAc,CAAC,IAAI,CACtB,YAAY,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;gBAC3B,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;oBACzB,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;gBACvC,CAAC;YACH,CAAC,CAAC,CACH,CAAC;YAEF,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,IAAI,EAAE,WAAW,CAAC,CAAC;YAC/C,IAAI,GAAG,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,IAAI,CAAC;QAC7C,CAAC;QACD,WAAW,CAAC,UAAU,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;IACrC,CAAC;IAED;;;;;OAKG;IACH,cAAc,CAAC,WAAgB;QAC7B,MAAM,EAAE,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAC9B,MAAM,aAAa,GAAG,cAAc,CAAC,aAAa,CAAC,WAAW,EAAE,EAAE,EAAE,SAAS,CAAC,SAAS,CAAC,CAAC;QAEzF,gDAAgD;QAChD,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;YACzB,aAAa,CAAC,IAAI,CAAC,cAAc,CAAC,yBAAyB,CAAC,CAAC;QAC/D,CAAC;QAED,OAAO,aAAa,CAAC;IACvB,CAAC;IAED;;OAEG;IACH,WAAW;QACT,MAAM,EAAE,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAC9B,OAAO,EAAE,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,kBAAkB,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC,SAAS,CAAC,SAAS,EAAE,EAAE,CAAC,IAAI,EAAE,IAAI,CAAC,gBAAgB,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;IAC7I,CAAC;IAED;;;;;OAKG;IACH,WAAW,CAAC,KAAU,EAAE,IAAS;QAC/B,IAAI,KAAK,IAAI,IAAI,EAAE,CAAC;YAClB,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;QAC/C,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED,uDAAuD;IACvD,iBAAiB,CAAC,QAAe,EAAE,IAA6B;QAC9D,QAAQ,EAAE,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;YAC5B,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,OAAO,CAAC,EAAE,CAAC;gBAClC,OAAO,CAAC,KAAK,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;YAChD,CAAC;iBAAM,CAAC;gBACN,OAAO,OAAO,CAAC,KAAK,CAAC;YACvB,CAAC;YACD,IAAI,OAAO,CAAC,IAAI,KAAK,OAAO,IAAI,OAAO,CAAC,QAAQ,IAAI,OAAO,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAChF,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;YACjD,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,SAAS,CAAC,IAA6B,EAAE,OAAY;QACnD,+DAA+D;QAC/D,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC;YACxB,OAAO,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,QAAQ,KAAK,OAAO,CAAC,EAAE,CAAC,CAAC;QACjE,CAAC;aAAM,CAAC;YACN,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QAClD,CAAC;IACH,CAAC;IAED,SAAS,CAAC,IAA6B,EAAE,OAAY;QACnD,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;IAC5B,CAAC;IAED,WAAW;QACT,IAAI,CAAC,cAAc,EAAE,CAAC;QACtB,IAAI,CAAC,wBAAwB,CAAC,oBAAoB,CAAC,IAAI,CAAC,CAAC;IAC3D,CAAC;IAED,eAAe;QACb,IAAI,CAAC,MAAM,CAAC,aAAa,EAAE,CAAC;IAC9B,CAAC;IAED,mEAAmE;IACnE,oEAAoE;IACpE,0EAA0E;IAClE,cAAc;QACpB,IAAI,IAAI,CAAC,cAAc,CAAC,MAAM,EAAE,CAAC;YAC/B,OAAO,CAAC,KAAK,CAAC,oBAAoB,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,GAAG,0BAA0B,CAAC,CAAC;YAC9F,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;YACpD,IAAI,CAAC,cAAc,GAAG,EAAE,CAAC;QAC3B,CAAC;IACH,CAAC;+GA1nBU,mBAAmB;mGAAnB,mBAAmB,8yBATnB,CAAC,iBAAiB,EAAE,uBAAuB,EAAE,uBAAuB,CAAC,0BCrBlF,2dAWA,wmBDSY,YAAY,8BAAE,mBAAmB,4UAAE,wBAAwB;;4FAU1D,mBAAmB;kBAb/B,SAAS;+BACE,iBAAiB,cACf,IAAI,WACP,CAAC,YAAY,EAAE,mBAAmB,EAAE,wBAAwB,CAAC,aAC3D,CAAC,iBAAiB,EAAE,uBAAuB,EAAE,uBAAuB,CAAC,iBAGjE,iBAAiB,CAAC,IAAI,QAC/B;wBACJ,KAAK,EAAE,iBAAiB;wBACxB,cAAc,EAAE,uBAAuB;qBACxC","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { AfterViewInit, ChangeDetectorRef, Component, effect, ElementRef, inject, input, OnDestroy, output, ViewEncapsulation } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { ReactiveFormsModule, ValidatorFn, Validators } from '@angular/forms';\nimport { Situation, SystemService, UserService, Utils } from '@yuuvis/client-core';\nimport { debounceTime, Observable, of, Subscription } from 'rxjs';\nimport { FormScriptingApiService } from './form-scripting.service';\nimport { IObjectFormElementExtension } from './object-form-extension.interface';\nimport { ObjectFormGroupComponent } from './object-form-group/object-form-group.component';\nimport { ObjectFormScriptService } from './object-form-script/object-form-script.service';\nimport { ObjectFormScriptingScope } from './object-form-script/object-form-scripting-scope';\nimport { FormStatusChangedEvent, ObjectFormControlWrapper, ObjectFormModelChange, ObjectFormOptions } from './object-form.interface';\nimport { ObjectFormControl, ObjectFormGroup } from './object-form.model';\nimport { ObjectFormService } from './object-form.service';\nimport { ObjectFormUtils } from './object-form.utils';\nimport { FormValidation } from './object-form.validation';\n\n@Component({\n  selector: 'yuv-object-form',\n  standalone: true,\n  imports: [CommonModule, ReactiveFormsModule, ObjectFormGroupComponent],\n  providers: [ObjectFormService, ObjectFormScriptService, FormScriptingApiService],\n  templateUrl: './object-form.component.html',\n  styleUrl: './object-form.component.scss',\n  encapsulation: ViewEncapsulation.None,\n  host: {\n    class: 'yuv-object-form',\n    '[attr.inert]': 'inert() ? true : null'\n  }\n})\nexport class ObjectFormComponent implements OnDestroy, AfterViewInit {\n  #elementRef = inject(ElementRef);\n  #systemService = inject(SystemService);\n  #formScriptService = inject(ObjectFormScriptService);\n  #formHelperService = inject(ObjectFormService);\n  #userService = inject(UserService);\n  #cdRef = inject(ChangeDetectorRef);\n  #formScriptingApiService = inject(FormScriptingApiService);\n\n  #skipTranslationsFor = ['core', 'data'];\n  #namePrefix = 'fg_';\n  gCount = 0;\n\n  id = Utils.uuid();\n\n  fo?: ObjectFormOptions;\n  formOptions = input<ObjectFormOptions>();\n  #formOptionsEffect = effect(() => {\n    this.#init(this.formOptions());\n  });\n\n  inert = input<boolean>();\n  elementExtensions = input<IObjectFormElementExtension[]>([]);\n  #elementExtensionsEffect = effect(() => {\n    this.#formHelperService.setElementExtensions(this.elementExtensions());\n  });\n\n  /**\n   * There are special scenarios where forms are within a form themselves.\n   * Setting this property to true, will handle the current form in a\n   * slightly different way when it comes to form scripting.\n   */\n  isInnerTableForm = input<boolean>();\n\n  /**\n   * triggered when the forms state has been changed\n   */\n  statusChanged = output<FormStatusChangedEvent>();\n\n  /**\n   * handler to be executed after the form has been set up\n   */\n  onFormReady = output();\n\n  // the actual form instance\n  form?: ObjectFormGroup;\n  // property for holding the forms data used for comparison when a form-changed-event\n  // is fetched to indicate wher or not the indexdata were changed or just the properties\n  // of the form elements (eg. form script setting fields to readonly)\n  formData: any;\n  #scriptModel: any = {};\n  #scriptingScope?: ObjectFormScriptingScope;\n  #subscriptions: Subscription[] = [];\n\n  // local store for all the form control references\n  #formControls: any = {};\n  #initialValidators: any = {};\n\n  constructor() {\n    this.#formScriptingApiService.formId = this.id;\n    this.#formScriptingApiService.api.events\n      .on(FormScriptingApiService.EVENT_MODEL_CHANGED)\n      .pipe(takeUntilDestroyed())\n      .subscribe((event) => event.data && this.#onScriptingModelChanged(event.data.formControlName, event.data.change, event.data.formId));\n\n    this.#formScriptingApiService.formId = this.id;    \n  }\n\n  // initialize the form based on the provided form options\n  #init(fo: ObjectFormOptions | undefined) {\n    this.form = undefined;\n    if (!fo) return;\n    setTimeout(() => {\n      this.#initialValidators = {};\n      this.unsubscribeAll();\n      this.#buildReactiveForm(fo);\n    }, 0);\n  }\n\n  #initScriptingScope(formOptions: ObjectFormOptions, dataFormModel: any) {\n    const { data, actions, objects, context } = (formOptions || {}) as ObjectFormOptions;\n\n    if (this.#scriptingScope) {\n      this.#scriptingScope.setModel(this.#scriptModel);\n      /** provide access to actions (used inside of BPM-Forms) */\n      this.#scriptingScope.actions = actions;\n      /** provide access to additional objects (used for example in BPM-Start-Forms to\n       * add data of DMS-Objects to start the process for)\n       */\n      this.#scriptingScope.objects = objects || [];\n      this.#scriptingScope.context = context || { id: '', title: '', objectTypeId: '' };\n      /** provide readonly access to initial form data (which may also contain values that\n       * are not rendered as form elements (invisible values))\n       */\n      this.#scriptingScope.data = data;\n\n      /** by default, scripting scopes are applied to forms. But table elements create their own scope\n       * for editing rows. Being one of those inner forms should not run the form script again, but\n       * instead just provide the observing abilities of the scripting scope.\n       */\n      if (!this.isInnerTableForm() && dataFormModel) {\n        const scriptName = dataFormModel.name + '_' + dataFormModel.situation;\n        console.debug('executing form script ' + scriptName);\n        this.#formScriptService.runFormScript(this.#scriptingScope, dataFormModel.script, scriptName);\n      }\n    }\n  }\n\n  focusForm() {\n    this.#elementRef.nativeElement.querySelector('input').focus();\n  }\n\n  getGroup(id: string): ObjectFormGroup {\n    return this.form?.get(id) as ObjectFormGroup;\n  }\n\n  setFormData(data: Record<string, unknown>) {\n    const fo = this.formOptions();\n    if (!fo) return;\n    fo.data = data;\n    setTimeout(() => this.#init(fo), 0);\n  }\n\n  patchValue(data: Record<string, unknown>) {\n    Object.keys(data).forEach((key) => {\n      this.#onScriptingModelChanged(key, {\n        name: 'value',\n        newValue: data[key]\n      });\n    });\n  }\n\n  /**\n   * Extracts the values from the form model. Each form value is represented by one\n   * property on the result object holding the fields value. The keys (properties) are the `name`\n   * properties of the form element.\n   *\n   * How values are extracted is influenced by the forms situation.\n   *\n   * @return object of key value pairs\n   */\n  getFormData() {\n    return this.#formToData();\n  }\n\n  getFormElements() {\n    const formElements: any = {};\n    Object.keys(this.#formControls).forEach((k) => {\n      formElements[k] = this.#formControls[k].__eoFormElement;\n    });\n    return formElements;\n  }\n\n  setFormPristine() {\n    if (!this.form) return;\n    this.form.markAsPristine();\n  }\n\n  // reset the form to its initial state\n  resetForm() {\n    const fo = this.formOptions();\n    if (!fo) return;\n    this.setFormData(fo.data);\n    this.#emitFormChangedEvent();\n  }\n\n  /**\n   * Returns the observed model that was passed to the current form script running. If there is\n   * no form script, this method will return NULL.\n   * @returns\n   */\n  getObservedScriptModel() {\n    return this.#scriptingScope ? this.#scriptingScope.getModel() : null;\n  }\n\n  // Create a reactive form from the enaio form model\n  #buildReactiveForm(formOptions: ObjectFormOptions) {\n    this.#scriptingScope = undefined;\n    this.#scriptModel = {};\n    this.#formControls = {};\n    const formModel = this.#dataToForm(formOptions.formModel, formOptions.data);\n\n    if (!formModel) {\n      return;\n    }\n    // if a script is available, we'll init the form scripting for the\n    // current form\n    if (this.isInnerTableForm() || (formModel.script && formModel.script.length > 0)) {\n      console.debug('adding form scripting scope');\n      this.#scriptingScope = new ObjectFormScriptingScope(\n        formModel.situation,\n        this.#onScriptingModelChanged,\n        this.#formScriptingApiService.getApi(),\n        this.isInnerTableForm()\n      );\n      this.#scriptingScope.objectId = this.formOptions()!.objectId;\n    }\n\n    const form = new ObjectFormGroup({});\n    if (formModel?.elements[0]?.elements) {\n      this.#addFormControl(form, formModel.elements[0], 'core');\n    }\n    if (formModel?.elements[1]?.elements) {\n      this.#addFormControl(form, formModel.elements[1], 'data');\n    }\n    this.form = form;\n    this.#formScriptingApiService.registerActiveForm(this);\n\n    setTimeout(() => {\n      if (!this.form) return;\n      this.initValidators(this.form);\n      this.#subscriptions.push(this.form.valueChanges.pipe(debounceTime(500)).subscribe(() => this.#emitFormChangedEvent()));\n\n      this.#initScriptingScope(formOptions, formModel);\n\n      this.onFormReady.emit();\n      this.#emitFormChangedEvent(false);\n    }, 300);\n  }\n\n  initValidators(form: any) {\n    if (this.form) {\n      for (const key of Object.keys(form.controls)) {\n        const control = form.controls[key];\n        if (control.controls) {\n          this.initValidators(control);\n        } else {\n          this.#initialValidators[control._eoFormElement.name] = control.validator;\n          control.setValidators(Validators.compose(this.#getValidators(control._eoFormElement).concat([this.#initialValidators[control._eoFormElement.name]])));\n          control.updateValueAndValidity();\n        }\n      }\n    }\n  }\n\n  #emitFormChangedEvent(compare = true) {\n    // check if indexdata has been changed\n    const currentFormData = this.getFormData();\n    const idxChange = compare ? JSON.stringify(this.formData) !== JSON.stringify(currentFormData) : false;\n    this.formData = currentFormData;\n    if (this.form) {\n      this.statusChanged.emit({\n        invalid: this.form.invalid,\n        dirty: this.form.dirty,\n        data: this.formData,\n        indexdataChanged: idxChange\n      });\n    }\n  }\n\n  /**\n   * This method will be called each time the script changes its internal model.\n   * It is used to transfer the script changes to the actual form model.\n   *\n   * To ensure the right context, we define an instance method as callback for the scripting scope\n   * @see: https://blog.johnnyreilly.com/2014/04/typescript-instance-methods.html\n   *\n   * @param formControlName\n   * @param change\n   */\n  #onScriptingModelChanged = (formControlName: string, change: ObjectFormModelChange, formId?: string) => {\n    if (!this.form || (formId && this.id !== formId)) return;\n    // find the target control\n    const fc: ObjectFormControl = this.#formControls[formControlName] as ObjectFormControl;\n    if (fc) {\n      // change only allowed properties\n      switch (change.name) {\n        case 'value': {\n          if (Array.isArray(change.newValue)) {\n            this.#processArrayValueChange(fc, change);\n          } else {\n            fc._eoFormElement.value = change.newValue;\n            if (fc.value !== change.newValue) {\n              fc.patchValue(change.newValue);\n              fc.updateValueAndValidity();\n              fc.markAsDirty();\n            }\n          }\n          this.form.markAsDirty();\n          break;\n        }\n        case 'required': {\n          fc._eoFormElement.required = change.newValue;\n          // apply new validators\n          // @see: https://scotch.io/tutorials/how-to-implement-conditional-validation-in-angular-2-model-driven-forms\n          fc.setValidators(Validators.compose(this.#getValidators(fc._eoFormElement).concat([this.#initialValidators[fc._eoFormElement.name]])));\n          // need to mark form control as touched because otherwise form validation will not show\n          // error messages\n          fc.markAsTouched();\n          fc.updateValueAndValidity();\n          break;\n        }\n        case 'readonly': {\n          fc._eoFormElement.readonly = change.newValue;\n          if (change.newValue === true) {\n            fc.disable();\n          } else {\n            fc.enable();\n          }\n          break;\n        }\n        case 'error': {\n          fc._eoFormElement.error = change.newValue;\n          fc.markAsTouched();\n          fc.updateValueAndValidity();\n          break;\n        }\n        // new onrowedit function was applied by the script\n        case 'onrowedit': {\n          fc._eoFormElement.onrowedit = change.newValue;\n          break;\n        }\n        // new onchange function was applied by the script\n        case 'onchange': {\n          fc._eoFormElement.onchange = change.newValue;\n          break;\n        }\n      }\n      this.#cdRef.detectChanges();\n    }\n  };\n\n  #processArrayValueChange(fc: any, change: any) {\n    const newVal = change.newValue;\n    const targetType = fc._eoFormElement.type;\n    // for some types we have to ensure that meta data are provided as well\n    switch (targetType) {\n      case 'ORGANIZATION': {\n        this.#getDataMeta(fc._eoFormElement, newVal).subscribe((m) => {\n          fc._eoFormElement.dataMeta = m;\n        });\n        break;\n      }\n      case 'TABLE': {\n        const dataToBeProcessed: Record<string, unknown> = {};\n        fc._eoFormElement.elements.forEach((e: any) => {\n          if (e.type === 'ORGANIZATION' || e.type === 'CODESYSTEM') {\n            dataToBeProcessed[e.name] = e;\n          }\n        });\n        if (Object.keys(dataToBeProcessed).length) {\n          newVal.forEach((rowData: any) => {\n            Object.keys(rowData).forEach((key) => {\n              if (dataToBeProcessed[key]) {\n                this.#getDataMeta(dataToBeProcessed[key], rowData[key]).subscribe((m) => {\n                  if (m) {\n                    rowData[key + '_meta'] = m;\n                  } else {\n                    delete rowData[key + '_meta'];\n                  }\n                  this.#updateArrayValue(fc, newVal);\n                });\n              }\n            });\n          });\n        }\n        break;\n      }\n    }\n    this.#updateArrayValue(fc, newVal);\n  }\n\n  #updateArrayValue(fc: any, newValue: any) {\n    fc._eoFormElement.value = [].concat(newValue);\n    fc.patchValue([].concat(newValue));\n    fc.updateValueAndValidity();\n    fc.markAsDirty();\n  }\n\n  #getDataMeta(formElement: any, newValue: any): Observable<any> {\n    if (newValue) {\n      switch (formElement.type) {\n        case 'ORGANIZATION': {\n          return this.#userService.getUserById(newValue);\n        }\n      }\n    }\n    return of(null);\n  }\n\n  #patchFormValue(formValue: string[] | string) {\n    let value: any;\n    if (Array.isArray(formValue)) {\n      value = [];\n      // copy by value for arrays of objects (e.g. table data)\n      formValue.forEach((o) => value.push(JSON.parse(JSON.stringify(o))));\n    } else {\n      value = formValue;\n    }\n    return value;\n  }\n\n  /**\n   * Recursive method adding a new FormControl (group or control) to a parent form group\n   *\n   * @param parentGroup - the parent group to add the control to\n   * @param formElement - the enaio form model element to create the child control from\n   * @param [useName] - use this name instead of the one from the model\n   */\n  #addFormControl(parentGroup: ObjectFormGroup, formElement: any, useName?: string) {\n    const fo = this.formOptions();\n    if (!fo) return;\n\n    let ctrl;\n    let name;\n\n    // add a form group\n    if (formElement.type === 'o2mGroup' || formElement.type === 'o2mGroupStack') {\n      // do not add groups that are empty\n      const isRootGroup = formElement.name && (formElement.name !== 'data' || formElement.name !== 'core');\n      if (!isRootGroup && (!formElement.elements || formElement.elements.length === 0)) {\n        console.error('Found empty form group', formElement);\n        return;\n      }\n\n      ctrl = new ObjectFormGroup({});\n      ctrl._eoFormGroup = {\n        layout: formElement.layout,\n        type: formElement.type\n      };\n\n      if (formElement.name) {\n        if (this.#skipTranslationsFor.includes(formElement.name)) {\n          ctrl._eoFormGroup.label = formElement.name;\n        } else if (!formElement.label) {\n          // no label, so try to translate the technical name\n          ctrl._eoFormGroup.label = this.#systemService.getLocalizedResource(`${formElement.name}_label`) || formElement.name;\n        } else {\n          // there is a label so we take this one\n          ctrl._eoFormGroup.label = formElement.label;\n        }\n      }\n\n      if (useName === 'core' || useName === 'data') {\n        ctrl._eoFormGroup.label = useName;\n      }\n\n      for (const e of formElement.elements) {\n        this.#addFormControl(ctrl, e);\n      }\n      name = useName || 'fg' + this.gCount++;\n    } else {\n      // add form control\n      // To be able to integrate recursive form controls into the main form,\n      // we have to wrap them in a form group\n      ctrl = new ObjectFormControlWrapper({});\n      ctrl._eoFormControlWrapper = {\n        // the name of the wrapped FormControl\n        controlName: formElement.name,\n        situation: fo.formModel.situation\n      };\n\n      // do not set a reference as the form controls value\n      // otherwise we could not reset the form\n      const value = formElement?.value\n        ? this.#patchFormValue(formElement?.value)\n        : !Utils.isEmpty(formElement?.defaultvalue) && fo.formModel.situation === Situation.CREATE\n          ? this.#patchFormValue(formElement?.defaultvalue)\n          : formElement?.value;\n\n      // create the actual form control\n      const controlDisabled = fo.disabled || !!formElement.readonly;\n      const formControl = new ObjectFormControl({\n        value,\n        disabled: controlDisabled\n      });\n\n      // if form element has a 'label' property it will be shown as is ...\n      if (!formElement.label) {\n        // ... if there is no label but a 'labelKey' property we are going to localize that key ...\n        if (formElement.labelkey) {\n          formElement.label = this.#systemService.getLocalizedResource(`${formElement.labelkey}_label`) || formElement.labelkey;\n          formElement.description = this.#systemService.getLocalizedResource(`${formElement.labelkey}_description`);\n        } else {\n          // ... no 'label' and also no 'labelKey' means that we are showing technical name instead\n          formElement.label = formElement.name;\n        }\n      }\n      formElement.readonly = controlDisabled;\n      // we are using an internal type to distinguish between the components\n      // to be used to render certain form elements\n      formElement._internalType = this.#systemService.getInternalFormElementType(formElement.type, formElement.classifications);\n\n      formControl._eoFormElement = formElement;\n      this.#formControls[formElement.name] = formControl;\n\n      if (formElement.type === 'TABLE') {\n        // Add 'onrowedit' property even if it's not set.\n        // This is required because otherwise mobX-Observer would not recognize\n        // changes to this property applied by the form script\n        if (!Object.keys(formControl._eoFormElement).includes('onrowedit')) {\n          formControl._eoFormElement.onrowedit = null;\n        }\n\n        if (this.#scriptingScope && formControl._eoFormElement.value) {\n          // having a scripting scope and table rows means that we need to set empty\n          // columns to NULL, because otherwise mobX won't be able to track those values\n          const valueFields = formControl._eoFormElement.elements.map((e: any) => e.name);\n          formControl._eoFormElement.value.forEach((rowValue: any) => {\n            valueFields.forEach((valueField: string) => {\n              if (!Object.keys(rowValue).includes(valueField)) {\n                rowValue[valueField] = null;\n              }\n            });\n          });\n          // update form controls value as well to reflect the observed value\n          formControl.patchValue(formControl._eoFormElement.value, {\n            onlySelf: true,\n            emitEvent: false,\n            emitModelToViewChange: false,\n            emitViewToModelChange: false\n          });\n        }\n      }\n\n      if (formElement._internalType === 'string:organization') {\n        formControl._eoFormElement.setFilter = (filterObject: any) => {\n          formControl._eoFormElement.filter = filterObject;\n        };\n      }\n\n      ObjectFormUtils.updateFormElement(formElement);\n\n      // remove empty descriptions\n      const desc = formControl._eoFormElement.description;\n      if (desc && desc.trim().length === 0) {\n        formControl._eoFormElement.description = null;\n      }\n\n      // add the form element to the script model that will be injected into\n      // the forms scripting scope later on\n      this.#scriptModel[formElement.name] = formControl._eoFormElement;\n\n      // apply change listener to the form control, that will trigger\n      // the form elements onChange listener\n      const controlWatch = ctrl.valueChanges.pipe(debounceTime(500));\n      this.#subscriptions.push(\n        controlWatch.subscribe((v) => {\n          if (this.#scriptingScope) {\n            this.#scriptingScope.modelChanged(v);\n          }\n        })\n      );\n\n      ctrl.addControl(formElement.name, formControl);\n      name = this.#namePrefix + formElement.name;\n    }\n    parentGroup.addControl(name, ctrl);\n  }\n\n  /**\n   * Build validators for the given form element to be attached to\n   * a reactive formControl.\n   *\n   * @param formElement - form element object\n   */\n  #getValidators(formElement: any): ValidatorFn[] {\n    const fo = this.formOptions();\n    const elmValidators = FormValidation.getValidators(formElement, fo?.formModel.situation);\n\n    // add custom validator for script enabled forms\n    if (this.#scriptingScope) {\n      elmValidators.push(FormValidation.customScriptingValidation);\n    }\n\n    return elmValidators;\n  }\n\n  /**\n   * Extract the values from the form\n   */\n  #formToData() {\n    const fo = this.formOptions();\n    return fo && this.form ? this.#formHelperService.extractFormData(this.form, fo.formModel.situation, fo.data, this.isInnerTableForm()) : {};\n  }\n\n  /**\n   * Merge data into a form model.\n   *\n   * @param model - form model\n   * @param data - data object or array of SearchFilter objects in case of a search form\n   */\n  #dataToForm(model: any, data: any) {\n    if (model && data) {\n      this.#setElementValues(model.elements, data);\n    }\n    return model;\n  }\n\n  // recursive method for adding values to model elements\n  #setElementValues(elements: any[], data: Record<string, unknown>) {\n    elements?.forEach((element) => {\n      if (this.#hasValue(data, element)) {\n        element.value = this.#getValue(data, element);\n      } else {\n        delete element.value;\n      }\n      if (element.type !== 'TABLE' && element.elements && element.elements.length > 0) {\n        this.#setElementValues(element.elements, data);\n      }\n    });\n  }\n\n  #hasValue(data: Record<string, unknown>, element: any) {\n    // differ between array of SearchFilters and a form data object\n    if (Array.isArray(data)) {\n      return !!data.find((filter) => filter.property === element.id);\n    } else {\n      return Object.keys(data).includes(element.name);\n    }\n  }\n\n  #getValue(data: Record<string, unknown>, element: any) {\n    return data[element.name];\n  }\n\n  ngOnDestroy() {\n    this.unsubscribeAll();\n    this.#formScriptingApiService.unregisterActiveForm(this);\n  }\n\n  ngAfterViewInit() {\n    this.#cdRef.detectChanges();\n  }\n\n  // unsubscribe from all value change listeners for the current form\n  // to avoid memory leaks. This method will also be called every time\n  // a new form is rendered to get rid of the old form element subscriptions\n  private unsubscribeAll() {\n    if (this.#subscriptions.length) {\n      console.debug('unsubscribed from ' + this.#subscriptions.length + ' value change listeners.');\n      this.#subscriptions.forEach((s) => s.unsubscribe());\n      this.#subscriptions = [];\n    }\n  }\n}\n","@let fo = formOptions();\n@if (fo && form) {\n  <div class=\"yuv-object-form\" [formGroup]=\"form\">\n    <div class=\"form-element-wrap core\">\n      <yuv-object-form-group [group]=\"getGroup('core')\" [situation]=\"fo.formModel.situation\"></yuv-object-form-group>\n    </div>\n    <div class=\"form-element-wrap data\">\n      <yuv-object-form-group [group]=\"getGroup('data')\" [situation]=\"fo.formModel.situation\"></yuv-object-form-group>\n    </div>\n  </div>\n}\n"]}
|