@angular/forms 19.0.0-next.1 → 19.0.0-next.10

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 (47) hide show
  1. package/LICENSE +21 -0
  2. package/fesm2022/forms.mjs +186 -132
  3. package/fesm2022/forms.mjs.map +1 -1
  4. package/index.d.ts +6 -5
  5. package/package.json +4 -6
  6. package/esm2022/forms.mjs +0 -5
  7. package/esm2022/index.mjs +0 -13
  8. package/esm2022/public_api.mjs +0 -15
  9. package/esm2022/src/directives/abstract_control_directive.mjs +0 -279
  10. package/esm2022/src/directives/abstract_form_group_directive.mjs +0 -61
  11. package/esm2022/src/directives/checkbox_value_accessor.mjs +0 -58
  12. package/esm2022/src/directives/control_container.mjs +0 -32
  13. package/esm2022/src/directives/control_value_accessor.mjs +0 -92
  14. package/esm2022/src/directives/default_value_accessor.mjs +0 -124
  15. package/esm2022/src/directives/error_examples.mjs +0 -57
  16. package/esm2022/src/directives/form_interface.mjs +0 -9
  17. package/esm2022/src/directives/ng_control.mjs +0 -38
  18. package/esm2022/src/directives/ng_control_status.mjs +0 -139
  19. package/esm2022/src/directives/ng_form.mjs +0 -313
  20. package/esm2022/src/directives/ng_model.mjs +0 -331
  21. package/esm2022/src/directives/ng_model_group.mjs +0 -95
  22. package/esm2022/src/directives/ng_no_validate_directive.mjs +0 -39
  23. package/esm2022/src/directives/number_value_accessor.mjs +0 -70
  24. package/esm2022/src/directives/radio_control_value_accessor.mjs +0 -201
  25. package/esm2022/src/directives/range_value_accessor.mjs +0 -72
  26. package/esm2022/src/directives/reactive_directives/form_control_directive.mjs +0 -183
  27. package/esm2022/src/directives/reactive_directives/form_control_name.mjs +0 -216
  28. package/esm2022/src/directives/reactive_directives/form_group_directive.mjs +0 -359
  29. package/esm2022/src/directives/reactive_directives/form_group_name.mjs +0 -246
  30. package/esm2022/src/directives/reactive_errors.mjs +0 -119
  31. package/esm2022/src/directives/select_control_value_accessor.mjs +0 -220
  32. package/esm2022/src/directives/select_multiple_control_value_accessor.mjs +0 -252
  33. package/esm2022/src/directives/shared.mjs +0 -350
  34. package/esm2022/src/directives/template_driven_errors.mjs +0 -54
  35. package/esm2022/src/directives/validators.mjs +0 -505
  36. package/esm2022/src/directives.mjs +0 -128
  37. package/esm2022/src/errors.mjs +0 -9
  38. package/esm2022/src/form_builder.mjs +0 -258
  39. package/esm2022/src/form_providers.mjs +0 -105
  40. package/esm2022/src/forms.mjs +0 -49
  41. package/esm2022/src/model/abstract_model.mjs +0 -992
  42. package/esm2022/src/model/form_array.mjs +0 -461
  43. package/esm2022/src/model/form_control.mjs +0 -126
  44. package/esm2022/src/model/form_group.mjs +0 -490
  45. package/esm2022/src/util.mjs +0 -13
  46. package/esm2022/src/validators.mjs +0 -680
  47. package/esm2022/src/version.mjs +0 -18
@@ -1,313 +0,0 @@
1
- /**
2
- * @license
3
- * Copyright Google LLC All Rights Reserved.
4
- *
5
- * Use of this source code is governed by an MIT-style license that can be
6
- * found in the LICENSE file at https://angular.io/license
7
- */
8
- import { computed, Directive, EventEmitter, forwardRef, Inject, Input, Optional, Self, signal, untracked, } from '@angular/core';
9
- import { FormGroup } from '../model/form_group';
10
- import { composeAsyncValidators, composeValidators, NG_ASYNC_VALIDATORS, NG_VALIDATORS, } from '../validators';
11
- import { ControlContainer } from './control_container';
12
- import { CALL_SET_DISABLED_STATE, setUpControl, setUpFormContainer, syncPendingControls, } from './shared';
13
- import * as i0 from "@angular/core";
14
- const formDirectiveProvider = {
15
- provide: ControlContainer,
16
- useExisting: forwardRef(() => NgForm),
17
- };
18
- const resolvedPromise = (() => Promise.resolve())();
19
- /**
20
- * @description
21
- * Creates a top-level `FormGroup` instance and binds it to a form
22
- * to track aggregate form value and validation status.
23
- *
24
- * As soon as you import the `FormsModule`, this directive becomes active by default on
25
- * all `<form>` tags. You don't need to add a special selector.
26
- *
27
- * You optionally export the directive into a local template variable using `ngForm` as the key
28
- * (ex: `#myForm="ngForm"`). This is optional, but useful. Many properties from the underlying
29
- * `FormGroup` instance are duplicated on the directive itself, so a reference to it
30
- * gives you access to the aggregate value and validity status of the form, as well as
31
- * user interaction properties like `dirty` and `touched`.
32
- *
33
- * To register child controls with the form, use `NgModel` with a `name`
34
- * attribute. You may use `NgModelGroup` to create sub-groups within the form.
35
- *
36
- * If necessary, listen to the directive's `ngSubmit` event to be notified when the user has
37
- * triggered a form submission. The `ngSubmit` event emits the original form
38
- * submission event.
39
- *
40
- * In template driven forms, all `<form>` tags are automatically tagged as `NgForm`.
41
- * To import the `FormsModule` but skip its usage in some forms,
42
- * for example, to use native HTML5 validation, add the `ngNoForm` and the `<form>`
43
- * tags won't create an `NgForm` directive. In reactive forms, using `ngNoForm` is
44
- * unnecessary because the `<form>` tags are inert. In that case, you would
45
- * refrain from using the `formGroup` directive.
46
- *
47
- * @usageNotes
48
- *
49
- * ### Listening for form submission
50
- *
51
- * The following example shows how to capture the form values from the "ngSubmit" event.
52
- *
53
- * {@example forms/ts/simpleForm/simple_form_example.ts region='Component'}
54
- *
55
- * ### Setting the update options
56
- *
57
- * The following example shows you how to change the "updateOn" option from its default using
58
- * ngFormOptions.
59
- *
60
- * ```html
61
- * <form [ngFormOptions]="{updateOn: 'blur'}">
62
- * <input name="one" ngModel> <!-- this ngModel will update on blur -->
63
- * </form>
64
- * ```
65
- *
66
- * ### Native DOM validation UI
67
- *
68
- * In order to prevent the native DOM form validation UI from interfering with Angular's form
69
- * validation, Angular automatically adds the `novalidate` attribute on any `<form>` whenever
70
- * `FormModule` or `ReactiveFormModule` are imported into the application.
71
- * If you want to explicitly enable native DOM validation UI with Angular forms, you can add the
72
- * `ngNativeValidate` attribute to the `<form>` element:
73
- *
74
- * ```html
75
- * <form ngNativeValidate>
76
- * ...
77
- * </form>
78
- * ```
79
- *
80
- * @ngModule FormsModule
81
- * @publicApi
82
- */
83
- export class NgForm extends ControlContainer {
84
- /**
85
- * @description
86
- * Returns whether the form submission has been triggered.
87
- */
88
- get submitted() {
89
- return untracked(this.submittedReactive);
90
- }
91
- constructor(validators, asyncValidators, callSetDisabledState) {
92
- super();
93
- this.callSetDisabledState = callSetDisabledState;
94
- /** @internal */
95
- this._submitted = computed(() => this.submittedReactive());
96
- this.submittedReactive = signal(false);
97
- this._directives = new Set();
98
- /**
99
- * @description
100
- * Event emitter for the "ngSubmit" event
101
- */
102
- this.ngSubmit = new EventEmitter();
103
- this.form = new FormGroup({}, composeValidators(validators), composeAsyncValidators(asyncValidators));
104
- }
105
- /** @nodoc */
106
- ngAfterViewInit() {
107
- this._setUpdateStrategy();
108
- }
109
- /**
110
- * @description
111
- * The directive instance.
112
- */
113
- get formDirective() {
114
- return this;
115
- }
116
- /**
117
- * @description
118
- * The internal `FormGroup` instance.
119
- */
120
- get control() {
121
- return this.form;
122
- }
123
- /**
124
- * @description
125
- * Returns an array representing the path to this group. Because this directive
126
- * always lives at the top level of a form, it is always an empty array.
127
- */
128
- get path() {
129
- return [];
130
- }
131
- /**
132
- * @description
133
- * Returns a map of the controls in this group.
134
- */
135
- get controls() {
136
- return this.form.controls;
137
- }
138
- /**
139
- * @description
140
- * Method that sets up the control directive in this group, re-calculates its value
141
- * and validity, and adds the instance to the internal list of directives.
142
- *
143
- * @param dir The `NgModel` directive instance.
144
- */
145
- addControl(dir) {
146
- resolvedPromise.then(() => {
147
- const container = this._findContainer(dir.path);
148
- dir.control = (container.registerControl(dir.name, dir.control));
149
- setUpControl(dir.control, dir, this.callSetDisabledState);
150
- dir.control.updateValueAndValidity({ emitEvent: false });
151
- this._directives.add(dir);
152
- });
153
- }
154
- /**
155
- * @description
156
- * Retrieves the `FormControl` instance from the provided `NgModel` directive.
157
- *
158
- * @param dir The `NgModel` directive instance.
159
- */
160
- getControl(dir) {
161
- return this.form.get(dir.path);
162
- }
163
- /**
164
- * @description
165
- * Removes the `NgModel` instance from the internal list of directives
166
- *
167
- * @param dir The `NgModel` directive instance.
168
- */
169
- removeControl(dir) {
170
- resolvedPromise.then(() => {
171
- const container = this._findContainer(dir.path);
172
- if (container) {
173
- container.removeControl(dir.name);
174
- }
175
- this._directives.delete(dir);
176
- });
177
- }
178
- /**
179
- * @description
180
- * Adds a new `NgModelGroup` directive instance to the form.
181
- *
182
- * @param dir The `NgModelGroup` directive instance.
183
- */
184
- addFormGroup(dir) {
185
- resolvedPromise.then(() => {
186
- const container = this._findContainer(dir.path);
187
- const group = new FormGroup({});
188
- setUpFormContainer(group, dir);
189
- container.registerControl(dir.name, group);
190
- group.updateValueAndValidity({ emitEvent: false });
191
- });
192
- }
193
- /**
194
- * @description
195
- * Removes the `NgModelGroup` directive instance from the form.
196
- *
197
- * @param dir The `NgModelGroup` directive instance.
198
- */
199
- removeFormGroup(dir) {
200
- resolvedPromise.then(() => {
201
- const container = this._findContainer(dir.path);
202
- if (container) {
203
- container.removeControl(dir.name);
204
- }
205
- });
206
- }
207
- /**
208
- * @description
209
- * Retrieves the `FormGroup` for a provided `NgModelGroup` directive instance
210
- *
211
- * @param dir The `NgModelGroup` directive instance.
212
- */
213
- getFormGroup(dir) {
214
- return this.form.get(dir.path);
215
- }
216
- /**
217
- * Sets the new value for the provided `NgControl` directive.
218
- *
219
- * @param dir The `NgControl` directive instance.
220
- * @param value The new value for the directive's control.
221
- */
222
- updateModel(dir, value) {
223
- resolvedPromise.then(() => {
224
- const ctrl = this.form.get(dir.path);
225
- ctrl.setValue(value);
226
- });
227
- }
228
- /**
229
- * @description
230
- * Sets the value for this `FormGroup`.
231
- *
232
- * @param value The new value
233
- */
234
- setValue(value) {
235
- this.control.setValue(value);
236
- }
237
- /**
238
- * @description
239
- * Method called when the "submit" event is triggered on the form.
240
- * Triggers the `ngSubmit` emitter to emit the "submit" event as its payload.
241
- *
242
- * @param $event The "submit" event object
243
- */
244
- onSubmit($event) {
245
- this.submittedReactive.set(true);
246
- syncPendingControls(this.form, this._directives);
247
- this.ngSubmit.emit($event);
248
- // Forms with `method="dialog"` have some special behavior
249
- // that won't reload the page and that shouldn't be prevented.
250
- return $event?.target?.method === 'dialog';
251
- }
252
- /**
253
- * @description
254
- * Method called when the "reset" event is triggered on the form.
255
- */
256
- onReset() {
257
- this.resetForm();
258
- }
259
- /**
260
- * @description
261
- * Resets the form to an initial value and resets its submitted status.
262
- *
263
- * @param value The new value for the form.
264
- */
265
- resetForm(value = undefined) {
266
- this.form.reset(value);
267
- this.submittedReactive.set(false);
268
- }
269
- _setUpdateStrategy() {
270
- if (this.options && this.options.updateOn != null) {
271
- this.form._updateOn = this.options.updateOn;
272
- }
273
- }
274
- _findContainer(path) {
275
- path.pop();
276
- return path.length ? this.form.get(path) : this.form;
277
- }
278
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.0.0-next.1", ngImport: i0, type: NgForm, deps: [{ token: NG_VALIDATORS, optional: true, self: true }, { token: NG_ASYNC_VALIDATORS, optional: true, self: true }, { token: CALL_SET_DISABLED_STATE, optional: true }], target: i0.ɵɵFactoryTarget.Directive }); }
279
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "19.0.0-next.1", type: NgForm, selector: "form:not([ngNoForm]):not([formGroup]),ng-form,[ngForm]", inputs: { options: ["ngFormOptions", "options"] }, outputs: { ngSubmit: "ngSubmit" }, host: { listeners: { "submit": "onSubmit($event)", "reset": "onReset()" } }, providers: [formDirectiveProvider], exportAs: ["ngForm"], usesInheritance: true, ngImport: i0 }); }
280
- }
281
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.0.0-next.1", ngImport: i0, type: NgForm, decorators: [{
282
- type: Directive,
283
- args: [{
284
- selector: 'form:not([ngNoForm]):not([formGroup]),ng-form,[ngForm]',
285
- providers: [formDirectiveProvider],
286
- host: { '(submit)': 'onSubmit($event)', '(reset)': 'onReset()' },
287
- outputs: ['ngSubmit'],
288
- exportAs: 'ngForm',
289
- }]
290
- }], ctorParameters: () => [{ type: undefined, decorators: [{
291
- type: Optional
292
- }, {
293
- type: Self
294
- }, {
295
- type: Inject,
296
- args: [NG_VALIDATORS]
297
- }] }, { type: undefined, decorators: [{
298
- type: Optional
299
- }, {
300
- type: Self
301
- }, {
302
- type: Inject,
303
- args: [NG_ASYNC_VALIDATORS]
304
- }] }, { type: undefined, decorators: [{
305
- type: Optional
306
- }, {
307
- type: Inject,
308
- args: [CALL_SET_DISABLED_STATE]
309
- }] }], propDecorators: { options: [{
310
- type: Input,
311
- args: ['ngFormOptions']
312
- }] } });
313
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng_form.js","sourceRoot":"","sources":["../../../../../../../packages/forms/src/directives/ng_form.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {\n  AfterViewInit,\n  computed,\n  Directive,\n  EventEmitter,\n  forwardRef,\n  Inject,\n  Input,\n  Optional,\n  Provider,\n  Self,\n  signal,\n  untracked,\n  ɵWritable as Writable,\n} from '@angular/core';\n\nimport {AbstractControl, FormHooks} from '../model/abstract_model';\nimport {FormControl} from '../model/form_control';\nimport {FormGroup} from '../model/form_group';\nimport {\n  composeAsyncValidators,\n  composeValidators,\n  NG_ASYNC_VALIDATORS,\n  NG_VALIDATORS,\n} from '../validators';\n\nimport {ControlContainer} from './control_container';\nimport {Form} from './form_interface';\nimport {NgControl} from './ng_control';\nimport {NgModel} from './ng_model';\nimport {NgModelGroup} from './ng_model_group';\nimport {\n  CALL_SET_DISABLED_STATE,\n  SetDisabledStateOption,\n  setUpControl,\n  setUpFormContainer,\n  syncPendingControls,\n} from './shared';\nimport {AsyncValidator, AsyncValidatorFn, Validator, ValidatorFn} from './validators';\n\nconst formDirectiveProvider: Provider = {\n  provide: ControlContainer,\n  useExisting: forwardRef(() => NgForm),\n};\n\nconst resolvedPromise = (() => Promise.resolve())();\n\n/**\n * @description\n * Creates a top-level `FormGroup` instance and binds it to a form\n * to track aggregate form value and validation status.\n *\n * As soon as you import the `FormsModule`, this directive becomes active by default on\n * all `<form>` tags.  You don't need to add a special selector.\n *\n * You optionally export the directive into a local template variable using `ngForm` as the key\n * (ex: `#myForm=\"ngForm\"`). This is optional, but useful.  Many properties from the underlying\n * `FormGroup` instance are duplicated on the directive itself, so a reference to it\n * gives you access to the aggregate value and validity status of the form, as well as\n * user interaction properties like `dirty` and `touched`.\n *\n * To register child controls with the form, use `NgModel` with a `name`\n * attribute. You may use `NgModelGroup` to create sub-groups within the form.\n *\n * If necessary, listen to the directive's `ngSubmit` event to be notified when the user has\n * triggered a form submission. The `ngSubmit` event emits the original form\n * submission event.\n *\n * In template driven forms, all `<form>` tags are automatically tagged as `NgForm`.\n * To import the `FormsModule` but skip its usage in some forms,\n * for example, to use native HTML5 validation, add the `ngNoForm` and the `<form>`\n * tags won't create an `NgForm` directive. In reactive forms, using `ngNoForm` is\n * unnecessary because the `<form>` tags are inert. In that case, you would\n * refrain from using the `formGroup` directive.\n *\n * @usageNotes\n *\n * ### Listening for form submission\n *\n * The following example shows how to capture the form values from the \"ngSubmit\" event.\n *\n * {@example forms/ts/simpleForm/simple_form_example.ts region='Component'}\n *\n * ### Setting the update options\n *\n * The following example shows you how to change the \"updateOn\" option from its default using\n * ngFormOptions.\n *\n * ```html\n * <form [ngFormOptions]=\"{updateOn: 'blur'}\">\n *    <input name=\"one\" ngModel>  <!-- this ngModel will update on blur -->\n * </form>\n * ```\n *\n * ### Native DOM validation UI\n *\n * In order to prevent the native DOM form validation UI from interfering with Angular's form\n * validation, Angular automatically adds the `novalidate` attribute on any `<form>` whenever\n * `FormModule` or `ReactiveFormModule` are imported into the application.\n * If you want to explicitly enable native DOM validation UI with Angular forms, you can add the\n * `ngNativeValidate` attribute to the `<form>` element:\n *\n * ```html\n * <form ngNativeValidate>\n *   ...\n * </form>\n * ```\n *\n * @ngModule FormsModule\n * @publicApi\n */\n@Directive({\n  selector: 'form:not([ngNoForm]):not([formGroup]),ng-form,[ngForm]',\n  providers: [formDirectiveProvider],\n  host: {'(submit)': 'onSubmit($event)', '(reset)': 'onReset()'},\n  outputs: ['ngSubmit'],\n  exportAs: 'ngForm',\n})\nexport class NgForm extends ControlContainer implements Form, AfterViewInit {\n  /**\n   * @description\n   * Returns whether the form submission has been triggered.\n   */\n  get submitted(): boolean {\n    return untracked(this.submittedReactive);\n  }\n  /** @internal */\n  readonly _submitted = computed(() => this.submittedReactive());\n  private readonly submittedReactive = signal(false);\n\n  private _directives = new Set<NgModel>();\n\n  /**\n   * @description\n   * The `FormGroup` instance created for this form.\n   */\n  form: FormGroup;\n\n  /**\n   * @description\n   * Event emitter for the \"ngSubmit\" event\n   */\n  ngSubmit = new EventEmitter();\n\n  /**\n   * @description\n   * Tracks options for the `NgForm` instance.\n   *\n   * **updateOn**: Sets the default `updateOn` value for all child `NgModels` below it\n   * unless explicitly set by a child `NgModel` using `ngModelOptions`). Defaults to 'change'.\n   * Possible values: `'change'` | `'blur'` | `'submit'`.\n   *\n   */\n  // TODO(issue/24571): remove '!'.\n  @Input('ngFormOptions') options!: {updateOn?: FormHooks};\n\n  constructor(\n    @Optional() @Self() @Inject(NG_VALIDATORS) validators: (Validator | ValidatorFn)[],\n    @Optional()\n    @Self()\n    @Inject(NG_ASYNC_VALIDATORS)\n    asyncValidators: (AsyncValidator | AsyncValidatorFn)[],\n    @Optional()\n    @Inject(CALL_SET_DISABLED_STATE)\n    private callSetDisabledState?: SetDisabledStateOption,\n  ) {\n    super();\n    this.form = new FormGroup(\n      {},\n      composeValidators(validators),\n      composeAsyncValidators(asyncValidators),\n    );\n  }\n\n  /** @nodoc */\n  ngAfterViewInit() {\n    this._setUpdateStrategy();\n  }\n\n  /**\n   * @description\n   * The directive instance.\n   */\n  override get formDirective(): Form {\n    return this;\n  }\n\n  /**\n   * @description\n   * The internal `FormGroup` instance.\n   */\n  override get control(): FormGroup {\n    return this.form;\n  }\n\n  /**\n   * @description\n   * Returns an array representing the path to this group. Because this directive\n   * always lives at the top level of a form, it is always an empty array.\n   */\n  override get path(): string[] {\n    return [];\n  }\n\n  /**\n   * @description\n   * Returns a map of the controls in this group.\n   */\n  get controls(): {[key: string]: AbstractControl} {\n    return this.form.controls;\n  }\n\n  /**\n   * @description\n   * Method that sets up the control directive in this group, re-calculates its value\n   * and validity, and adds the instance to the internal list of directives.\n   *\n   * @param dir The `NgModel` directive instance.\n   */\n  addControl(dir: NgModel): void {\n    resolvedPromise.then(() => {\n      const container = this._findContainer(dir.path);\n      (dir as Writable<NgModel>).control = <FormControl>(\n        container.registerControl(dir.name, dir.control)\n      );\n      setUpControl(dir.control, dir, this.callSetDisabledState);\n      dir.control.updateValueAndValidity({emitEvent: false});\n      this._directives.add(dir);\n    });\n  }\n\n  /**\n   * @description\n   * Retrieves the `FormControl` instance from the provided `NgModel` directive.\n   *\n   * @param dir The `NgModel` directive instance.\n   */\n  getControl(dir: NgModel): FormControl {\n    return <FormControl>this.form.get(dir.path);\n  }\n\n  /**\n   * @description\n   * Removes the `NgModel` instance from the internal list of directives\n   *\n   * @param dir The `NgModel` directive instance.\n   */\n  removeControl(dir: NgModel): void {\n    resolvedPromise.then(() => {\n      const container = this._findContainer(dir.path);\n      if (container) {\n        container.removeControl(dir.name);\n      }\n      this._directives.delete(dir);\n    });\n  }\n\n  /**\n   * @description\n   * Adds a new `NgModelGroup` directive instance to the form.\n   *\n   * @param dir The `NgModelGroup` directive instance.\n   */\n  addFormGroup(dir: NgModelGroup): void {\n    resolvedPromise.then(() => {\n      const container = this._findContainer(dir.path);\n      const group = new FormGroup({});\n      setUpFormContainer(group, dir);\n      container.registerControl(dir.name, group);\n      group.updateValueAndValidity({emitEvent: false});\n    });\n  }\n\n  /**\n   * @description\n   * Removes the `NgModelGroup` directive instance from the form.\n   *\n   * @param dir The `NgModelGroup` directive instance.\n   */\n  removeFormGroup(dir: NgModelGroup): void {\n    resolvedPromise.then(() => {\n      const container = this._findContainer(dir.path);\n      if (container) {\n        container.removeControl(dir.name);\n      }\n    });\n  }\n\n  /**\n   * @description\n   * Retrieves the `FormGroup` for a provided `NgModelGroup` directive instance\n   *\n   * @param dir The `NgModelGroup` directive instance.\n   */\n  getFormGroup(dir: NgModelGroup): FormGroup {\n    return <FormGroup>this.form.get(dir.path);\n  }\n\n  /**\n   * Sets the new value for the provided `NgControl` directive.\n   *\n   * @param dir The `NgControl` directive instance.\n   * @param value The new value for the directive's control.\n   */\n  updateModel(dir: NgControl, value: any): void {\n    resolvedPromise.then(() => {\n      const ctrl = <FormControl>this.form.get(dir.path!);\n      ctrl.setValue(value);\n    });\n  }\n\n  /**\n   * @description\n   * Sets the value for this `FormGroup`.\n   *\n   * @param value The new value\n   */\n  setValue(value: {[key: string]: any}): void {\n    this.control.setValue(value);\n  }\n\n  /**\n   * @description\n   * Method called when the \"submit\" event is triggered on the form.\n   * Triggers the `ngSubmit` emitter to emit the \"submit\" event as its payload.\n   *\n   * @param $event The \"submit\" event object\n   */\n  onSubmit($event: Event): boolean {\n    this.submittedReactive.set(true);\n    syncPendingControls(this.form, this._directives);\n    this.ngSubmit.emit($event);\n    // Forms with `method=\"dialog\"` have some special behavior\n    // that won't reload the page and that shouldn't be prevented.\n    return ($event?.target as HTMLFormElement | null)?.method === 'dialog';\n  }\n\n  /**\n   * @description\n   * Method called when the \"reset\" event is triggered on the form.\n   */\n  onReset(): void {\n    this.resetForm();\n  }\n\n  /**\n   * @description\n   * Resets the form to an initial value and resets its submitted status.\n   *\n   * @param value The new value for the form.\n   */\n  resetForm(value: any = undefined): void {\n    this.form.reset(value);\n    this.submittedReactive.set(false);\n  }\n\n  private _setUpdateStrategy() {\n    if (this.options && this.options.updateOn != null) {\n      this.form._updateOn = this.options.updateOn;\n    }\n  }\n\n  private _findContainer(path: string[]): FormGroup {\n    path.pop();\n    return path.length ? <FormGroup>this.form.get(path) : this.form;\n  }\n}\n"]}