@angular/forms 19.0.0-next.1 → 19.0.0-next.11
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/fesm2022/forms.mjs +186 -132
- package/fesm2022/forms.mjs.map +1 -1
- package/index.d.ts +6 -5
- package/package.json +4 -6
- package/esm2022/forms.mjs +0 -5
- package/esm2022/index.mjs +0 -13
- package/esm2022/public_api.mjs +0 -15
- package/esm2022/src/directives/abstract_control_directive.mjs +0 -279
- package/esm2022/src/directives/abstract_form_group_directive.mjs +0 -61
- package/esm2022/src/directives/checkbox_value_accessor.mjs +0 -58
- package/esm2022/src/directives/control_container.mjs +0 -32
- package/esm2022/src/directives/control_value_accessor.mjs +0 -92
- package/esm2022/src/directives/default_value_accessor.mjs +0 -124
- package/esm2022/src/directives/error_examples.mjs +0 -57
- package/esm2022/src/directives/form_interface.mjs +0 -9
- package/esm2022/src/directives/ng_control.mjs +0 -38
- package/esm2022/src/directives/ng_control_status.mjs +0 -139
- package/esm2022/src/directives/ng_form.mjs +0 -313
- package/esm2022/src/directives/ng_model.mjs +0 -331
- package/esm2022/src/directives/ng_model_group.mjs +0 -95
- package/esm2022/src/directives/ng_no_validate_directive.mjs +0 -39
- package/esm2022/src/directives/number_value_accessor.mjs +0 -70
- package/esm2022/src/directives/radio_control_value_accessor.mjs +0 -201
- package/esm2022/src/directives/range_value_accessor.mjs +0 -72
- package/esm2022/src/directives/reactive_directives/form_control_directive.mjs +0 -183
- package/esm2022/src/directives/reactive_directives/form_control_name.mjs +0 -216
- package/esm2022/src/directives/reactive_directives/form_group_directive.mjs +0 -359
- package/esm2022/src/directives/reactive_directives/form_group_name.mjs +0 -246
- package/esm2022/src/directives/reactive_errors.mjs +0 -119
- package/esm2022/src/directives/select_control_value_accessor.mjs +0 -220
- package/esm2022/src/directives/select_multiple_control_value_accessor.mjs +0 -252
- package/esm2022/src/directives/shared.mjs +0 -350
- package/esm2022/src/directives/template_driven_errors.mjs +0 -54
- package/esm2022/src/directives/validators.mjs +0 -505
- package/esm2022/src/directives.mjs +0 -128
- package/esm2022/src/errors.mjs +0 -9
- package/esm2022/src/form_builder.mjs +0 -258
- package/esm2022/src/form_providers.mjs +0 -105
- package/esm2022/src/forms.mjs +0 -49
- package/esm2022/src/model/abstract_model.mjs +0 -992
- package/esm2022/src/model/form_array.mjs +0 -461
- package/esm2022/src/model/form_control.mjs +0 -126
- package/esm2022/src/model/form_group.mjs +0 -490
- package/esm2022/src/util.mjs +0 -13
- package/esm2022/src/validators.mjs +0 -680
- package/esm2022/src/version.mjs +0 -18
|
@@ -1,246 +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 { Directive, forwardRef, Host, Inject, Input, Optional, Self, SkipSelf, } from '@angular/core';
|
|
9
|
-
import { NG_ASYNC_VALIDATORS, NG_VALIDATORS } from '../../validators';
|
|
10
|
-
import { AbstractFormGroupDirective } from '../abstract_form_group_directive';
|
|
11
|
-
import { ControlContainer } from '../control_container';
|
|
12
|
-
import { arrayParentException, groupParentException } from '../reactive_errors';
|
|
13
|
-
import { controlPath } from '../shared';
|
|
14
|
-
import { FormGroupDirective } from './form_group_directive';
|
|
15
|
-
import * as i0 from "@angular/core";
|
|
16
|
-
import * as i1 from "../control_container";
|
|
17
|
-
const formGroupNameProvider = {
|
|
18
|
-
provide: ControlContainer,
|
|
19
|
-
useExisting: forwardRef(() => FormGroupName),
|
|
20
|
-
};
|
|
21
|
-
/**
|
|
22
|
-
* @description
|
|
23
|
-
*
|
|
24
|
-
* Syncs a nested `FormGroup` or `FormRecord` to a DOM element.
|
|
25
|
-
*
|
|
26
|
-
* This directive can only be used with a parent `FormGroupDirective`.
|
|
27
|
-
*
|
|
28
|
-
* It accepts the string name of the nested `FormGroup` or `FormRecord` to link, and
|
|
29
|
-
* looks for a `FormGroup` or `FormRecord` registered with that name in the parent
|
|
30
|
-
* `FormGroup` instance you passed into `FormGroupDirective`.
|
|
31
|
-
*
|
|
32
|
-
* Use nested form groups to validate a sub-group of a
|
|
33
|
-
* form separately from the rest or to group the values of certain
|
|
34
|
-
* controls into their own nested object.
|
|
35
|
-
*
|
|
36
|
-
* @see [Reactive Forms Guide](guide/forms/reactive-forms)
|
|
37
|
-
*
|
|
38
|
-
* @usageNotes
|
|
39
|
-
*
|
|
40
|
-
* ### Access the group by name
|
|
41
|
-
*
|
|
42
|
-
* The following example uses the `AbstractControl.get` method to access the
|
|
43
|
-
* associated `FormGroup`
|
|
44
|
-
*
|
|
45
|
-
* ```ts
|
|
46
|
-
* this.form.get('name');
|
|
47
|
-
* ```
|
|
48
|
-
*
|
|
49
|
-
* ### Access individual controls in the group
|
|
50
|
-
*
|
|
51
|
-
* The following example uses the `AbstractControl.get` method to access
|
|
52
|
-
* individual controls within the group using dot syntax.
|
|
53
|
-
*
|
|
54
|
-
* ```ts
|
|
55
|
-
* this.form.get('name.first');
|
|
56
|
-
* ```
|
|
57
|
-
*
|
|
58
|
-
* ### Register a nested `FormGroup`.
|
|
59
|
-
*
|
|
60
|
-
* The following example registers a nested *name* `FormGroup` within an existing `FormGroup`,
|
|
61
|
-
* and provides methods to retrieve the nested `FormGroup` and individual controls.
|
|
62
|
-
*
|
|
63
|
-
* {@example forms/ts/nestedFormGroup/nested_form_group_example.ts region='Component'}
|
|
64
|
-
*
|
|
65
|
-
* @ngModule ReactiveFormsModule
|
|
66
|
-
* @publicApi
|
|
67
|
-
*/
|
|
68
|
-
export class FormGroupName extends AbstractFormGroupDirective {
|
|
69
|
-
constructor(parent, validators, asyncValidators) {
|
|
70
|
-
super();
|
|
71
|
-
/**
|
|
72
|
-
* @description
|
|
73
|
-
* Tracks the name of the `FormGroup` bound to the directive. The name corresponds
|
|
74
|
-
* to a key in the parent `FormGroup` or `FormArray`.
|
|
75
|
-
* Accepts a name as a string or a number.
|
|
76
|
-
* The name in the form of a string is useful for individual forms,
|
|
77
|
-
* while the numerical form allows for form groups to be bound
|
|
78
|
-
* to indices when iterating over groups in a `FormArray`.
|
|
79
|
-
*/
|
|
80
|
-
this.name = null;
|
|
81
|
-
this._parent = parent;
|
|
82
|
-
this._setValidators(validators);
|
|
83
|
-
this._setAsyncValidators(asyncValidators);
|
|
84
|
-
}
|
|
85
|
-
/** @internal */
|
|
86
|
-
_checkParentType() {
|
|
87
|
-
if (_hasInvalidParent(this._parent) && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
|
88
|
-
throw groupParentException();
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.0.0-next.1", ngImport: i0, type: FormGroupName, deps: [{ token: i1.ControlContainer, host: true, optional: true, skipSelf: true }, { token: NG_VALIDATORS, optional: true, self: true }, { token: NG_ASYNC_VALIDATORS, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
92
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "19.0.0-next.1", type: FormGroupName, selector: "[formGroupName]", inputs: { name: ["formGroupName", "name"] }, providers: [formGroupNameProvider], usesInheritance: true, ngImport: i0 }); }
|
|
93
|
-
}
|
|
94
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.0.0-next.1", ngImport: i0, type: FormGroupName, decorators: [{
|
|
95
|
-
type: Directive,
|
|
96
|
-
args: [{ selector: '[formGroupName]', providers: [formGroupNameProvider] }]
|
|
97
|
-
}], ctorParameters: () => [{ type: i1.ControlContainer, decorators: [{
|
|
98
|
-
type: Optional
|
|
99
|
-
}, {
|
|
100
|
-
type: Host
|
|
101
|
-
}, {
|
|
102
|
-
type: SkipSelf
|
|
103
|
-
}] }, { type: undefined, decorators: [{
|
|
104
|
-
type: Optional
|
|
105
|
-
}, {
|
|
106
|
-
type: Self
|
|
107
|
-
}, {
|
|
108
|
-
type: Inject,
|
|
109
|
-
args: [NG_VALIDATORS]
|
|
110
|
-
}] }, { type: undefined, decorators: [{
|
|
111
|
-
type: Optional
|
|
112
|
-
}, {
|
|
113
|
-
type: Self
|
|
114
|
-
}, {
|
|
115
|
-
type: Inject,
|
|
116
|
-
args: [NG_ASYNC_VALIDATORS]
|
|
117
|
-
}] }], propDecorators: { name: [{
|
|
118
|
-
type: Input,
|
|
119
|
-
args: ['formGroupName']
|
|
120
|
-
}] } });
|
|
121
|
-
export const formArrayNameProvider = {
|
|
122
|
-
provide: ControlContainer,
|
|
123
|
-
useExisting: forwardRef(() => FormArrayName),
|
|
124
|
-
};
|
|
125
|
-
/**
|
|
126
|
-
* @description
|
|
127
|
-
*
|
|
128
|
-
* Syncs a nested `FormArray` to a DOM element.
|
|
129
|
-
*
|
|
130
|
-
* This directive is designed to be used with a parent `FormGroupDirective` (selector:
|
|
131
|
-
* `[formGroup]`).
|
|
132
|
-
*
|
|
133
|
-
* It accepts the string name of the nested `FormArray` you want to link, and
|
|
134
|
-
* will look for a `FormArray` registered with that name in the parent
|
|
135
|
-
* `FormGroup` instance you passed into `FormGroupDirective`.
|
|
136
|
-
*
|
|
137
|
-
* @see [Reactive Forms Guide](guide/forms/reactive-forms)
|
|
138
|
-
* @see {@link AbstractControl}
|
|
139
|
-
*
|
|
140
|
-
* @usageNotes
|
|
141
|
-
*
|
|
142
|
-
* ### Example
|
|
143
|
-
*
|
|
144
|
-
* {@example forms/ts/nestedFormArray/nested_form_array_example.ts region='Component'}
|
|
145
|
-
*
|
|
146
|
-
* @ngModule ReactiveFormsModule
|
|
147
|
-
* @publicApi
|
|
148
|
-
*/
|
|
149
|
-
export class FormArrayName extends ControlContainer {
|
|
150
|
-
constructor(parent, validators, asyncValidators) {
|
|
151
|
-
super();
|
|
152
|
-
/**
|
|
153
|
-
* @description
|
|
154
|
-
* Tracks the name of the `FormArray` bound to the directive. The name corresponds
|
|
155
|
-
* to a key in the parent `FormGroup` or `FormArray`.
|
|
156
|
-
* Accepts a name as a string or a number.
|
|
157
|
-
* The name in the form of a string is useful for individual forms,
|
|
158
|
-
* while the numerical form allows for form arrays to be bound
|
|
159
|
-
* to indices when iterating over arrays in a `FormArray`.
|
|
160
|
-
*/
|
|
161
|
-
this.name = null;
|
|
162
|
-
this._parent = parent;
|
|
163
|
-
this._setValidators(validators);
|
|
164
|
-
this._setAsyncValidators(asyncValidators);
|
|
165
|
-
}
|
|
166
|
-
/**
|
|
167
|
-
* A lifecycle method called when the directive's inputs are initialized. For internal use only.
|
|
168
|
-
* @throws If the directive does not have a valid parent.
|
|
169
|
-
* @nodoc
|
|
170
|
-
*/
|
|
171
|
-
ngOnInit() {
|
|
172
|
-
this._checkParentType();
|
|
173
|
-
this.formDirective.addFormArray(this);
|
|
174
|
-
}
|
|
175
|
-
/**
|
|
176
|
-
* A lifecycle method called before the directive's instance is destroyed. For internal use only.
|
|
177
|
-
* @nodoc
|
|
178
|
-
*/
|
|
179
|
-
ngOnDestroy() {
|
|
180
|
-
if (this.formDirective) {
|
|
181
|
-
this.formDirective.removeFormArray(this);
|
|
182
|
-
}
|
|
183
|
-
}
|
|
184
|
-
/**
|
|
185
|
-
* @description
|
|
186
|
-
* The `FormArray` bound to this directive.
|
|
187
|
-
*/
|
|
188
|
-
get control() {
|
|
189
|
-
return this.formDirective.getFormArray(this);
|
|
190
|
-
}
|
|
191
|
-
/**
|
|
192
|
-
* @description
|
|
193
|
-
* The top-level directive for this group if present, otherwise null.
|
|
194
|
-
*/
|
|
195
|
-
get formDirective() {
|
|
196
|
-
return this._parent ? this._parent.formDirective : null;
|
|
197
|
-
}
|
|
198
|
-
/**
|
|
199
|
-
* @description
|
|
200
|
-
* Returns an array that represents the path from the top-level form to this control.
|
|
201
|
-
* Each index is the string name of the control on that level.
|
|
202
|
-
*/
|
|
203
|
-
get path() {
|
|
204
|
-
return controlPath(this.name == null ? this.name : this.name.toString(), this._parent);
|
|
205
|
-
}
|
|
206
|
-
_checkParentType() {
|
|
207
|
-
if (_hasInvalidParent(this._parent) && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
|
208
|
-
throw arrayParentException();
|
|
209
|
-
}
|
|
210
|
-
}
|
|
211
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.0.0-next.1", ngImport: i0, type: FormArrayName, deps: [{ token: i1.ControlContainer, host: true, optional: true, skipSelf: true }, { token: NG_VALIDATORS, optional: true, self: true }, { token: NG_ASYNC_VALIDATORS, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
212
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "19.0.0-next.1", type: FormArrayName, selector: "[formArrayName]", inputs: { name: ["formArrayName", "name"] }, providers: [formArrayNameProvider], usesInheritance: true, ngImport: i0 }); }
|
|
213
|
-
}
|
|
214
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.0.0-next.1", ngImport: i0, type: FormArrayName, decorators: [{
|
|
215
|
-
type: Directive,
|
|
216
|
-
args: [{ selector: '[formArrayName]', providers: [formArrayNameProvider] }]
|
|
217
|
-
}], ctorParameters: () => [{ type: i1.ControlContainer, decorators: [{
|
|
218
|
-
type: Optional
|
|
219
|
-
}, {
|
|
220
|
-
type: Host
|
|
221
|
-
}, {
|
|
222
|
-
type: SkipSelf
|
|
223
|
-
}] }, { type: undefined, decorators: [{
|
|
224
|
-
type: Optional
|
|
225
|
-
}, {
|
|
226
|
-
type: Self
|
|
227
|
-
}, {
|
|
228
|
-
type: Inject,
|
|
229
|
-
args: [NG_VALIDATORS]
|
|
230
|
-
}] }, { type: undefined, decorators: [{
|
|
231
|
-
type: Optional
|
|
232
|
-
}, {
|
|
233
|
-
type: Self
|
|
234
|
-
}, {
|
|
235
|
-
type: Inject,
|
|
236
|
-
args: [NG_ASYNC_VALIDATORS]
|
|
237
|
-
}] }], propDecorators: { name: [{
|
|
238
|
-
type: Input,
|
|
239
|
-
args: ['formArrayName']
|
|
240
|
-
}] } });
|
|
241
|
-
function _hasInvalidParent(parent) {
|
|
242
|
-
return (!(parent instanceof FormGroupName) &&
|
|
243
|
-
!(parent instanceof FormGroupDirective) &&
|
|
244
|
-
!(parent instanceof FormArrayName));
|
|
245
|
-
}
|
|
246
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form_group_name.js","sourceRoot":"","sources":["../../../../../../../../packages/forms/src/directives/reactive_directives/form_group_name.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  Directive,\n  forwardRef,\n  Host,\n  Inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Provider,\n  Self,\n  SkipSelf,\n} from '@angular/core';\n\nimport {FormArray} from '../../model/form_array';\nimport {NG_ASYNC_VALIDATORS, NG_VALIDATORS} from '../../validators';\nimport {AbstractFormGroupDirective} from '../abstract_form_group_directive';\nimport {ControlContainer} from '../control_container';\nimport {arrayParentException, groupParentException} from '../reactive_errors';\nimport {controlPath} from '../shared';\nimport {AsyncValidator, AsyncValidatorFn, Validator, ValidatorFn} from '../validators';\n\nimport {FormGroupDirective} from './form_group_directive';\n\nconst formGroupNameProvider: Provider = {\n  provide: ControlContainer,\n  useExisting: forwardRef(() => FormGroupName),\n};\n\n/**\n * @description\n *\n * Syncs a nested `FormGroup` or `FormRecord` to a DOM element.\n *\n * This directive can only be used with a parent `FormGroupDirective`.\n *\n * It accepts the string name of the nested `FormGroup` or `FormRecord` to link, and\n * looks for a `FormGroup` or `FormRecord` registered with that name in the parent\n * `FormGroup` instance you passed into `FormGroupDirective`.\n *\n * Use nested form groups to validate a sub-group of a\n * form separately from the rest or to group the values of certain\n * controls into their own nested object.\n *\n * @see [Reactive Forms Guide](guide/forms/reactive-forms)\n *\n * @usageNotes\n *\n * ### Access the group by name\n *\n * The following example uses the `AbstractControl.get` method to access the\n * associated `FormGroup`\n *\n * ```ts\n *   this.form.get('name');\n * ```\n *\n * ### Access individual controls in the group\n *\n * The following example uses the `AbstractControl.get` method to access\n * individual controls within the group using dot syntax.\n *\n * ```ts\n *   this.form.get('name.first');\n * ```\n *\n * ### Register a nested `FormGroup`.\n *\n * The following example registers a nested *name* `FormGroup` within an existing `FormGroup`,\n * and provides methods to retrieve the nested `FormGroup` and individual controls.\n *\n * {@example forms/ts/nestedFormGroup/nested_form_group_example.ts region='Component'}\n *\n * @ngModule ReactiveFormsModule\n * @publicApi\n */\n@Directive({selector: '[formGroupName]', providers: [formGroupNameProvider]})\nexport class FormGroupName extends AbstractFormGroupDirective implements OnInit, OnDestroy {\n  /**\n   * @description\n   * Tracks the name of the `FormGroup` bound to the directive. The name corresponds\n   * to a key in the parent `FormGroup` or `FormArray`.\n   * Accepts a name as a string or a number.\n   * The name in the form of a string is useful for individual forms,\n   * while the numerical form allows for form groups to be bound\n   * to indices when iterating over groups in a `FormArray`.\n   */\n  @Input('formGroupName') override name: string | number | null = null;\n\n  constructor(\n    @Optional() @Host() @SkipSelf() parent: ControlContainer,\n    @Optional() @Self() @Inject(NG_VALIDATORS) validators: (Validator | ValidatorFn)[],\n    @Optional()\n    @Self()\n    @Inject(NG_ASYNC_VALIDATORS)\n    asyncValidators: (AsyncValidator | AsyncValidatorFn)[],\n  ) {\n    super();\n    this._parent = parent;\n    this._setValidators(validators);\n    this._setAsyncValidators(asyncValidators);\n  }\n\n  /** @internal */\n  override _checkParentType(): void {\n    if (_hasInvalidParent(this._parent) && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw groupParentException();\n    }\n  }\n}\n\nexport const formArrayNameProvider: any = {\n  provide: ControlContainer,\n  useExisting: forwardRef(() => FormArrayName),\n};\n\n/**\n * @description\n *\n * Syncs a nested `FormArray` to a DOM element.\n *\n * This directive is designed to be used with a parent `FormGroupDirective` (selector:\n * `[formGroup]`).\n *\n * It accepts the string name of the nested `FormArray` you want to link, and\n * will look for a `FormArray` registered with that name in the parent\n * `FormGroup` instance you passed into `FormGroupDirective`.\n *\n * @see [Reactive Forms Guide](guide/forms/reactive-forms)\n * @see {@link AbstractControl}\n *\n * @usageNotes\n *\n * ### Example\n *\n * {@example forms/ts/nestedFormArray/nested_form_array_example.ts region='Component'}\n *\n * @ngModule ReactiveFormsModule\n * @publicApi\n */\n@Directive({selector: '[formArrayName]', providers: [formArrayNameProvider]})\nexport class FormArrayName extends ControlContainer implements OnInit, OnDestroy {\n  /** @internal */\n  _parent: ControlContainer;\n\n  /**\n   * @description\n   * Tracks the name of the `FormArray` bound to the directive. The name corresponds\n   * to a key in the parent `FormGroup` or `FormArray`.\n   * Accepts a name as a string or a number.\n   * The name in the form of a string is useful for individual forms,\n   * while the numerical form allows for form arrays to be bound\n   * to indices when iterating over arrays in a `FormArray`.\n   */\n  @Input('formArrayName') override name: string | number | null = null;\n\n  constructor(\n    @Optional() @Host() @SkipSelf() parent: ControlContainer,\n    @Optional() @Self() @Inject(NG_VALIDATORS) validators: (Validator | ValidatorFn)[],\n    @Optional()\n    @Self()\n    @Inject(NG_ASYNC_VALIDATORS)\n    asyncValidators: (AsyncValidator | AsyncValidatorFn)[],\n  ) {\n    super();\n    this._parent = parent;\n    this._setValidators(validators);\n    this._setAsyncValidators(asyncValidators);\n  }\n\n  /**\n   * A lifecycle method called when the directive's inputs are initialized. For internal use only.\n   * @throws If the directive does not have a valid parent.\n   * @nodoc\n   */\n  ngOnInit(): void {\n    this._checkParentType();\n    this.formDirective!.addFormArray(this);\n  }\n\n  /**\n   * A lifecycle method called before the directive's instance is destroyed. For internal use only.\n   * @nodoc\n   */\n  ngOnDestroy(): void {\n    if (this.formDirective) {\n      this.formDirective.removeFormArray(this);\n    }\n  }\n\n  /**\n   * @description\n   * The `FormArray` bound to this directive.\n   */\n  override get control(): FormArray {\n    return this.formDirective!.getFormArray(this);\n  }\n\n  /**\n   * @description\n   * The top-level directive for this group if present, otherwise null.\n   */\n  override get formDirective(): FormGroupDirective | null {\n    return this._parent ? <FormGroupDirective>this._parent.formDirective : null;\n  }\n\n  /**\n   * @description\n   * Returns an array that represents the path from the top-level form to this control.\n   * Each index is the string name of the control on that level.\n   */\n  override get path(): string[] {\n    return controlPath(this.name == null ? this.name : this.name.toString(), this._parent);\n  }\n\n  private _checkParentType(): void {\n    if (_hasInvalidParent(this._parent) && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw arrayParentException();\n    }\n  }\n}\n\nfunction _hasInvalidParent(parent: ControlContainer): boolean {\n  return (\n    !(parent instanceof FormGroupName) &&\n    !(parent instanceof FormGroupDirective) &&\n    !(parent instanceof FormArrayName)\n  );\n}\n"]}
|
|
@@ -1,119 +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 { formArrayNameExample, formControlNameExample, formGroupNameExample, ngModelGroupExample, } from './error_examples';
|
|
10
|
-
export function controlParentException(nameOrIndex) {
|
|
11
|
-
return new RuntimeError(1050 /* RuntimeErrorCode.FORM_CONTROL_NAME_MISSING_PARENT */, `formControlName must be used with a parent formGroup directive. You'll want to add a formGroup
|
|
12
|
-
directive and pass it an existing FormGroup instance (you can create one in your class).
|
|
13
|
-
|
|
14
|
-
${describeFormControl(nameOrIndex)}
|
|
15
|
-
|
|
16
|
-
Example:
|
|
17
|
-
|
|
18
|
-
${formControlNameExample}`);
|
|
19
|
-
}
|
|
20
|
-
function describeFormControl(nameOrIndex) {
|
|
21
|
-
if (nameOrIndex == null || nameOrIndex === '') {
|
|
22
|
-
return '';
|
|
23
|
-
}
|
|
24
|
-
const valueType = typeof nameOrIndex === 'string' ? 'name' : 'index';
|
|
25
|
-
return `Affected Form Control ${valueType}: "${nameOrIndex}"`;
|
|
26
|
-
}
|
|
27
|
-
export function ngModelGroupException() {
|
|
28
|
-
return new RuntimeError(1051 /* RuntimeErrorCode.FORM_CONTROL_NAME_INSIDE_MODEL_GROUP */, `formControlName cannot be used with an ngModelGroup parent. It is only compatible with parents
|
|
29
|
-
that also have a "form" prefix: formGroupName, formArrayName, or formGroup.
|
|
30
|
-
|
|
31
|
-
Option 1: Update the parent to be formGroupName (reactive form strategy)
|
|
32
|
-
|
|
33
|
-
${formGroupNameExample}
|
|
34
|
-
|
|
35
|
-
Option 2: Use ngModel instead of formControlName (template-driven strategy)
|
|
36
|
-
|
|
37
|
-
${ngModelGroupExample}`);
|
|
38
|
-
}
|
|
39
|
-
export function missingFormException() {
|
|
40
|
-
return new RuntimeError(1052 /* RuntimeErrorCode.FORM_GROUP_MISSING_INSTANCE */, `formGroup expects a FormGroup instance. Please pass one in.
|
|
41
|
-
|
|
42
|
-
Example:
|
|
43
|
-
|
|
44
|
-
${formControlNameExample}`);
|
|
45
|
-
}
|
|
46
|
-
export function groupParentException() {
|
|
47
|
-
return new RuntimeError(1053 /* RuntimeErrorCode.FORM_GROUP_NAME_MISSING_PARENT */, `formGroupName must be used with a parent formGroup directive. You'll want to add a formGroup
|
|
48
|
-
directive and pass it an existing FormGroup instance (you can create one in your class).
|
|
49
|
-
|
|
50
|
-
Example:
|
|
51
|
-
|
|
52
|
-
${formGroupNameExample}`);
|
|
53
|
-
}
|
|
54
|
-
export function arrayParentException() {
|
|
55
|
-
return new RuntimeError(1054 /* RuntimeErrorCode.FORM_ARRAY_NAME_MISSING_PARENT */, `formArrayName must be used with a parent formGroup directive. You'll want to add a formGroup
|
|
56
|
-
directive and pass it an existing FormGroup instance (you can create one in your class).
|
|
57
|
-
|
|
58
|
-
Example:
|
|
59
|
-
|
|
60
|
-
${formArrayNameExample}`);
|
|
61
|
-
}
|
|
62
|
-
export const disabledAttrWarning = `
|
|
63
|
-
It looks like you're using the disabled attribute with a reactive form directive. If you set disabled to true
|
|
64
|
-
when you set up this control in your component class, the disabled attribute will actually be set in the DOM for
|
|
65
|
-
you. We recommend using this approach to avoid 'changed after checked' errors.
|
|
66
|
-
|
|
67
|
-
Example:
|
|
68
|
-
// Specify the \`disabled\` property at control creation time:
|
|
69
|
-
form = new FormGroup({
|
|
70
|
-
first: new FormControl({value: 'Nancy', disabled: true}, Validators.required),
|
|
71
|
-
last: new FormControl('Drew', Validators.required)
|
|
72
|
-
});
|
|
73
|
-
|
|
74
|
-
// Controls can also be enabled/disabled after creation:
|
|
75
|
-
form.get('first')?.enable();
|
|
76
|
-
form.get('last')?.disable();
|
|
77
|
-
`;
|
|
78
|
-
export const asyncValidatorsDroppedWithOptsWarning = `
|
|
79
|
-
It looks like you're constructing using a FormControl with both an options argument and an
|
|
80
|
-
async validators argument. Mixing these arguments will cause your async validators to be dropped.
|
|
81
|
-
You should either put all your validators in the options object, or in separate validators
|
|
82
|
-
arguments. For example:
|
|
83
|
-
|
|
84
|
-
// Using validators arguments
|
|
85
|
-
fc = new FormControl(42, Validators.required, myAsyncValidator);
|
|
86
|
-
|
|
87
|
-
// Using AbstractControlOptions
|
|
88
|
-
fc = new FormControl(42, {validators: Validators.required, asyncValidators: myAV});
|
|
89
|
-
|
|
90
|
-
// Do NOT mix them: async validators will be dropped!
|
|
91
|
-
fc = new FormControl(42, {validators: Validators.required}, /* Oops! */ myAsyncValidator);
|
|
92
|
-
`;
|
|
93
|
-
export function ngModelWarning(directiveName) {
|
|
94
|
-
return `
|
|
95
|
-
It looks like you're using ngModel on the same form field as ${directiveName}.
|
|
96
|
-
Support for using the ngModel input property and ngModelChange event with
|
|
97
|
-
reactive form directives has been deprecated in Angular v6 and will be removed
|
|
98
|
-
in a future version of Angular.
|
|
99
|
-
|
|
100
|
-
For more information on this, see our API docs here:
|
|
101
|
-
https://angular.io/api/forms/${directiveName === 'formControl' ? 'FormControlDirective' : 'FormControlName'}#use-with-ngmodel
|
|
102
|
-
`;
|
|
103
|
-
}
|
|
104
|
-
function describeKey(isFormGroup, key) {
|
|
105
|
-
return isFormGroup ? `with name: '${key}'` : `at index: ${key}`;
|
|
106
|
-
}
|
|
107
|
-
export function noControlsError(isFormGroup) {
|
|
108
|
-
return `
|
|
109
|
-
There are no form controls registered with this ${isFormGroup ? 'group' : 'array'} yet. If you're using ngModel,
|
|
110
|
-
you may want to check next tick (e.g. use setTimeout).
|
|
111
|
-
`;
|
|
112
|
-
}
|
|
113
|
-
export function missingControlError(isFormGroup, key) {
|
|
114
|
-
return `Cannot find form control ${describeKey(isFormGroup, key)}`;
|
|
115
|
-
}
|
|
116
|
-
export function missingControlValueError(isFormGroup, key) {
|
|
117
|
-
return `Must supply a value for form control ${describeKey(isFormGroup, key)}`;
|
|
118
|
-
}
|
|
119
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"reactive_errors.js","sourceRoot":"","sources":["../../../../../../../packages/forms/src/directives/reactive_errors.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAAC,aAAa,IAAI,YAAY,EAAC,MAAM,eAAe,CAAC;AAI5D,OAAO,EACL,oBAAoB,EACpB,sBAAsB,EACtB,oBAAoB,EACpB,mBAAmB,GACpB,MAAM,kBAAkB,CAAC;AAE1B,MAAM,UAAU,sBAAsB,CAAC,WAAmC;IACxE,OAAO,IAAI,YAAY,+DAErB;;;QAGI,mBAAmB,CAAC,WAAW,CAAC;;;;MAIlC,sBAAsB,EAAE,CAC3B,CAAC;AACJ,CAAC;AAED,SAAS,mBAAmB,CAAC,WAAmC;IAC9D,IAAI,WAAW,IAAI,IAAI,IAAI,WAAW,KAAK,EAAE,EAAE,CAAC;QAC9C,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,MAAM,SAAS,GAAG,OAAO,WAAW,KAAK,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,OAAO,CAAC;IAErE,OAAO,yBAAyB,SAAS,MAAM,WAAW,GAAG,CAAC;AAChE,CAAC;AAED,MAAM,UAAU,qBAAqB;IACnC,OAAO,IAAI,YAAY,mEAErB;;;;;QAKI,oBAAoB;;;;QAIpB,mBAAmB,EAAE,CAC1B,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,oBAAoB;IAClC,OAAO,IAAI,YAAY,0DAErB;;;;QAII,sBAAsB,EAAE,CAC7B,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,oBAAoB;IAClC,OAAO,IAAI,YAAY,6DAErB;;;;;MAKE,oBAAoB,EAAE,CACzB,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,oBAAoB;IAClC,OAAO,IAAI,YAAY,6DAErB;;;;;QAKI,oBAAoB,EAAE,CAC3B,CAAC;AACJ,CAAC;AAED,MAAM,CAAC,MAAM,mBAAmB,GAAG;;;;;;;;;;;;;;;CAelC,CAAC;AAEF,MAAM,CAAC,MAAM,qCAAqC,GAAG;;;;;;;;;;;;;;CAcpD,CAAC;AAEF,MAAM,UAAU,cAAc,CAAC,aAAqB;IAClD,OAAO;iEACwD,aAAa;;;;;;iCAO1E,aAAa,KAAK,aAAa,CAAC,CAAC,CAAC,sBAAsB,CAAC,CAAC,CAAC,iBAC7D;GACC,CAAC;AACJ,CAAC;AAED,SAAS,WAAW,CAAC,WAAoB,EAAE,GAAoB;IAC7D,OAAO,WAAW,CAAC,CAAC,CAAC,eAAe,GAAG,GAAG,CAAC,CAAC,CAAC,aAAa,GAAG,EAAE,CAAC;AAClE,CAAC;AAED,MAAM,UAAU,eAAe,CAAC,WAAoB;IAClD,OAAO;sDAEH,WAAW,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,OAC1B;;GAED,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,mBAAmB,CAAC,WAAoB,EAAE,GAAoB;IAC5E,OAAO,4BAA4B,WAAW,CAAC,WAAW,EAAE,GAAG,CAAC,EAAE,CAAC;AACrE,CAAC;AAED,MAAM,UAAU,wBAAwB,CAAC,WAAoB,EAAE,GAAoB;IACjF,OAAO,wCAAwC,WAAW,CAAC,WAAW,EAAE,GAAG,CAAC,EAAE,CAAC;AACjF,CAAC","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 {ɵRuntimeError as RuntimeError} from '@angular/core';\n\nimport {RuntimeErrorCode} from '../errors';\n\nimport {\n  formArrayNameExample,\n  formControlNameExample,\n  formGroupNameExample,\n  ngModelGroupExample,\n} from './error_examples';\n\nexport function controlParentException(nameOrIndex: string | number | null): Error {\n  return new RuntimeError(\n    RuntimeErrorCode.FORM_CONTROL_NAME_MISSING_PARENT,\n    `formControlName must be used with a parent formGroup directive. You'll want to add a formGroup\n      directive and pass it an existing FormGroup instance (you can create one in your class).\n\n      ${describeFormControl(nameOrIndex)}\n\n    Example:\n\n    ${formControlNameExample}`,\n  );\n}\n\nfunction describeFormControl(nameOrIndex: string | number | null): string {\n  if (nameOrIndex == null || nameOrIndex === '') {\n    return '';\n  }\n\n  const valueType = typeof nameOrIndex === 'string' ? 'name' : 'index';\n\n  return `Affected Form Control ${valueType}: \"${nameOrIndex}\"`;\n}\n\nexport function ngModelGroupException(): Error {\n  return new RuntimeError(\n    RuntimeErrorCode.FORM_CONTROL_NAME_INSIDE_MODEL_GROUP,\n    `formControlName cannot be used with an ngModelGroup parent. It is only compatible with parents\n      that also have a \"form\" prefix: formGroupName, formArrayName, or formGroup.\n\n      Option 1:  Update the parent to be formGroupName (reactive form strategy)\n\n      ${formGroupNameExample}\n\n      Option 2: Use ngModel instead of formControlName (template-driven strategy)\n\n      ${ngModelGroupExample}`,\n  );\n}\n\nexport function missingFormException(): Error {\n  return new RuntimeError(\n    RuntimeErrorCode.FORM_GROUP_MISSING_INSTANCE,\n    `formGroup expects a FormGroup instance. Please pass one in.\n\n      Example:\n\n      ${formControlNameExample}`,\n  );\n}\n\nexport function groupParentException(): Error {\n  return new RuntimeError(\n    RuntimeErrorCode.FORM_GROUP_NAME_MISSING_PARENT,\n    `formGroupName must be used with a parent formGroup directive.  You'll want to add a formGroup\n    directive and pass it an existing FormGroup instance (you can create one in your class).\n\n    Example:\n\n    ${formGroupNameExample}`,\n  );\n}\n\nexport function arrayParentException(): Error {\n  return new RuntimeError(\n    RuntimeErrorCode.FORM_ARRAY_NAME_MISSING_PARENT,\n    `formArrayName must be used with a parent formGroup directive.  You'll want to add a formGroup\n      directive and pass it an existing FormGroup instance (you can create one in your class).\n\n      Example:\n\n      ${formArrayNameExample}`,\n  );\n}\n\nexport const disabledAttrWarning = `\n  It looks like you're using the disabled attribute with a reactive form directive. If you set disabled to true\n  when you set up this control in your component class, the disabled attribute will actually be set in the DOM for\n  you. We recommend using this approach to avoid 'changed after checked' errors.\n\n  Example:\n  // Specify the \\`disabled\\` property at control creation time:\n  form = new FormGroup({\n    first: new FormControl({value: 'Nancy', disabled: true}, Validators.required),\n    last: new FormControl('Drew', Validators.required)\n  });\n\n  // Controls can also be enabled/disabled after creation:\n  form.get('first')?.enable();\n  form.get('last')?.disable();\n`;\n\nexport const asyncValidatorsDroppedWithOptsWarning = `\n  It looks like you're constructing using a FormControl with both an options argument and an\n  async validators argument. Mixing these arguments will cause your async validators to be dropped.\n  You should either put all your validators in the options object, or in separate validators\n  arguments. For example:\n\n  // Using validators arguments\n  fc = new FormControl(42, Validators.required, myAsyncValidator);\n\n  // Using AbstractControlOptions\n  fc = new FormControl(42, {validators: Validators.required, asyncValidators: myAV});\n\n  // Do NOT mix them: async validators will be dropped!\n  fc = new FormControl(42, {validators: Validators.required}, /* Oops! */ myAsyncValidator);\n`;\n\nexport function ngModelWarning(directiveName: string): string {\n  return `\n  It looks like you're using ngModel on the same form field as ${directiveName}.\n  Support for using the ngModel input property and ngModelChange event with\n  reactive form directives has been deprecated in Angular v6 and will be removed\n  in a future version of Angular.\n\n  For more information on this, see our API docs here:\n  https://angular.io/api/forms/${\n    directiveName === 'formControl' ? 'FormControlDirective' : 'FormControlName'\n  }#use-with-ngmodel\n  `;\n}\n\nfunction describeKey(isFormGroup: boolean, key: string | number): string {\n  return isFormGroup ? `with name: '${key}'` : `at index: ${key}`;\n}\n\nexport function noControlsError(isFormGroup: boolean): string {\n  return `\n    There are no form controls registered with this ${\n      isFormGroup ? 'group' : 'array'\n    } yet. If you're using ngModel,\n    you may want to check next tick (e.g. use setTimeout).\n  `;\n}\n\nexport function missingControlError(isFormGroup: boolean, key: string | number): string {\n  return `Cannot find form control ${describeKey(isFormGroup, key)}`;\n}\n\nexport function missingControlValueError(isFormGroup: boolean, key: string | number): string {\n  return `Must supply a value for form control ${describeKey(isFormGroup, key)}`;\n}\n"]}
|
|
@@ -1,220 +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 { Directive, ElementRef, forwardRef, Host, Input, Optional, Renderer2, ɵRuntimeError as RuntimeError, } from '@angular/core';
|
|
9
|
-
import { BuiltInControlValueAccessor, NG_VALUE_ACCESSOR, } from './control_value_accessor';
|
|
10
|
-
import * as i0 from "@angular/core";
|
|
11
|
-
const SELECT_VALUE_ACCESSOR = {
|
|
12
|
-
provide: NG_VALUE_ACCESSOR,
|
|
13
|
-
useExisting: forwardRef(() => SelectControlValueAccessor),
|
|
14
|
-
multi: true,
|
|
15
|
-
};
|
|
16
|
-
function _buildValueString(id, value) {
|
|
17
|
-
if (id == null)
|
|
18
|
-
return `${value}`;
|
|
19
|
-
if (value && typeof value === 'object')
|
|
20
|
-
value = 'Object';
|
|
21
|
-
return `${id}: ${value}`.slice(0, 50);
|
|
22
|
-
}
|
|
23
|
-
function _extractId(valueString) {
|
|
24
|
-
return valueString.split(':')[0];
|
|
25
|
-
}
|
|
26
|
-
/**
|
|
27
|
-
* @description
|
|
28
|
-
* The `ControlValueAccessor` for writing select control values and listening to select control
|
|
29
|
-
* changes. The value accessor is used by the `FormControlDirective`, `FormControlName`, and
|
|
30
|
-
* `NgModel` directives.
|
|
31
|
-
*
|
|
32
|
-
* @usageNotes
|
|
33
|
-
*
|
|
34
|
-
* ### Using select controls in a reactive form
|
|
35
|
-
*
|
|
36
|
-
* The following examples show how to use a select control in a reactive form.
|
|
37
|
-
*
|
|
38
|
-
* {@example forms/ts/reactiveSelectControl/reactive_select_control_example.ts region='Component'}
|
|
39
|
-
*
|
|
40
|
-
* ### Using select controls in a template-driven form
|
|
41
|
-
*
|
|
42
|
-
* To use a select in a template-driven form, simply add an `ngModel` and a `name`
|
|
43
|
-
* attribute to the main `<select>` tag.
|
|
44
|
-
*
|
|
45
|
-
* {@example forms/ts/selectControl/select_control_example.ts region='Component'}
|
|
46
|
-
*
|
|
47
|
-
* ### Customizing option selection
|
|
48
|
-
*
|
|
49
|
-
* Angular uses object identity to select option. It's possible for the identities of items
|
|
50
|
-
* to change while the data does not. This can happen, for example, if the items are produced
|
|
51
|
-
* from an RPC to the server, and that RPC is re-run. Even if the data hasn't changed, the
|
|
52
|
-
* second response will produce objects with different identities.
|
|
53
|
-
*
|
|
54
|
-
* To customize the default option comparison algorithm, `<select>` supports `compareWith` input.
|
|
55
|
-
* `compareWith` takes a **function** which has two arguments: `option1` and `option2`.
|
|
56
|
-
* If `compareWith` is given, Angular selects option by the return value of the function.
|
|
57
|
-
*
|
|
58
|
-
* ```ts
|
|
59
|
-
* const selectedCountriesControl = new FormControl();
|
|
60
|
-
* ```
|
|
61
|
-
*
|
|
62
|
-
* ```
|
|
63
|
-
* <select [compareWith]="compareFn" [formControl]="selectedCountriesControl">
|
|
64
|
-
* <option *ngFor="let country of countries" [ngValue]="country">
|
|
65
|
-
* {{country.name}}
|
|
66
|
-
* </option>
|
|
67
|
-
* </select>
|
|
68
|
-
*
|
|
69
|
-
* compareFn(c1: Country, c2: Country): boolean {
|
|
70
|
-
* return c1 && c2 ? c1.id === c2.id : c1 === c2;
|
|
71
|
-
* }
|
|
72
|
-
* ```
|
|
73
|
-
*
|
|
74
|
-
* **Note:** We listen to the 'change' event because 'input' events aren't fired
|
|
75
|
-
* for selects in IE, see:
|
|
76
|
-
* https://developer.mozilla.org/en-US/docs/Web/API/HTMLElement/input_event#browser_compatibility
|
|
77
|
-
*
|
|
78
|
-
* @ngModule ReactiveFormsModule
|
|
79
|
-
* @ngModule FormsModule
|
|
80
|
-
* @publicApi
|
|
81
|
-
*/
|
|
82
|
-
export class SelectControlValueAccessor extends BuiltInControlValueAccessor {
|
|
83
|
-
constructor() {
|
|
84
|
-
super(...arguments);
|
|
85
|
-
/** @internal */
|
|
86
|
-
this._optionMap = new Map();
|
|
87
|
-
/** @internal */
|
|
88
|
-
this._idCounter = 0;
|
|
89
|
-
this._compareWith = Object.is;
|
|
90
|
-
}
|
|
91
|
-
/**
|
|
92
|
-
* @description
|
|
93
|
-
* Tracks the option comparison algorithm for tracking identities when
|
|
94
|
-
* checking for changes.
|
|
95
|
-
*/
|
|
96
|
-
set compareWith(fn) {
|
|
97
|
-
if (typeof fn !== 'function' && (typeof ngDevMode === 'undefined' || ngDevMode)) {
|
|
98
|
-
throw new RuntimeError(1201 /* RuntimeErrorCode.COMPAREWITH_NOT_A_FN */, `compareWith must be a function, but received ${JSON.stringify(fn)}`);
|
|
99
|
-
}
|
|
100
|
-
this._compareWith = fn;
|
|
101
|
-
}
|
|
102
|
-
/**
|
|
103
|
-
* Sets the "value" property on the select element.
|
|
104
|
-
* @nodoc
|
|
105
|
-
*/
|
|
106
|
-
writeValue(value) {
|
|
107
|
-
this.value = value;
|
|
108
|
-
const id = this._getOptionId(value);
|
|
109
|
-
const valueString = _buildValueString(id, value);
|
|
110
|
-
this.setProperty('value', valueString);
|
|
111
|
-
}
|
|
112
|
-
/**
|
|
113
|
-
* Registers a function called when the control value changes.
|
|
114
|
-
* @nodoc
|
|
115
|
-
*/
|
|
116
|
-
registerOnChange(fn) {
|
|
117
|
-
this.onChange = (valueString) => {
|
|
118
|
-
this.value = this._getOptionValue(valueString);
|
|
119
|
-
fn(this.value);
|
|
120
|
-
};
|
|
121
|
-
}
|
|
122
|
-
/** @internal */
|
|
123
|
-
_registerOption() {
|
|
124
|
-
return (this._idCounter++).toString();
|
|
125
|
-
}
|
|
126
|
-
/** @internal */
|
|
127
|
-
_getOptionId(value) {
|
|
128
|
-
for (const id of this._optionMap.keys()) {
|
|
129
|
-
if (this._compareWith(this._optionMap.get(id), value))
|
|
130
|
-
return id;
|
|
131
|
-
}
|
|
132
|
-
return null;
|
|
133
|
-
}
|
|
134
|
-
/** @internal */
|
|
135
|
-
_getOptionValue(valueString) {
|
|
136
|
-
const id = _extractId(valueString);
|
|
137
|
-
return this._optionMap.has(id) ? this._optionMap.get(id) : valueString;
|
|
138
|
-
}
|
|
139
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.0.0-next.1", ngImport: i0, type: SelectControlValueAccessor, deps: null, target: i0.ɵɵFactoryTarget.Directive }); }
|
|
140
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "19.0.0-next.1", type: SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: { compareWith: "compareWith" }, host: { listeners: { "change": "onChange($event.target.value)", "blur": "onTouched()" } }, providers: [SELECT_VALUE_ACCESSOR], usesInheritance: true, ngImport: i0 }); }
|
|
141
|
-
}
|
|
142
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.0.0-next.1", ngImport: i0, type: SelectControlValueAccessor, decorators: [{
|
|
143
|
-
type: Directive,
|
|
144
|
-
args: [{
|
|
145
|
-
selector: 'select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]',
|
|
146
|
-
host: { '(change)': 'onChange($event.target.value)', '(blur)': 'onTouched()' },
|
|
147
|
-
providers: [SELECT_VALUE_ACCESSOR],
|
|
148
|
-
}]
|
|
149
|
-
}], propDecorators: { compareWith: [{
|
|
150
|
-
type: Input
|
|
151
|
-
}] } });
|
|
152
|
-
/**
|
|
153
|
-
* @description
|
|
154
|
-
* Marks `<option>` as dynamic, so Angular can be notified when options change.
|
|
155
|
-
*
|
|
156
|
-
* @see {@link SelectControlValueAccessor}
|
|
157
|
-
*
|
|
158
|
-
* @ngModule ReactiveFormsModule
|
|
159
|
-
* @ngModule FormsModule
|
|
160
|
-
* @publicApi
|
|
161
|
-
*/
|
|
162
|
-
export class NgSelectOption {
|
|
163
|
-
constructor(_element, _renderer, _select) {
|
|
164
|
-
this._element = _element;
|
|
165
|
-
this._renderer = _renderer;
|
|
166
|
-
this._select = _select;
|
|
167
|
-
if (this._select)
|
|
168
|
-
this.id = this._select._registerOption();
|
|
169
|
-
}
|
|
170
|
-
/**
|
|
171
|
-
* @description
|
|
172
|
-
* Tracks the value bound to the option element. Unlike the value binding,
|
|
173
|
-
* ngValue supports binding to objects.
|
|
174
|
-
*/
|
|
175
|
-
set ngValue(value) {
|
|
176
|
-
if (this._select == null)
|
|
177
|
-
return;
|
|
178
|
-
this._select._optionMap.set(this.id, value);
|
|
179
|
-
this._setElementValue(_buildValueString(this.id, value));
|
|
180
|
-
this._select.writeValue(this._select.value);
|
|
181
|
-
}
|
|
182
|
-
/**
|
|
183
|
-
* @description
|
|
184
|
-
* Tracks simple string values bound to the option element.
|
|
185
|
-
* For objects, use the `ngValue` input binding.
|
|
186
|
-
*/
|
|
187
|
-
set value(value) {
|
|
188
|
-
this._setElementValue(value);
|
|
189
|
-
if (this._select)
|
|
190
|
-
this._select.writeValue(this._select.value);
|
|
191
|
-
}
|
|
192
|
-
/** @internal */
|
|
193
|
-
_setElementValue(value) {
|
|
194
|
-
this._renderer.setProperty(this._element.nativeElement, 'value', value);
|
|
195
|
-
}
|
|
196
|
-
/** @nodoc */
|
|
197
|
-
ngOnDestroy() {
|
|
198
|
-
if (this._select) {
|
|
199
|
-
this._select._optionMap.delete(this.id);
|
|
200
|
-
this._select.writeValue(this._select.value);
|
|
201
|
-
}
|
|
202
|
-
}
|
|
203
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.0.0-next.1", ngImport: i0, type: NgSelectOption, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }, { token: SelectControlValueAccessor, host: true, optional: true }], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
204
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "19.0.0-next.1", type: NgSelectOption, selector: "option", inputs: { ngValue: "ngValue", value: "value" }, ngImport: i0 }); }
|
|
205
|
-
}
|
|
206
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.0.0-next.1", ngImport: i0, type: NgSelectOption, decorators: [{
|
|
207
|
-
type: Directive,
|
|
208
|
-
args: [{ selector: 'option' }]
|
|
209
|
-
}], ctorParameters: () => [{ type: i0.ElementRef }, { type: i0.Renderer2 }, { type: SelectControlValueAccessor, decorators: [{
|
|
210
|
-
type: Optional
|
|
211
|
-
}, {
|
|
212
|
-
type: Host
|
|
213
|
-
}] }], propDecorators: { ngValue: [{
|
|
214
|
-
type: Input,
|
|
215
|
-
args: ['ngValue']
|
|
216
|
-
}], value: [{
|
|
217
|
-
type: Input,
|
|
218
|
-
args: ['value']
|
|
219
|
-
}] } });
|
|
220
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select_control_value_accessor.js","sourceRoot":"","sources":["../../../../../../../packages/forms/src/directives/select_control_value_accessor.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  Directive,\n  ElementRef,\n  forwardRef,\n  Host,\n  Input,\n  OnDestroy,\n  Optional,\n  Provider,\n  Renderer2,\n  ɵRuntimeError as RuntimeError,\n} from '@angular/core';\n\nimport {RuntimeErrorCode} from '../errors';\n\nimport {\n  BuiltInControlValueAccessor,\n  ControlValueAccessor,\n  NG_VALUE_ACCESSOR,\n} from './control_value_accessor';\n\nconst SELECT_VALUE_ACCESSOR: Provider = {\n  provide: NG_VALUE_ACCESSOR,\n  useExisting: forwardRef(() => SelectControlValueAccessor),\n  multi: true,\n};\n\nfunction _buildValueString(id: string | null, value: any): string {\n  if (id == null) return `${value}`;\n  if (value && typeof value === 'object') value = 'Object';\n  return `${id}: ${value}`.slice(0, 50);\n}\n\nfunction _extractId(valueString: string): string {\n  return valueString.split(':')[0];\n}\n\n/**\n * @description\n * The `ControlValueAccessor` for writing select control values and listening to select control\n * changes. The value accessor is used by the `FormControlDirective`, `FormControlName`, and\n * `NgModel` directives.\n *\n * @usageNotes\n *\n * ### Using select controls in a reactive form\n *\n * The following examples show how to use a select control in a reactive form.\n *\n * {@example forms/ts/reactiveSelectControl/reactive_select_control_example.ts region='Component'}\n *\n * ### Using select controls in a template-driven form\n *\n * To use a select in a template-driven form, simply add an `ngModel` and a `name`\n * attribute to the main `<select>` tag.\n *\n * {@example forms/ts/selectControl/select_control_example.ts region='Component'}\n *\n * ### Customizing option selection\n *\n * Angular uses object identity to select option. It's possible for the identities of items\n * to change while the data does not. This can happen, for example, if the items are produced\n * from an RPC to the server, and that RPC is re-run. Even if the data hasn't changed, the\n * second response will produce objects with different identities.\n *\n * To customize the default option comparison algorithm, `<select>` supports `compareWith` input.\n * `compareWith` takes a **function** which has two arguments: `option1` and `option2`.\n * If `compareWith` is given, Angular selects option by the return value of the function.\n *\n * ```ts\n * const selectedCountriesControl = new FormControl();\n * ```\n *\n * ```\n * <select [compareWith]=\"compareFn\"  [formControl]=\"selectedCountriesControl\">\n *     <option *ngFor=\"let country of countries\" [ngValue]=\"country\">\n *         {{country.name}}\n *     </option>\n * </select>\n *\n * compareFn(c1: Country, c2: Country): boolean {\n *     return c1 && c2 ? c1.id === c2.id : c1 === c2;\n * }\n * ```\n *\n * **Note:** We listen to the 'change' event because 'input' events aren't fired\n * for selects in IE, see:\n * https://developer.mozilla.org/en-US/docs/Web/API/HTMLElement/input_event#browser_compatibility\n *\n * @ngModule ReactiveFormsModule\n * @ngModule FormsModule\n * @publicApi\n */\n@Directive({\n  selector:\n    'select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]',\n  host: {'(change)': 'onChange($event.target.value)', '(blur)': 'onTouched()'},\n  providers: [SELECT_VALUE_ACCESSOR],\n})\nexport class SelectControlValueAccessor\n  extends BuiltInControlValueAccessor\n  implements ControlValueAccessor\n{\n  /** @nodoc */\n  value: any;\n\n  /** @internal */\n  _optionMap: Map<string, any> = new Map<string, any>();\n\n  /** @internal */\n  _idCounter: number = 0;\n\n  /**\n   * @description\n   * Tracks the option comparison algorithm for tracking identities when\n   * checking for changes.\n   */\n  @Input()\n  set compareWith(fn: (o1: any, o2: any) => boolean) {\n    if (typeof fn !== 'function' && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n      throw new RuntimeError(\n        RuntimeErrorCode.COMPAREWITH_NOT_A_FN,\n        `compareWith must be a function, but received ${JSON.stringify(fn)}`,\n      );\n    }\n    this._compareWith = fn;\n  }\n\n  private _compareWith: (o1: any, o2: any) => boolean = Object.is;\n\n  /**\n   * Sets the \"value\" property on the select element.\n   * @nodoc\n   */\n  writeValue(value: any): void {\n    this.value = value;\n    const id: string | null = this._getOptionId(value);\n    const valueString = _buildValueString(id, value);\n    this.setProperty('value', valueString);\n  }\n\n  /**\n   * Registers a function called when the control value changes.\n   * @nodoc\n   */\n  override registerOnChange(fn: (value: any) => any): void {\n    this.onChange = (valueString: string) => {\n      this.value = this._getOptionValue(valueString);\n      fn(this.value);\n    };\n  }\n\n  /** @internal */\n  _registerOption(): string {\n    return (this._idCounter++).toString();\n  }\n\n  /** @internal */\n  _getOptionId(value: any): string | null {\n    for (const id of this._optionMap.keys()) {\n      if (this._compareWith(this._optionMap.get(id), value)) return id;\n    }\n    return null;\n  }\n\n  /** @internal */\n  _getOptionValue(valueString: string): any {\n    const id: string = _extractId(valueString);\n    return this._optionMap.has(id) ? this._optionMap.get(id) : valueString;\n  }\n}\n\n/**\n * @description\n * Marks `<option>` as dynamic, so Angular can be notified when options change.\n *\n * @see {@link SelectControlValueAccessor}\n *\n * @ngModule ReactiveFormsModule\n * @ngModule FormsModule\n * @publicApi\n */\n@Directive({selector: 'option'})\nexport class NgSelectOption implements OnDestroy {\n  /**\n   * @description\n   * ID of the option element\n   */\n  // TODO(issue/24571): remove '!'.\n  id!: string;\n\n  constructor(\n    private _element: ElementRef,\n    private _renderer: Renderer2,\n    @Optional() @Host() private _select: SelectControlValueAccessor,\n  ) {\n    if (this._select) this.id = this._select._registerOption();\n  }\n\n  /**\n   * @description\n   * Tracks the value bound to the option element. Unlike the value binding,\n   * ngValue supports binding to objects.\n   */\n  @Input('ngValue')\n  set ngValue(value: any) {\n    if (this._select == null) return;\n    this._select._optionMap.set(this.id, value);\n    this._setElementValue(_buildValueString(this.id, value));\n    this._select.writeValue(this._select.value);\n  }\n\n  /**\n   * @description\n   * Tracks simple string values bound to the option element.\n   * For objects, use the `ngValue` input binding.\n   */\n  @Input('value')\n  set value(value: any) {\n    this._setElementValue(value);\n    if (this._select) this._select.writeValue(this._select.value);\n  }\n\n  /** @internal */\n  _setElementValue(value: string): void {\n    this._renderer.setProperty(this._element.nativeElement, 'value', value);\n  }\n\n  /** @nodoc */\n  ngOnDestroy(): void {\n    if (this._select) {\n      this._select._optionMap.delete(this.id);\n      this._select.writeValue(this._select.value);\n    }\n  }\n}\n"]}
|