@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,350 +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 { InjectionToken, ɵRuntimeError as RuntimeError } from '@angular/core';
9
- import { getControlAsyncValidators, getControlValidators, mergeValidators } from '../validators';
10
- import { BuiltInControlValueAccessor } from './control_value_accessor';
11
- import { DefaultValueAccessor } from './default_value_accessor';
12
- import { ngModelWarning } from './reactive_errors';
13
- /**
14
- * Token to provide to allow SetDisabledState to always be called when a CVA is added, regardless of
15
- * whether the control is disabled or enabled.
16
- *
17
- * @see {@link FormsModule#withconfig}
18
- */
19
- export const CALL_SET_DISABLED_STATE = new InjectionToken('CallSetDisabledState', {
20
- providedIn: 'root',
21
- factory: () => setDisabledStateDefault,
22
- });
23
- /**
24
- * Whether to use the fixed setDisabledState behavior by default.
25
- */
26
- export const setDisabledStateDefault = 'always';
27
- export function controlPath(name, parent) {
28
- return [...parent.path, name];
29
- }
30
- /**
31
- * Links a Form control and a Form directive by setting up callbacks (such as `onChange`) on both
32
- * instances. This function is typically invoked when form directive is being initialized.
33
- *
34
- * @param control Form control instance that should be linked.
35
- * @param dir Directive that should be linked with a given control.
36
- */
37
- export function setUpControl(control, dir, callSetDisabledState = setDisabledStateDefault) {
38
- if (typeof ngDevMode === 'undefined' || ngDevMode) {
39
- if (!control)
40
- _throwError(dir, 'Cannot find control with');
41
- if (!dir.valueAccessor)
42
- _throwMissingValueAccessorError(dir);
43
- }
44
- setUpValidators(control, dir);
45
- dir.valueAccessor.writeValue(control.value);
46
- // The legacy behavior only calls the CVA's `setDisabledState` if the control is disabled.
47
- // If the `callSetDisabledState` option is set to `always`, then this bug is fixed and
48
- // the method is always called.
49
- if (control.disabled || callSetDisabledState === 'always') {
50
- dir.valueAccessor.setDisabledState?.(control.disabled);
51
- }
52
- setUpViewChangePipeline(control, dir);
53
- setUpModelChangePipeline(control, dir);
54
- setUpBlurPipeline(control, dir);
55
- setUpDisabledChangeHandler(control, dir);
56
- }
57
- /**
58
- * Reverts configuration performed by the `setUpControl` control function.
59
- * Effectively disconnects form control with a given form directive.
60
- * This function is typically invoked when corresponding form directive is being destroyed.
61
- *
62
- * @param control Form control which should be cleaned up.
63
- * @param dir Directive that should be disconnected from a given control.
64
- * @param validateControlPresenceOnChange Flag that indicates whether onChange handler should
65
- * contain asserts to verify that it's not called once directive is destroyed. We need this flag
66
- * to avoid potentially breaking changes caused by better control cleanup introduced in #39235.
67
- */
68
- export function cleanUpControl(control, dir, validateControlPresenceOnChange = true) {
69
- const noop = () => {
70
- if (validateControlPresenceOnChange && (typeof ngDevMode === 'undefined' || ngDevMode)) {
71
- _noControlError(dir);
72
- }
73
- };
74
- // The `valueAccessor` field is typically defined on FromControl and FormControlName directive
75
- // instances and there is a logic in `selectValueAccessor` function that throws if it's not the
76
- // case. We still check the presence of `valueAccessor` before invoking its methods to make sure
77
- // that cleanup works correctly if app code or tests are setup to ignore the error thrown from
78
- // `selectValueAccessor`. See https://github.com/angular/angular/issues/40521.
79
- if (dir.valueAccessor) {
80
- dir.valueAccessor.registerOnChange(noop);
81
- dir.valueAccessor.registerOnTouched(noop);
82
- }
83
- cleanUpValidators(control, dir);
84
- if (control) {
85
- dir._invokeOnDestroyCallbacks();
86
- control._registerOnCollectionChange(() => { });
87
- }
88
- }
89
- function registerOnValidatorChange(validators, onChange) {
90
- validators.forEach((validator) => {
91
- if (validator.registerOnValidatorChange)
92
- validator.registerOnValidatorChange(onChange);
93
- });
94
- }
95
- /**
96
- * Sets up disabled change handler function on a given form control if ControlValueAccessor
97
- * associated with a given directive instance supports the `setDisabledState` call.
98
- *
99
- * @param control Form control where disabled change handler should be setup.
100
- * @param dir Corresponding directive instance associated with this control.
101
- */
102
- export function setUpDisabledChangeHandler(control, dir) {
103
- if (dir.valueAccessor.setDisabledState) {
104
- const onDisabledChange = (isDisabled) => {
105
- dir.valueAccessor.setDisabledState(isDisabled);
106
- };
107
- control.registerOnDisabledChange(onDisabledChange);
108
- // Register a callback function to cleanup disabled change handler
109
- // from a control instance when a directive is destroyed.
110
- dir._registerOnDestroy(() => {
111
- control._unregisterOnDisabledChange(onDisabledChange);
112
- });
113
- }
114
- }
115
- /**
116
- * Sets up sync and async directive validators on provided form control.
117
- * This function merges validators from the directive into the validators of the control.
118
- *
119
- * @param control Form control where directive validators should be setup.
120
- * @param dir Directive instance that contains validators to be setup.
121
- */
122
- export function setUpValidators(control, dir) {
123
- const validators = getControlValidators(control);
124
- if (dir.validator !== null) {
125
- control.setValidators(mergeValidators(validators, dir.validator));
126
- }
127
- else if (typeof validators === 'function') {
128
- // If sync validators are represented by a single validator function, we force the
129
- // `Validators.compose` call to happen by executing the `setValidators` function with
130
- // an array that contains that function. We need this to avoid possible discrepancies in
131
- // validators behavior, so sync validators are always processed by the `Validators.compose`.
132
- // Note: we should consider moving this logic inside the `setValidators` function itself, so we
133
- // have consistent behavior on AbstractControl API level. The same applies to the async
134
- // validators logic below.
135
- control.setValidators([validators]);
136
- }
137
- const asyncValidators = getControlAsyncValidators(control);
138
- if (dir.asyncValidator !== null) {
139
- control.setAsyncValidators(mergeValidators(asyncValidators, dir.asyncValidator));
140
- }
141
- else if (typeof asyncValidators === 'function') {
142
- control.setAsyncValidators([asyncValidators]);
143
- }
144
- // Re-run validation when validator binding changes, e.g. minlength=3 -> minlength=4
145
- const onValidatorChange = () => control.updateValueAndValidity();
146
- registerOnValidatorChange(dir._rawValidators, onValidatorChange);
147
- registerOnValidatorChange(dir._rawAsyncValidators, onValidatorChange);
148
- }
149
- /**
150
- * Cleans up sync and async directive validators on provided form control.
151
- * This function reverts the setup performed by the `setUpValidators` function, i.e.
152
- * removes directive-specific validators from a given control instance.
153
- *
154
- * @param control Form control from where directive validators should be removed.
155
- * @param dir Directive instance that contains validators to be removed.
156
- * @returns true if a control was updated as a result of this action.
157
- */
158
- export function cleanUpValidators(control, dir) {
159
- let isControlUpdated = false;
160
- if (control !== null) {
161
- if (dir.validator !== null) {
162
- const validators = getControlValidators(control);
163
- if (Array.isArray(validators) && validators.length > 0) {
164
- // Filter out directive validator function.
165
- const updatedValidators = validators.filter((validator) => validator !== dir.validator);
166
- if (updatedValidators.length !== validators.length) {
167
- isControlUpdated = true;
168
- control.setValidators(updatedValidators);
169
- }
170
- }
171
- }
172
- if (dir.asyncValidator !== null) {
173
- const asyncValidators = getControlAsyncValidators(control);
174
- if (Array.isArray(asyncValidators) && asyncValidators.length > 0) {
175
- // Filter out directive async validator function.
176
- const updatedAsyncValidators = asyncValidators.filter((asyncValidator) => asyncValidator !== dir.asyncValidator);
177
- if (updatedAsyncValidators.length !== asyncValidators.length) {
178
- isControlUpdated = true;
179
- control.setAsyncValidators(updatedAsyncValidators);
180
- }
181
- }
182
- }
183
- }
184
- // Clear onValidatorChange callbacks by providing a noop function.
185
- const noop = () => { };
186
- registerOnValidatorChange(dir._rawValidators, noop);
187
- registerOnValidatorChange(dir._rawAsyncValidators, noop);
188
- return isControlUpdated;
189
- }
190
- function setUpViewChangePipeline(control, dir) {
191
- dir.valueAccessor.registerOnChange((newValue) => {
192
- control._pendingValue = newValue;
193
- control._pendingChange = true;
194
- control._pendingDirty = true;
195
- if (control.updateOn === 'change')
196
- updateControl(control, dir);
197
- });
198
- }
199
- function setUpBlurPipeline(control, dir) {
200
- dir.valueAccessor.registerOnTouched(() => {
201
- control._pendingTouched = true;
202
- if (control.updateOn === 'blur' && control._pendingChange)
203
- updateControl(control, dir);
204
- if (control.updateOn !== 'submit')
205
- control.markAsTouched();
206
- });
207
- }
208
- function updateControl(control, dir) {
209
- if (control._pendingDirty)
210
- control.markAsDirty();
211
- control.setValue(control._pendingValue, { emitModelToViewChange: false });
212
- dir.viewToModelUpdate(control._pendingValue);
213
- control._pendingChange = false;
214
- }
215
- function setUpModelChangePipeline(control, dir) {
216
- const onChange = (newValue, emitModelEvent) => {
217
- // control -> view
218
- dir.valueAccessor.writeValue(newValue);
219
- // control -> ngModel
220
- if (emitModelEvent)
221
- dir.viewToModelUpdate(newValue);
222
- };
223
- control.registerOnChange(onChange);
224
- // Register a callback function to cleanup onChange handler
225
- // from a control instance when a directive is destroyed.
226
- dir._registerOnDestroy(() => {
227
- control._unregisterOnChange(onChange);
228
- });
229
- }
230
- /**
231
- * Links a FormGroup or FormArray instance and corresponding Form directive by setting up validators
232
- * present in the view.
233
- *
234
- * @param control FormGroup or FormArray instance that should be linked.
235
- * @param dir Directive that provides view validators.
236
- */
237
- export function setUpFormContainer(control, dir) {
238
- if (control == null && (typeof ngDevMode === 'undefined' || ngDevMode))
239
- _throwError(dir, 'Cannot find control with');
240
- setUpValidators(control, dir);
241
- }
242
- /**
243
- * Reverts the setup performed by the `setUpFormContainer` function.
244
- *
245
- * @param control FormGroup or FormArray instance that should be cleaned up.
246
- * @param dir Directive that provided view validators.
247
- * @returns true if a control was updated as a result of this action.
248
- */
249
- export function cleanUpFormContainer(control, dir) {
250
- return cleanUpValidators(control, dir);
251
- }
252
- function _noControlError(dir) {
253
- return _throwError(dir, 'There is no FormControl instance attached to form control element with');
254
- }
255
- function _throwError(dir, message) {
256
- const messageEnd = _describeControlLocation(dir);
257
- throw new Error(`${message} ${messageEnd}`);
258
- }
259
- function _describeControlLocation(dir) {
260
- const path = dir.path;
261
- if (path && path.length > 1)
262
- return `path: '${path.join(' -> ')}'`;
263
- if (path?.[0])
264
- return `name: '${path}'`;
265
- return 'unspecified name attribute';
266
- }
267
- function _throwMissingValueAccessorError(dir) {
268
- const loc = _describeControlLocation(dir);
269
- throw new RuntimeError(-1203 /* RuntimeErrorCode.NG_MISSING_VALUE_ACCESSOR */, `No value accessor for form control ${loc}.`);
270
- }
271
- function _throwInvalidValueAccessorError(dir) {
272
- const loc = _describeControlLocation(dir);
273
- throw new RuntimeError(1200 /* RuntimeErrorCode.NG_VALUE_ACCESSOR_NOT_PROVIDED */, `Value accessor was not provided as an array for form control with ${loc}. ` +
274
- `Check that the \`NG_VALUE_ACCESSOR\` token is configured as a \`multi: true\` provider.`);
275
- }
276
- export function isPropertyUpdated(changes, viewModel) {
277
- if (!changes.hasOwnProperty('model'))
278
- return false;
279
- const change = changes['model'];
280
- if (change.isFirstChange())
281
- return true;
282
- return !Object.is(viewModel, change.currentValue);
283
- }
284
- export function isBuiltInAccessor(valueAccessor) {
285
- // Check if a given value accessor is an instance of a class that directly extends
286
- // `BuiltInControlValueAccessor` one.
287
- return Object.getPrototypeOf(valueAccessor.constructor) === BuiltInControlValueAccessor;
288
- }
289
- export function syncPendingControls(form, directives) {
290
- form._syncPendingControls();
291
- directives.forEach((dir) => {
292
- const control = dir.control;
293
- if (control.updateOn === 'submit' && control._pendingChange) {
294
- dir.viewToModelUpdate(control._pendingValue);
295
- control._pendingChange = false;
296
- }
297
- });
298
- }
299
- // TODO: vsavkin remove it once https://github.com/angular/angular/issues/3011 is implemented
300
- export function selectValueAccessor(dir, valueAccessors) {
301
- if (!valueAccessors)
302
- return null;
303
- if (!Array.isArray(valueAccessors) && (typeof ngDevMode === 'undefined' || ngDevMode))
304
- _throwInvalidValueAccessorError(dir);
305
- let defaultAccessor = undefined;
306
- let builtinAccessor = undefined;
307
- let customAccessor = undefined;
308
- valueAccessors.forEach((v) => {
309
- if (v.constructor === DefaultValueAccessor) {
310
- defaultAccessor = v;
311
- }
312
- else if (isBuiltInAccessor(v)) {
313
- if (builtinAccessor && (typeof ngDevMode === 'undefined' || ngDevMode))
314
- _throwError(dir, 'More than one built-in value accessor matches form control with');
315
- builtinAccessor = v;
316
- }
317
- else {
318
- if (customAccessor && (typeof ngDevMode === 'undefined' || ngDevMode))
319
- _throwError(dir, 'More than one custom value accessor matches form control with');
320
- customAccessor = v;
321
- }
322
- });
323
- if (customAccessor)
324
- return customAccessor;
325
- if (builtinAccessor)
326
- return builtinAccessor;
327
- if (defaultAccessor)
328
- return defaultAccessor;
329
- if (typeof ngDevMode === 'undefined' || ngDevMode) {
330
- _throwError(dir, 'No valid value accessor for form control with');
331
- }
332
- return null;
333
- }
334
- export function removeListItem(list, el) {
335
- const index = list.indexOf(el);
336
- if (index > -1)
337
- list.splice(index, 1);
338
- }
339
- // TODO(kara): remove after deprecation period
340
- export function _ngModelWarning(name, type, instance, warningConfig) {
341
- if (warningConfig === 'never')
342
- return;
343
- if (((warningConfig === null || warningConfig === 'once') && !type._ngModelWarningSentOnce) ||
344
- (warningConfig === 'always' && !instance._ngModelWarningSent)) {
345
- console.warn(ngModelWarning(name));
346
- type._ngModelWarningSentOnce = true;
347
- instance._ngModelWarningSent = true;
348
- }
349
- }
350
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"shared.js","sourceRoot":"","sources":["../../../../../../../packages/forms/src/directives/shared.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 {Inject, InjectionToken, ɵRuntimeError as RuntimeError} from '@angular/core';\n\nimport {RuntimeErrorCode} from '../errors';\nimport {AbstractControl} from '../model/abstract_model';\nimport {FormArray} from '../model/form_array';\nimport {FormControl} from '../model/form_control';\nimport {FormGroup} from '../model/form_group';\nimport {getControlAsyncValidators, getControlValidators, mergeValidators} from '../validators';\n\nimport {AbstractControlDirective} from './abstract_control_directive';\nimport {AbstractFormGroupDirective} from './abstract_form_group_directive';\nimport {ControlContainer} from './control_container';\nimport {BuiltInControlValueAccessor, ControlValueAccessor} from './control_value_accessor';\nimport {DefaultValueAccessor} from './default_value_accessor';\nimport {NgControl} from './ng_control';\nimport {FormArrayName} from './reactive_directives/form_group_name';\nimport {ngModelWarning} from './reactive_errors';\nimport {AsyncValidatorFn, Validator, ValidatorFn} from './validators';\n\n/**\n * Token to provide to allow SetDisabledState to always be called when a CVA is added, regardless of\n * whether the control is disabled or enabled.\n *\n * @see {@link FormsModule#withconfig}\n */\nexport const CALL_SET_DISABLED_STATE = new InjectionToken('CallSetDisabledState', {\n  providedIn: 'root',\n  factory: () => setDisabledStateDefault,\n});\n\n/**\n * The type for CALL_SET_DISABLED_STATE. If `always`, then ControlValueAccessor will always call\n * `setDisabledState` when attached, which is the most correct behavior. Otherwise, it will only be\n * called when disabled, which is the legacy behavior for compatibility.\n *\n * @publicApi\n * @see {@link FormsModule#withconfig}\n */\nexport type SetDisabledStateOption = 'whenDisabledForLegacyCode' | 'always';\n\n/**\n * Whether to use the fixed setDisabledState behavior by default.\n */\nexport const setDisabledStateDefault: SetDisabledStateOption = 'always';\n\nexport function controlPath(name: string | null, parent: ControlContainer): string[] {\n  return [...parent.path!, name!];\n}\n\n/**\n * Links a Form control and a Form directive by setting up callbacks (such as `onChange`) on both\n * instances. This function is typically invoked when form directive is being initialized.\n *\n * @param control Form control instance that should be linked.\n * @param dir Directive that should be linked with a given control.\n */\nexport function setUpControl(\n  control: FormControl,\n  dir: NgControl,\n  callSetDisabledState: SetDisabledStateOption = setDisabledStateDefault,\n): void {\n  if (typeof ngDevMode === 'undefined' || ngDevMode) {\n    if (!control) _throwError(dir, 'Cannot find control with');\n    if (!dir.valueAccessor) _throwMissingValueAccessorError(dir);\n  }\n\n  setUpValidators(control, dir);\n\n  dir.valueAccessor!.writeValue(control.value);\n\n  // The legacy behavior only calls the CVA's `setDisabledState` if the control is disabled.\n  // If the `callSetDisabledState` option is set to `always`, then this bug is fixed and\n  // the method is always called.\n  if (control.disabled || callSetDisabledState === 'always') {\n    dir.valueAccessor!.setDisabledState?.(control.disabled);\n  }\n\n  setUpViewChangePipeline(control, dir);\n  setUpModelChangePipeline(control, dir);\n\n  setUpBlurPipeline(control, dir);\n\n  setUpDisabledChangeHandler(control, dir);\n}\n\n/**\n * Reverts configuration performed by the `setUpControl` control function.\n * Effectively disconnects form control with a given form directive.\n * This function is typically invoked when corresponding form directive is being destroyed.\n *\n * @param control Form control which should be cleaned up.\n * @param dir Directive that should be disconnected from a given control.\n * @param validateControlPresenceOnChange Flag that indicates whether onChange handler should\n *     contain asserts to verify that it's not called once directive is destroyed. We need this flag\n *     to avoid potentially breaking changes caused by better control cleanup introduced in #39235.\n */\nexport function cleanUpControl(\n  control: FormControl | null,\n  dir: NgControl,\n  validateControlPresenceOnChange: boolean = true,\n): void {\n  const noop = () => {\n    if (validateControlPresenceOnChange && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      _noControlError(dir);\n    }\n  };\n\n  // The `valueAccessor` field is typically defined on FromControl and FormControlName directive\n  // instances and there is a logic in `selectValueAccessor` function that throws if it's not the\n  // case. We still check the presence of `valueAccessor` before invoking its methods to make sure\n  // that cleanup works correctly if app code or tests are setup to ignore the error thrown from\n  // `selectValueAccessor`. See https://github.com/angular/angular/issues/40521.\n  if (dir.valueAccessor) {\n    dir.valueAccessor.registerOnChange(noop);\n    dir.valueAccessor.registerOnTouched(noop);\n  }\n\n  cleanUpValidators(control, dir);\n\n  if (control) {\n    dir._invokeOnDestroyCallbacks();\n    control._registerOnCollectionChange(() => {});\n  }\n}\n\nfunction registerOnValidatorChange<V>(validators: (V | Validator)[], onChange: () => void): void {\n  validators.forEach((validator: V | Validator) => {\n    if ((<Validator>validator).registerOnValidatorChange)\n      (<Validator>validator).registerOnValidatorChange!(onChange);\n  });\n}\n\n/**\n * Sets up disabled change handler function on a given form control if ControlValueAccessor\n * associated with a given directive instance supports the `setDisabledState` call.\n *\n * @param control Form control where disabled change handler should be setup.\n * @param dir Corresponding directive instance associated with this control.\n */\nexport function setUpDisabledChangeHandler(control: FormControl, dir: NgControl): void {\n  if (dir.valueAccessor!.setDisabledState) {\n    const onDisabledChange = (isDisabled: boolean) => {\n      dir.valueAccessor!.setDisabledState!(isDisabled);\n    };\n    control.registerOnDisabledChange(onDisabledChange);\n\n    // Register a callback function to cleanup disabled change handler\n    // from a control instance when a directive is destroyed.\n    dir._registerOnDestroy(() => {\n      control._unregisterOnDisabledChange(onDisabledChange);\n    });\n  }\n}\n\n/**\n * Sets up sync and async directive validators on provided form control.\n * This function merges validators from the directive into the validators of the control.\n *\n * @param control Form control where directive validators should be setup.\n * @param dir Directive instance that contains validators to be setup.\n */\nexport function setUpValidators(control: AbstractControl, dir: AbstractControlDirective): void {\n  const validators = getControlValidators(control);\n  if (dir.validator !== null) {\n    control.setValidators(mergeValidators<ValidatorFn>(validators, dir.validator));\n  } else if (typeof validators === 'function') {\n    // If sync validators are represented by a single validator function, we force the\n    // `Validators.compose` call to happen by executing the `setValidators` function with\n    // an array that contains that function. We need this to avoid possible discrepancies in\n    // validators behavior, so sync validators are always processed by the `Validators.compose`.\n    // Note: we should consider moving this logic inside the `setValidators` function itself, so we\n    // have consistent behavior on AbstractControl API level. The same applies to the async\n    // validators logic below.\n    control.setValidators([validators]);\n  }\n\n  const asyncValidators = getControlAsyncValidators(control);\n  if (dir.asyncValidator !== null) {\n    control.setAsyncValidators(\n      mergeValidators<AsyncValidatorFn>(asyncValidators, dir.asyncValidator),\n    );\n  } else if (typeof asyncValidators === 'function') {\n    control.setAsyncValidators([asyncValidators]);\n  }\n\n  // Re-run validation when validator binding changes, e.g. minlength=3 -> minlength=4\n  const onValidatorChange = () => control.updateValueAndValidity();\n  registerOnValidatorChange<ValidatorFn>(dir._rawValidators, onValidatorChange);\n  registerOnValidatorChange<AsyncValidatorFn>(dir._rawAsyncValidators, onValidatorChange);\n}\n\n/**\n * Cleans up sync and async directive validators on provided form control.\n * This function reverts the setup performed by the `setUpValidators` function, i.e.\n * removes directive-specific validators from a given control instance.\n *\n * @param control Form control from where directive validators should be removed.\n * @param dir Directive instance that contains validators to be removed.\n * @returns true if a control was updated as a result of this action.\n */\nexport function cleanUpValidators(\n  control: AbstractControl | null,\n  dir: AbstractControlDirective,\n): boolean {\n  let isControlUpdated = false;\n  if (control !== null) {\n    if (dir.validator !== null) {\n      const validators = getControlValidators(control);\n      if (Array.isArray(validators) && validators.length > 0) {\n        // Filter out directive validator function.\n        const updatedValidators = validators.filter((validator) => validator !== dir.validator);\n        if (updatedValidators.length !== validators.length) {\n          isControlUpdated = true;\n          control.setValidators(updatedValidators);\n        }\n      }\n    }\n\n    if (dir.asyncValidator !== null) {\n      const asyncValidators = getControlAsyncValidators(control);\n      if (Array.isArray(asyncValidators) && asyncValidators.length > 0) {\n        // Filter out directive async validator function.\n        const updatedAsyncValidators = asyncValidators.filter(\n          (asyncValidator) => asyncValidator !== dir.asyncValidator,\n        );\n        if (updatedAsyncValidators.length !== asyncValidators.length) {\n          isControlUpdated = true;\n          control.setAsyncValidators(updatedAsyncValidators);\n        }\n      }\n    }\n  }\n\n  // Clear onValidatorChange callbacks by providing a noop function.\n  const noop = () => {};\n  registerOnValidatorChange<ValidatorFn>(dir._rawValidators, noop);\n  registerOnValidatorChange<AsyncValidatorFn>(dir._rawAsyncValidators, noop);\n\n  return isControlUpdated;\n}\n\nfunction setUpViewChangePipeline(control: FormControl, dir: NgControl): void {\n  dir.valueAccessor!.registerOnChange((newValue: any) => {\n    control._pendingValue = newValue;\n    control._pendingChange = true;\n    control._pendingDirty = true;\n\n    if (control.updateOn === 'change') updateControl(control, dir);\n  });\n}\n\nfunction setUpBlurPipeline(control: FormControl, dir: NgControl): void {\n  dir.valueAccessor!.registerOnTouched(() => {\n    control._pendingTouched = true;\n\n    if (control.updateOn === 'blur' && control._pendingChange) updateControl(control, dir);\n    if (control.updateOn !== 'submit') control.markAsTouched();\n  });\n}\n\nfunction updateControl(control: FormControl, dir: NgControl): void {\n  if (control._pendingDirty) control.markAsDirty();\n  control.setValue(control._pendingValue, {emitModelToViewChange: false});\n  dir.viewToModelUpdate(control._pendingValue);\n  control._pendingChange = false;\n}\n\nfunction setUpModelChangePipeline(control: FormControl, dir: NgControl): void {\n  const onChange = (newValue?: any, emitModelEvent?: boolean) => {\n    // control -> view\n    dir.valueAccessor!.writeValue(newValue);\n\n    // control -> ngModel\n    if (emitModelEvent) dir.viewToModelUpdate(newValue);\n  };\n  control.registerOnChange(onChange);\n\n  // Register a callback function to cleanup onChange handler\n  // from a control instance when a directive is destroyed.\n  dir._registerOnDestroy(() => {\n    control._unregisterOnChange(onChange);\n  });\n}\n\n/**\n * Links a FormGroup or FormArray instance and corresponding Form directive by setting up validators\n * present in the view.\n *\n * @param control FormGroup or FormArray instance that should be linked.\n * @param dir Directive that provides view validators.\n */\nexport function setUpFormContainer(\n  control: FormGroup | FormArray,\n  dir: AbstractFormGroupDirective | FormArrayName,\n) {\n  if (control == null && (typeof ngDevMode === 'undefined' || ngDevMode))\n    _throwError(dir, 'Cannot find control with');\n  setUpValidators(control, dir);\n}\n\n/**\n * Reverts the setup performed by the `setUpFormContainer` function.\n *\n * @param control FormGroup or FormArray instance that should be cleaned up.\n * @param dir Directive that provided view validators.\n * @returns true if a control was updated as a result of this action.\n */\nexport function cleanUpFormContainer(\n  control: FormGroup | FormArray,\n  dir: AbstractFormGroupDirective | FormArrayName,\n): boolean {\n  return cleanUpValidators(control, dir);\n}\n\nfunction _noControlError(dir: NgControl) {\n  return _throwError(dir, 'There is no FormControl instance attached to form control element with');\n}\n\nfunction _throwError(dir: AbstractControlDirective, message: string): void {\n  const messageEnd = _describeControlLocation(dir);\n  throw new Error(`${message} ${messageEnd}`);\n}\n\nfunction _describeControlLocation(dir: AbstractControlDirective): string {\n  const path = dir.path;\n  if (path && path.length > 1) return `path: '${path.join(' -> ')}'`;\n  if (path?.[0]) return `name: '${path}'`;\n  return 'unspecified name attribute';\n}\n\nfunction _throwMissingValueAccessorError(dir: AbstractControlDirective) {\n  const loc = _describeControlLocation(dir);\n  throw new RuntimeError(\n    RuntimeErrorCode.NG_MISSING_VALUE_ACCESSOR,\n    `No value accessor for form control ${loc}.`,\n  );\n}\n\nfunction _throwInvalidValueAccessorError(dir: AbstractControlDirective) {\n  const loc = _describeControlLocation(dir);\n  throw new RuntimeError(\n    RuntimeErrorCode.NG_VALUE_ACCESSOR_NOT_PROVIDED,\n    `Value accessor was not provided as an array for form control with ${loc}. ` +\n      `Check that the \\`NG_VALUE_ACCESSOR\\` token is configured as a \\`multi: true\\` provider.`,\n  );\n}\n\nexport function isPropertyUpdated(changes: {[key: string]: any}, viewModel: any): boolean {\n  if (!changes.hasOwnProperty('model')) return false;\n  const change = changes['model'];\n\n  if (change.isFirstChange()) return true;\n  return !Object.is(viewModel, change.currentValue);\n}\n\nexport function isBuiltInAccessor(valueAccessor: ControlValueAccessor): boolean {\n  // Check if a given value accessor is an instance of a class that directly extends\n  // `BuiltInControlValueAccessor` one.\n  return Object.getPrototypeOf(valueAccessor.constructor) === BuiltInControlValueAccessor;\n}\n\nexport function syncPendingControls(\n  form: FormGroup,\n  directives: Set<NgControl> | NgControl[],\n): void {\n  form._syncPendingControls();\n  directives.forEach((dir: NgControl) => {\n    const control = dir.control as FormControl;\n    if (control.updateOn === 'submit' && control._pendingChange) {\n      dir.viewToModelUpdate(control._pendingValue);\n      control._pendingChange = false;\n    }\n  });\n}\n\n// TODO: vsavkin remove it once https://github.com/angular/angular/issues/3011 is implemented\nexport function selectValueAccessor(\n  dir: NgControl,\n  valueAccessors: ControlValueAccessor[],\n): ControlValueAccessor | null {\n  if (!valueAccessors) return null;\n\n  if (!Array.isArray(valueAccessors) && (typeof ngDevMode === 'undefined' || ngDevMode))\n    _throwInvalidValueAccessorError(dir);\n\n  let defaultAccessor: ControlValueAccessor | undefined = undefined;\n  let builtinAccessor: ControlValueAccessor | undefined = undefined;\n  let customAccessor: ControlValueAccessor | undefined = undefined;\n\n  valueAccessors.forEach((v: ControlValueAccessor) => {\n    if (v.constructor === DefaultValueAccessor) {\n      defaultAccessor = v;\n    } else if (isBuiltInAccessor(v)) {\n      if (builtinAccessor && (typeof ngDevMode === 'undefined' || ngDevMode))\n        _throwError(dir, 'More than one built-in value accessor matches form control with');\n      builtinAccessor = v;\n    } else {\n      if (customAccessor && (typeof ngDevMode === 'undefined' || ngDevMode))\n        _throwError(dir, 'More than one custom value accessor matches form control with');\n      customAccessor = v;\n    }\n  });\n\n  if (customAccessor) return customAccessor;\n  if (builtinAccessor) return builtinAccessor;\n  if (defaultAccessor) return defaultAccessor;\n\n  if (typeof ngDevMode === 'undefined' || ngDevMode) {\n    _throwError(dir, 'No valid value accessor for form control with');\n  }\n  return null;\n}\n\nexport function removeListItem<T>(list: T[], el: T): void {\n  const index = list.indexOf(el);\n  if (index > -1) list.splice(index, 1);\n}\n\n// TODO(kara): remove after deprecation period\nexport function _ngModelWarning(\n  name: string,\n  type: {_ngModelWarningSentOnce: boolean},\n  instance: {_ngModelWarningSent: boolean},\n  warningConfig: string | null,\n) {\n  if (warningConfig === 'never') return;\n\n  if (\n    ((warningConfig === null || warningConfig === 'once') && !type._ngModelWarningSentOnce) ||\n    (warningConfig === 'always' && !instance._ngModelWarningSent)\n  ) {\n    console.warn(ngModelWarning(name));\n    type._ngModelWarningSentOnce = true;\n    instance._ngModelWarningSent = true;\n  }\n}\n"]}
@@ -1,54 +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 { ɵRuntimeError as RuntimeError } from '@angular/core';
9
- import { formControlNameExample, formGroupNameExample, ngModelGroupExample, ngModelWithFormGroupExample, } from './error_examples';
10
- export function modelParentException() {
11
- return new RuntimeError(1350 /* RuntimeErrorCode.NGMODEL_IN_FORM_GROUP */, `
12
- ngModel cannot be used to register form controls with a parent formGroup directive. Try using
13
- formGroup's partner directive "formControlName" instead. Example:
14
-
15
- ${formControlNameExample}
16
-
17
- Or, if you'd like to avoid registering this form control, indicate that it's standalone in ngModelOptions:
18
-
19
- Example:
20
-
21
- ${ngModelWithFormGroupExample}`);
22
- }
23
- export function formGroupNameException() {
24
- return new RuntimeError(1351 /* RuntimeErrorCode.NGMODEL_IN_FORM_GROUP_NAME */, `
25
- ngModel cannot be used to register form controls with a parent formGroupName or formArrayName directive.
26
-
27
- Option 1: Use formControlName instead of ngModel (reactive strategy):
28
-
29
- ${formGroupNameExample}
30
-
31
- Option 2: Update ngModel's parent be ngModelGroup (template-driven strategy):
32
-
33
- ${ngModelGroupExample}`);
34
- }
35
- export function missingNameException() {
36
- return new RuntimeError(1352 /* RuntimeErrorCode.NGMODEL_WITHOUT_NAME */, `If ngModel is used within a form tag, either the name attribute must be set or the form
37
- control must be defined as 'standalone' in ngModelOptions.
38
-
39
- Example 1: <input [(ngModel)]="person.firstName" name="first">
40
- Example 2: <input [(ngModel)]="person.firstName" [ngModelOptions]="{standalone: true}">`);
41
- }
42
- export function modelGroupParentException() {
43
- return new RuntimeError(1353 /* RuntimeErrorCode.NGMODELGROUP_IN_FORM_GROUP */, `
44
- ngModelGroup cannot be used with a parent formGroup directive.
45
-
46
- Option 1: Use formGroupName instead of ngModelGroup (reactive strategy):
47
-
48
- ${formGroupNameExample}
49
-
50
- Option 2: Use a regular form tag instead of the formGroup directive (template-driven strategy):
51
-
52
- ${ngModelGroupExample}`);
53
- }
54
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVtcGxhdGVfZHJpdmVuX2Vycm9ycy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2Zvcm1zL3NyYy9kaXJlY3RpdmVzL3RlbXBsYXRlX2RyaXZlbl9lcnJvcnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7OztHQU1HO0FBRUgsT0FBTyxFQUFDLGFBQWEsSUFBSSxZQUFZLEVBQUMsTUFBTSxlQUFlLENBQUM7QUFJNUQsT0FBTyxFQUNMLHNCQUFzQixFQUN0QixvQkFBb0IsRUFDcEIsbUJBQW1CLEVBQ25CLDJCQUEyQixHQUM1QixNQUFNLGtCQUFrQixDQUFDO0FBRTFCLE1BQU0sVUFBVSxvQkFBb0I7SUFDbEMsT0FBTyxJQUFJLFlBQVksb0RBRXJCOzs7O01BSUUsc0JBQXNCOzs7Ozs7TUFNdEIsMkJBQTJCLEVBQUUsQ0FDaEMsQ0FBQztBQUNKLENBQUM7QUFFRCxNQUFNLFVBQVUsc0JBQXNCO0lBQ3BDLE9BQU8sSUFBSSxZQUFZLHlEQUVyQjs7Ozs7TUFLRSxvQkFBb0I7Ozs7TUFJcEIsbUJBQW1CLEVBQUUsQ0FDeEIsQ0FBQztBQUNKLENBQUM7QUFFRCxNQUFNLFVBQVUsb0JBQW9CO0lBQ2xDLE9BQU8sSUFBSSxZQUFZLG1EQUVyQjs7Ozs0RkFJd0YsQ0FDekYsQ0FBQztBQUNKLENBQUM7QUFFRCxNQUFNLFVBQVUseUJBQXlCO0lBQ3ZDLE9BQU8sSUFBSSxZQUFZLHlEQUVyQjs7Ozs7TUFLRSxvQkFBb0I7Ozs7TUFJcEIsbUJBQW1CLEVBQUUsQ0FDeEIsQ0FBQztBQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cblxuaW1wb3J0IHvJtVJ1bnRpbWVFcnJvciBhcyBSdW50aW1lRXJyb3J9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5pbXBvcnQge1J1bnRpbWVFcnJvckNvZGV9IGZyb20gJy4uL2Vycm9ycyc7XG5cbmltcG9ydCB7XG4gIGZvcm1Db250cm9sTmFtZUV4YW1wbGUsXG4gIGZvcm1Hcm91cE5hbWVFeGFtcGxlLFxuICBuZ01vZGVsR3JvdXBFeGFtcGxlLFxuICBuZ01vZGVsV2l0aEZvcm1Hcm91cEV4YW1wbGUsXG59IGZyb20gJy4vZXJyb3JfZXhhbXBsZXMnO1xuXG5leHBvcnQgZnVuY3Rpb24gbW9kZWxQYXJlbnRFeGNlcHRpb24oKTogRXJyb3Ige1xuICByZXR1cm4gbmV3IFJ1bnRpbWVFcnJvcihcbiAgICBSdW50aW1lRXJyb3JDb2RlLk5HTU9ERUxfSU5fRk9STV9HUk9VUCxcbiAgICBgXG4gICAgbmdNb2RlbCBjYW5ub3QgYmUgdXNlZCB0byByZWdpc3RlciBmb3JtIGNvbnRyb2xzIHdpdGggYSBwYXJlbnQgZm9ybUdyb3VwIGRpcmVjdGl2ZS4gIFRyeSB1c2luZ1xuICAgIGZvcm1Hcm91cCdzIHBhcnRuZXIgZGlyZWN0aXZlIFwiZm9ybUNvbnRyb2xOYW1lXCIgaW5zdGVhZC4gIEV4YW1wbGU6XG5cbiAgICAke2Zvcm1Db250cm9sTmFtZUV4YW1wbGV9XG5cbiAgICBPciwgaWYgeW91J2QgbGlrZSB0byBhdm9pZCByZWdpc3RlcmluZyB0aGlzIGZvcm0gY29udHJvbCwgaW5kaWNhdGUgdGhhdCBpdCdzIHN0YW5kYWxvbmUgaW4gbmdNb2RlbE9wdGlvbnM6XG5cbiAgICBFeGFtcGxlOlxuXG4gICAgJHtuZ01vZGVsV2l0aEZvcm1Hcm91cEV4YW1wbGV9YCxcbiAgKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGZvcm1Hcm91cE5hbWVFeGNlcHRpb24oKTogRXJyb3Ige1xuICByZXR1cm4gbmV3IFJ1bnRpbWVFcnJvcihcbiAgICBSdW50aW1lRXJyb3JDb2RlLk5HTU9ERUxfSU5fRk9STV9HUk9VUF9OQU1FLFxuICAgIGBcbiAgICBuZ01vZGVsIGNhbm5vdCBiZSB1c2VkIHRvIHJlZ2lzdGVyIGZvcm0gY29udHJvbHMgd2l0aCBhIHBhcmVudCBmb3JtR3JvdXBOYW1lIG9yIGZvcm1BcnJheU5hbWUgZGlyZWN0aXZlLlxuXG4gICAgT3B0aW9uIDE6IFVzZSBmb3JtQ29udHJvbE5hbWUgaW5zdGVhZCBvZiBuZ01vZGVsIChyZWFjdGl2ZSBzdHJhdGVneSk6XG5cbiAgICAke2Zvcm1Hcm91cE5hbWVFeGFtcGxlfVxuXG4gICAgT3B0aW9uIDI6ICBVcGRhdGUgbmdNb2RlbCdzIHBhcmVudCBiZSBuZ01vZGVsR3JvdXAgKHRlbXBsYXRlLWRyaXZlbiBzdHJhdGVneSk6XG5cbiAgICAke25nTW9kZWxHcm91cEV4YW1wbGV9YCxcbiAgKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIG1pc3NpbmdOYW1lRXhjZXB0aW9uKCk6IEVycm9yIHtcbiAgcmV0dXJuIG5ldyBSdW50aW1lRXJyb3IoXG4gICAgUnVudGltZUVycm9yQ29kZS5OR01PREVMX1dJVEhPVVRfTkFNRSxcbiAgICBgSWYgbmdNb2RlbCBpcyB1c2VkIHdpdGhpbiBhIGZvcm0gdGFnLCBlaXRoZXIgdGhlIG5hbWUgYXR0cmlidXRlIG11c3QgYmUgc2V0IG9yIHRoZSBmb3JtXG4gICAgY29udHJvbCBtdXN0IGJlIGRlZmluZWQgYXMgJ3N0YW5kYWxvbmUnIGluIG5nTW9kZWxPcHRpb25zLlxuXG4gICAgRXhhbXBsZSAxOiA8aW5wdXQgWyhuZ01vZGVsKV09XCJwZXJzb24uZmlyc3ROYW1lXCIgbmFtZT1cImZpcnN0XCI+XG4gICAgRXhhbXBsZSAyOiA8aW5wdXQgWyhuZ01vZGVsKV09XCJwZXJzb24uZmlyc3ROYW1lXCIgW25nTW9kZWxPcHRpb25zXT1cIntzdGFuZGFsb25lOiB0cnVlfVwiPmAsXG4gICk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBtb2RlbEdyb3VwUGFyZW50RXhjZXB0aW9uKCk6IEVycm9yIHtcbiAgcmV0dXJuIG5ldyBSdW50aW1lRXJyb3IoXG4gICAgUnVudGltZUVycm9yQ29kZS5OR01PREVMR1JPVVBfSU5fRk9STV9HUk9VUCxcbiAgICBgXG4gICAgbmdNb2RlbEdyb3VwIGNhbm5vdCBiZSB1c2VkIHdpdGggYSBwYXJlbnQgZm9ybUdyb3VwIGRpcmVjdGl2ZS5cblxuICAgIE9wdGlvbiAxOiBVc2UgZm9ybUdyb3VwTmFtZSBpbnN0ZWFkIG9mIG5nTW9kZWxHcm91cCAocmVhY3RpdmUgc3RyYXRlZ3kpOlxuXG4gICAgJHtmb3JtR3JvdXBOYW1lRXhhbXBsZX1cblxuICAgIE9wdGlvbiAyOiAgVXNlIGEgcmVndWxhciBmb3JtIHRhZyBpbnN0ZWFkIG9mIHRoZSBmb3JtR3JvdXAgZGlyZWN0aXZlICh0ZW1wbGF0ZS1kcml2ZW4gc3RyYXRlZ3kpOlxuXG4gICAgJHtuZ01vZGVsR3JvdXBFeGFtcGxlfWAsXG4gICk7XG59XG4iXX0=