@covalent/dynamic-forms 6.3.0 → 6.4.0

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.
Files changed (32) hide show
  1. package/esm2022/lib/dynamic-element.component.mjs +229 -0
  2. package/esm2022/lib/dynamic-elements/dynamic-checkbox/dynamic-checkbox.component.mjs +18 -0
  3. package/esm2022/lib/dynamic-elements/dynamic-datepicker/dynamic-datepicker.component.mjs +26 -0
  4. package/esm2022/lib/dynamic-elements/dynamic-file-input/dynamic-file-input.component.mjs +28 -0
  5. package/esm2022/lib/dynamic-elements/dynamic-input/dynamic-input.component.mjs +27 -0
  6. package/esm2022/lib/dynamic-elements/dynamic-select/dynamic-select.component.mjs +25 -0
  7. package/esm2022/lib/dynamic-elements/dynamic-slide-toggle/dynamic-slide-toggle.component.mjs +18 -0
  8. package/esm2022/lib/dynamic-elements/dynamic-slider/dynamic-slider.component.mjs +30 -0
  9. package/esm2022/lib/dynamic-elements/dynamic-textarea/dynamic-textarea.component.mjs +22 -0
  10. package/esm2022/lib/dynamic-forms.component.mjs +191 -0
  11. package/{esm2020 → esm2022}/lib/dynamic-forms.module.mjs +48 -48
  12. package/{esm2020 → esm2022}/lib/services/dynamic-forms.service.mjs +4 -4
  13. package/fesm2022/covalent-dynamic-forms.mjs +814 -0
  14. package/{fesm2015 → fesm2022}/covalent-dynamic-forms.mjs.map +1 -1
  15. package/lib/dynamic-element.component.d.ts +2 -2
  16. package/lib/dynamic-forms.component.d.ts +1 -1
  17. package/package.json +9 -17
  18. package/esm2020/lib/dynamic-element.component.mjs +0 -176
  19. package/esm2020/lib/dynamic-elements/dynamic-checkbox/dynamic-checkbox.component.mjs +0 -19
  20. package/esm2020/lib/dynamic-elements/dynamic-datepicker/dynamic-datepicker.component.mjs +0 -23
  21. package/esm2020/lib/dynamic-elements/dynamic-file-input/dynamic-file-input.component.mjs +0 -28
  22. package/esm2020/lib/dynamic-elements/dynamic-input/dynamic-input.component.mjs +0 -22
  23. package/esm2020/lib/dynamic-elements/dynamic-select/dynamic-select.component.mjs +0 -23
  24. package/esm2020/lib/dynamic-elements/dynamic-slide-toggle/dynamic-slide-toggle.component.mjs +0 -19
  25. package/esm2020/lib/dynamic-elements/dynamic-slider/dynamic-slider.component.mjs +0 -26
  26. package/esm2020/lib/dynamic-elements/dynamic-textarea/dynamic-textarea.component.mjs +0 -22
  27. package/esm2020/lib/dynamic-forms.component.mjs +0 -185
  28. package/fesm2015/covalent-dynamic-forms.mjs +0 -745
  29. package/fesm2020/covalent-dynamic-forms.mjs +0 -743
  30. package/fesm2020/covalent-dynamic-forms.mjs.map +0 -1
  31. /package/{esm2020 → esm2022}/covalent-dynamic-forms.mjs +0 -0
  32. /package/{esm2020 → esm2022}/public_api.mjs +0 -0
@@ -1,185 +0,0 @@
1
- import { Component, Input, ChangeDetectionStrategy, ChangeDetectorRef, ContentChildren, QueryList, } from '@angular/core';
2
- import { UntypedFormBuilder, } from '@angular/forms';
3
- import { TdDynamicFormsService, } from './services/dynamic-forms.service';
4
- import { TdDynamicFormsErrorTemplateDirective } from './dynamic-element.component';
5
- import { timer, Subject } from 'rxjs';
6
- import { takeUntil, filter } from 'rxjs/operators';
7
- import * as i0 from "@angular/core";
8
- import * as i1 from "@angular/forms";
9
- import * as i2 from "./services/dynamic-forms.service";
10
- import * as i3 from "@angular/common";
11
- import * as i4 from "./dynamic-element.component";
12
- export class TdDynamicFormsComponent {
13
- /**
14
- * elements: ITdDynamicElementConfig[]
15
- * JS Object that will render the elements depending on its config.
16
- * [name] property is required.
17
- */
18
- set elements(elements) {
19
- if (elements) {
20
- this._elements = elements;
21
- }
22
- else {
23
- this._elements = [];
24
- }
25
- this._rerenderElements();
26
- }
27
- get elements() {
28
- return this._renderedElements;
29
- }
30
- /**
31
- * Getter property for dynamic [FormGroup].
32
- */
33
- get form() {
34
- return this.dynamicForm;
35
- }
36
- /**
37
- * Getter property for [valid] of dynamic [FormGroup].
38
- */
39
- get valid() {
40
- if (this.dynamicForm) {
41
- return this.dynamicForm.valid;
42
- }
43
- return false;
44
- }
45
- /**
46
- * Getter property for [value] of dynamic [FormGroup].
47
- */
48
- get value() {
49
- if (this.dynamicForm) {
50
- return this.dynamicForm.value;
51
- }
52
- return {};
53
- }
54
- /**
55
- * Getter property for [errors] of dynamic [FormGroup].
56
- */
57
- get errors() {
58
- if (this.dynamicForm) {
59
- const errors = {};
60
- for (const name of Object.keys(this.dynamicForm.controls)) {
61
- errors[name] = this.dynamicForm.controls[name].errors;
62
- }
63
- return errors;
64
- }
65
- return {};
66
- }
67
- /**
68
- * Getter property for [controls] of dynamic [FormGroup].
69
- */
70
- get controls() {
71
- if (this.dynamicForm) {
72
- return this.dynamicForm.controls;
73
- }
74
- return {};
75
- }
76
- constructor(_formBuilder, _dynamicFormsService, _changeDetectorRef) {
77
- this._formBuilder = _formBuilder;
78
- this._dynamicFormsService = _dynamicFormsService;
79
- this._changeDetectorRef = _changeDetectorRef;
80
- this._renderedElements = [];
81
- this._templateMap = new Map();
82
- this._destroy$ = new Subject();
83
- this._destroyControl$ = new Subject();
84
- this.dynamicForm = this._formBuilder.group({});
85
- }
86
- ngAfterContentInit() {
87
- this._updateErrorTemplates();
88
- }
89
- ngOnDestroy() {
90
- this._destroy$.next(0);
91
- this._destroy$.complete();
92
- this._destroyControl$.complete();
93
- }
94
- /**
95
- * Refreshes the form and rerenders all validator/element modifications.
96
- */
97
- refresh() {
98
- this._rerenderElements();
99
- this._updateErrorTemplates();
100
- }
101
- /**
102
- * Getter method for error template references
103
- */
104
- getErrorTemplateRef(name) {
105
- return this._templateMap.get(name);
106
- }
107
- /**
108
- * Loads error templates and sets them in a map for faster access.
109
- */
110
- _updateErrorTemplates() {
111
- this._templateMap = new Map();
112
- for (const errorTemplate of this._errorTemplates.toArray()) {
113
- this._templateMap.set(errorTemplate.tdDynamicFormsError ?? '', errorTemplate.templateRef);
114
- }
115
- }
116
- _rerenderElements() {
117
- this._clearRemovedElements();
118
- this._renderedElements = [];
119
- const duplicates = [];
120
- this._elements.forEach((elem) => {
121
- this._dynamicFormsService.validateDynamicElementName(elem.name);
122
- if (duplicates.indexOf(elem.name) > -1) {
123
- throw new Error(`Dynamic element name: "${elem.name}" is duplicated`);
124
- }
125
- duplicates.push(elem.name);
126
- const dynamicElement = this.dynamicForm.get(elem.name);
127
- if (!dynamicElement) {
128
- this.dynamicForm.addControl(elem.name, this._dynamicFormsService.createFormControl(elem));
129
- this._subscribeToControlStatusChanges(elem.name);
130
- }
131
- else {
132
- dynamicElement.setValue(elem.default);
133
- dynamicElement.markAsPristine();
134
- dynamicElement.markAsUntouched();
135
- if (elem.disabled) {
136
- dynamicElement.disable();
137
- }
138
- else {
139
- dynamicElement.enable();
140
- }
141
- dynamicElement.setValidators(this._dynamicFormsService.createValidators(elem));
142
- }
143
- // copy objects so they are only changes when calling this method
144
- this._renderedElements.push(Object.assign({}, elem));
145
- });
146
- // call a change detection since the whole form might change
147
- this._changeDetectorRef.detectChanges();
148
- timer(0)
149
- .toPromise()
150
- .then(() => {
151
- // call a markForCheck so elements are rendered correctly in OnPush
152
- this._changeDetectorRef.markForCheck();
153
- });
154
- }
155
- _clearRemovedElements() {
156
- this._renderedElements = this._renderedElements.filter((renderedElement) => !this._elements.some((element) => element.name === renderedElement.name));
157
- // remove elements that were removed from the array
158
- this._renderedElements.forEach((elem) => {
159
- this._destroyControl$.next(elem.name);
160
- this.dynamicForm.removeControl(elem.name);
161
- });
162
- }
163
- // Updates component when manually adding errors to controls
164
- _subscribeToControlStatusChanges(elementName) {
165
- const control = this.controls[elementName];
166
- const controlDestroyed$ = this._destroyControl$.pipe(filter((destroyedElementName) => destroyedElementName === elementName));
167
- control.statusChanges
168
- .pipe(takeUntil(this._destroy$), takeUntil(controlDestroyed$))
169
- .subscribe(() => {
170
- this._changeDetectorRef.markForCheck();
171
- });
172
- }
173
- }
174
- TdDynamicFormsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.5", ngImport: i0, type: TdDynamicFormsComponent, deps: [{ token: i1.UntypedFormBuilder }, { token: i2.TdDynamicFormsService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
175
- TdDynamicFormsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.5", type: TdDynamicFormsComponent, selector: "td-dynamic-forms", inputs: { elements: "elements" }, queries: [{ propertyName: "_errorTemplates", predicate: TdDynamicFormsErrorTemplateDirective, descendants: true }], ngImport: i0, template: "<form [formGroup]=\"dynamicForm\" novalidate>\n <div class=\"td-dynamic-form-wrapper\">\n <ng-template let-element ngFor [ngForOf]=\"elements\">\n <div\n class=\"td-dynamic-element-wrapper\"\n [style.max-width.%]=\"element.flex ? element.flex : 100\"\n [style.flex]=\"'1 1 ' + (element.flex ? element.flex : 100) + '%'\"\n [style.-ms-flex]=\"'1 1 ' + (element.flex ? element.flex : 100) + '%'\"\n [style.-webkit-box-flex]=\"1\"\n >\n <td-dynamic-element\n #dynamicElement\n *ngIf=\"dynamicForm.controls[element.name]\"\n [formControlName]=\"element.name\"\n [dynamicControl]=\"dynamicForm.controls[element.name]\"\n [id]=\"element.name\"\n [name]=\"element.name\"\n [label]=\"element.label || element.name\"\n [hint]=\"element.hint\"\n [type]=\"element.type\"\n [required]=\"element.required ?? false\"\n [min]=\"element.min\"\n [max]=\"element.max\"\n [minLength]=\"element.minLength\"\n [maxLength]=\"element.maxLength\"\n [selections]=\"element.selections\"\n [multiple]=\"element.multiple\"\n [customConfig]=\"element.customConfig\"\n [errorMessageTemplate]=\"getErrorTemplateRef(element.name)\"\n [placeholder]=\"element.placeholder\"\n ></td-dynamic-element>\n </div>\n </ng-template>\n </div>\n <ng-content></ng-content>\n</form>\n", styles: [".td-dynamic-form-wrapper{flex-flow:row wrap;box-sizing:border-box;display:flex;align-items:center;align-content:center;max-width:100%;justify-content:flex-start}.td-dynamic-form-wrapper ::ng-deep .mat-form-field-infix{width:auto}.td-dynamic-form-wrapper ::ng-deep .td-dynamic-element-hint{font-size:75%;display:block}.td-dynamic-form-wrapper .td-dynamic-element-wrapper{max-height:100%;box-sizing:border-box;position:relative;padding:4px 4px 8px}\n"], dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: i4.TdDynamicElementComponent, selector: "td-dynamic-element", inputs: ["dynamicControl", "label", "hint", "name", "type", "required", "min", "max", "minLength", "maxLength", "selections", "multiple", "customConfig", "errorMessageTemplate", "placeholder"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
176
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.5", ngImport: i0, type: TdDynamicFormsComponent, decorators: [{
177
- type: Component,
178
- args: [{ selector: 'td-dynamic-forms', changeDetection: ChangeDetectionStrategy.OnPush, template: "<form [formGroup]=\"dynamicForm\" novalidate>\n <div class=\"td-dynamic-form-wrapper\">\n <ng-template let-element ngFor [ngForOf]=\"elements\">\n <div\n class=\"td-dynamic-element-wrapper\"\n [style.max-width.%]=\"element.flex ? element.flex : 100\"\n [style.flex]=\"'1 1 ' + (element.flex ? element.flex : 100) + '%'\"\n [style.-ms-flex]=\"'1 1 ' + (element.flex ? element.flex : 100) + '%'\"\n [style.-webkit-box-flex]=\"1\"\n >\n <td-dynamic-element\n #dynamicElement\n *ngIf=\"dynamicForm.controls[element.name]\"\n [formControlName]=\"element.name\"\n [dynamicControl]=\"dynamicForm.controls[element.name]\"\n [id]=\"element.name\"\n [name]=\"element.name\"\n [label]=\"element.label || element.name\"\n [hint]=\"element.hint\"\n [type]=\"element.type\"\n [required]=\"element.required ?? false\"\n [min]=\"element.min\"\n [max]=\"element.max\"\n [minLength]=\"element.minLength\"\n [maxLength]=\"element.maxLength\"\n [selections]=\"element.selections\"\n [multiple]=\"element.multiple\"\n [customConfig]=\"element.customConfig\"\n [errorMessageTemplate]=\"getErrorTemplateRef(element.name)\"\n [placeholder]=\"element.placeholder\"\n ></td-dynamic-element>\n </div>\n </ng-template>\n </div>\n <ng-content></ng-content>\n</form>\n", styles: [".td-dynamic-form-wrapper{flex-flow:row wrap;box-sizing:border-box;display:flex;align-items:center;align-content:center;max-width:100%;justify-content:flex-start}.td-dynamic-form-wrapper ::ng-deep .mat-form-field-infix{width:auto}.td-dynamic-form-wrapper ::ng-deep .td-dynamic-element-hint{font-size:75%;display:block}.td-dynamic-form-wrapper .td-dynamic-element-wrapper{max-height:100%;box-sizing:border-box;position:relative;padding:4px 4px 8px}\n"] }]
179
- }], ctorParameters: function () { return [{ type: i1.UntypedFormBuilder }, { type: i2.TdDynamicFormsService }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { _errorTemplates: [{
180
- type: ContentChildren,
181
- args: [TdDynamicFormsErrorTemplateDirective, { descendants: true }]
182
- }], elements: [{
183
- type: Input
184
- }] } });
185
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dynamic-forms.component.js","sourceRoot":"","sources":["../../../../../libs/angular-dynamic-forms/src/lib/dynamic-forms.component.ts","../../../../../libs/angular-dynamic-forms/src/lib/dynamic-forms.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  ContentChildren,\n  TemplateRef,\n  QueryList,\n  AfterContentInit,\n  OnDestroy,\n} from '@angular/core';\nimport {\n  UntypedFormGroup,\n  UntypedFormBuilder,\n  AbstractControl,\n} from '@angular/forms';\n\nimport {\n  TdDynamicFormsService,\n  ITdDynamicElementConfig,\n} from './services/dynamic-forms.service';\nimport { TdDynamicFormsErrorTemplateDirective } from './dynamic-element.component';\n\nimport { timer, Subject, Observable } from 'rxjs';\nimport { takeUntil, filter } from 'rxjs/operators';\n\n@Component({\n  selector: 'td-dynamic-forms',\n  templateUrl: './dynamic-forms.component.html',\n  styleUrls: ['./dynamic-forms.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class TdDynamicFormsComponent implements AfterContentInit, OnDestroy {\n  private _renderedElements: ITdDynamicElementConfig[] = [];\n  private _elements!: ITdDynamicElementConfig[];\n  private _templateMap: Map<string, TemplateRef<any>> = new Map<\n    string,\n    TemplateRef<any>\n  >();\n  private _destroy$: Subject<any> = new Subject();\n  private _destroyControl$: Subject<string> = new Subject();\n\n  @ContentChildren(TdDynamicFormsErrorTemplateDirective, { descendants: true })\n  _errorTemplates!: QueryList<TdDynamicFormsErrorTemplateDirective>;\n  dynamicForm!: UntypedFormGroup;\n\n  /**\n   * elements: ITdDynamicElementConfig[]\n   * JS Object that will render the elements depending on its config.\n   * [name] property is required.\n   */\n  @Input()\n  set elements(elements: ITdDynamicElementConfig[]) {\n    if (elements) {\n      this._elements = elements;\n    } else {\n      this._elements = [];\n    }\n    this._rerenderElements();\n  }\n  get elements(): ITdDynamicElementConfig[] {\n    return this._renderedElements;\n  }\n\n  /**\n   * Getter property for dynamic [FormGroup].\n   */\n  get form(): UntypedFormGroup {\n    return this.dynamicForm;\n  }\n\n  /**\n   * Getter property for [valid] of dynamic [FormGroup].\n   */\n  get valid(): boolean {\n    if (this.dynamicForm) {\n      return this.dynamicForm.valid;\n    }\n    return false;\n  }\n\n  /**\n   * Getter property for [value] of dynamic [FormGroup].\n   */\n  get value(): any {\n    if (this.dynamicForm) {\n      return this.dynamicForm.value;\n    }\n    return {};\n  }\n\n  /**\n   * Getter property for [errors] of dynamic [FormGroup].\n   */\n  get errors(): { [name: string]: any } {\n    if (this.dynamicForm) {\n      const errors: { [name: string]: any } = {};\n      for (const name of Object.keys(this.dynamicForm.controls)) {\n        errors[name] = this.dynamicForm.controls[name].errors;\n      }\n      return errors;\n    }\n    return {};\n  }\n\n  /**\n   * Getter property for [controls] of dynamic [FormGroup].\n   */\n  get controls(): { [key: string]: AbstractControl } {\n    if (this.dynamicForm) {\n      return this.dynamicForm.controls;\n    }\n    return {};\n  }\n\n  constructor(\n    private _formBuilder: UntypedFormBuilder,\n    private _dynamicFormsService: TdDynamicFormsService,\n    private _changeDetectorRef: ChangeDetectorRef\n  ) {\n    this.dynamicForm = this._formBuilder.group({});\n  }\n\n  ngAfterContentInit(): void {\n    this._updateErrorTemplates();\n  }\n\n  ngOnDestroy(): void {\n    this._destroy$.next(0);\n    this._destroy$.complete();\n    this._destroyControl$.complete();\n  }\n\n  /**\n   * Refreshes the form and rerenders all validator/element modifications.\n   */\n  refresh(): void {\n    this._rerenderElements();\n    this._updateErrorTemplates();\n  }\n\n  /**\n   * Getter method for error template references\n   */\n  getErrorTemplateRef(name: string): TemplateRef<any> | undefined {\n    return this._templateMap.get(name);\n  }\n\n  /**\n   * Loads error templates and sets them in a map for faster access.\n   */\n  private _updateErrorTemplates(): void {\n    this._templateMap = new Map<string, TemplateRef<any>>();\n    for (const errorTemplate of this._errorTemplates.toArray()) {\n      this._templateMap.set(\n        errorTemplate.tdDynamicFormsError ?? '',\n        errorTemplate.templateRef\n      );\n    }\n  }\n\n  private _rerenderElements(): void {\n    this._clearRemovedElements();\n    this._renderedElements = [];\n    const duplicates: string[] = [];\n    this._elements.forEach((elem: ITdDynamicElementConfig) => {\n      this._dynamicFormsService.validateDynamicElementName(elem.name);\n      if (duplicates.indexOf(elem.name) > -1) {\n        throw new Error(`Dynamic element name: \"${elem.name}\" is duplicated`);\n      }\n      duplicates.push(elem.name);\n      const dynamicElement: AbstractControl | null = this.dynamicForm.get(\n        elem.name\n      );\n      if (!dynamicElement) {\n        this.dynamicForm.addControl(\n          elem.name,\n          this._dynamicFormsService.createFormControl(elem)\n        );\n        this._subscribeToControlStatusChanges(elem.name);\n      } else {\n        dynamicElement.setValue(elem.default);\n        dynamicElement.markAsPristine();\n        dynamicElement.markAsUntouched();\n        if (elem.disabled) {\n          dynamicElement.disable();\n        } else {\n          dynamicElement.enable();\n        }\n        dynamicElement.setValidators(\n          this._dynamicFormsService.createValidators(elem)\n        );\n      }\n      // copy objects so they are only changes when calling this method\n      this._renderedElements.push(Object.assign({}, elem));\n    });\n    // call a change detection since the whole form might change\n    this._changeDetectorRef.detectChanges();\n    timer(0)\n      .toPromise()\n      .then(() => {\n        // call a markForCheck so elements are rendered correctly in OnPush\n        this._changeDetectorRef.markForCheck();\n      });\n  }\n\n  private _clearRemovedElements(): void {\n    this._renderedElements = this._renderedElements.filter(\n      (renderedElement: ITdDynamicElementConfig) =>\n        !this._elements.some(\n          (element: ITdDynamicElementConfig) =>\n            element.name === renderedElement.name\n        )\n    );\n    // remove elements that were removed from the array\n    this._renderedElements.forEach((elem: ITdDynamicElementConfig) => {\n      this._destroyControl$.next(elem.name);\n      this.dynamicForm.removeControl(elem.name);\n    });\n  }\n\n  // Updates component when manually adding errors to controls\n  private _subscribeToControlStatusChanges(elementName: string): void {\n    const control: AbstractControl = this.controls[elementName];\n\n    const controlDestroyed$: Observable<any> = this._destroyControl$.pipe(\n      filter(\n        (destroyedElementName: string) => destroyedElementName === elementName\n      )\n    );\n\n    control.statusChanges\n      .pipe(takeUntil(this._destroy$), takeUntil(controlDestroyed$))\n      .subscribe(() => {\n        this._changeDetectorRef.markForCheck();\n      });\n  }\n}\n","<form [formGroup]=\"dynamicForm\" novalidate>\n  <div class=\"td-dynamic-form-wrapper\">\n    <ng-template let-element ngFor [ngForOf]=\"elements\">\n      <div\n        class=\"td-dynamic-element-wrapper\"\n        [style.max-width.%]=\"element.flex ? element.flex : 100\"\n        [style.flex]=\"'1 1 ' + (element.flex ? element.flex : 100) + '%'\"\n        [style.-ms-flex]=\"'1 1 ' + (element.flex ? element.flex : 100) + '%'\"\n        [style.-webkit-box-flex]=\"1\"\n      >\n        <td-dynamic-element\n          #dynamicElement\n          *ngIf=\"dynamicForm.controls[element.name]\"\n          [formControlName]=\"element.name\"\n          [dynamicControl]=\"dynamicForm.controls[element.name]\"\n          [id]=\"element.name\"\n          [name]=\"element.name\"\n          [label]=\"element.label || element.name\"\n          [hint]=\"element.hint\"\n          [type]=\"element.type\"\n          [required]=\"element.required ?? false\"\n          [min]=\"element.min\"\n          [max]=\"element.max\"\n          [minLength]=\"element.minLength\"\n          [maxLength]=\"element.maxLength\"\n          [selections]=\"element.selections\"\n          [multiple]=\"element.multiple\"\n          [customConfig]=\"element.customConfig\"\n          [errorMessageTemplate]=\"getErrorTemplateRef(element.name)\"\n          [placeholder]=\"element.placeholder\"\n        ></td-dynamic-element>\n      </div>\n    </ng-template>\n  </div>\n  <ng-content></ng-content>\n</form>\n"]}