feat(forms): add updateOn option to FormBuilder (#24599)
PR Close #24599
This commit is contained in:
parent
aed95fd8c7
commit
e9e804fb02
|
@ -22,7 +22,7 @@ import {FormBuilder, FormControl, FormGroup, Validators} from '@angular/forms';
|
||||||
<input formControlName="email" placeholder="Email">
|
<input formControlName="email" placeholder="Email">
|
||||||
<button>Submit</button>
|
<button>Submit</button>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
<p>Value: {{ form.value | json }}</p>
|
<p>Value: {{ form.value | json }}</p>
|
||||||
<p>Validation status: {{ form.status }}</p>
|
<p>Validation status: {{ form.status }}</p>
|
||||||
`
|
`
|
||||||
|
@ -31,13 +31,15 @@ export class FormBuilderComp {
|
||||||
form: FormGroup;
|
form: FormGroup;
|
||||||
|
|
||||||
constructor(@Inject(FormBuilder) fb: FormBuilder) {
|
constructor(@Inject(FormBuilder) fb: FormBuilder) {
|
||||||
this.form = fb.group({
|
this.form = fb.group(
|
||||||
name: fb.group({
|
{
|
||||||
first: ['Nancy', Validators.minLength(2)],
|
name: fb.group({
|
||||||
last: 'Drew',
|
first: ['Nancy', Validators.minLength(2)],
|
||||||
}),
|
last: 'Drew',
|
||||||
email: '',
|
}),
|
||||||
});
|
email: '',
|
||||||
|
},
|
||||||
|
{updateOn: 'change'});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// #enddocregion
|
// #enddocregion
|
||||||
|
@ -56,4 +58,4 @@ export class DisabledFormControlComponent {
|
||||||
this.control = fb.control({value: 'my val', disabled: true});
|
this.control = fb.control({value: 'my val', disabled: true});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// #enddocregion disabled-control
|
// #enddocregion disabled-control
|
||||||
|
|
|
@ -9,7 +9,7 @@
|
||||||
import {Injectable} from '@angular/core';
|
import {Injectable} from '@angular/core';
|
||||||
|
|
||||||
import {AsyncValidatorFn, ValidatorFn} from './directives/validators';
|
import {AsyncValidatorFn, ValidatorFn} from './directives/validators';
|
||||||
import {AbstractControl, FormArray, FormControl, FormGroup} from './model';
|
import {AbstractControl, AbstractControlOptions, FormArray, FormControl, FormGroup, FormHooks} from './model';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @description
|
* @description
|
||||||
|
@ -37,25 +37,43 @@ export class FormBuilder {
|
||||||
* * `asyncValidator`: A single async validator or array of async validator functions
|
* * `asyncValidator`: A single async validator or array of async validator functions
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
group(controlsConfig: {[key: string]: any}, extra: {[key: string]: any}|null = null): FormGroup {
|
group(controlsConfig: {[key: string]: any}, legacyOrOpts: {[key: string]: any}|null = null):
|
||||||
|
FormGroup {
|
||||||
const controls = this._reduceControls(controlsConfig);
|
const controls = this._reduceControls(controlsConfig);
|
||||||
const validator: ValidatorFn = extra != null ? extra['validator'] : null;
|
|
||||||
const asyncValidator: AsyncValidatorFn = extra != null ? extra['asyncValidator'] : null;
|
let validators: ValidatorFn|ValidatorFn[]|null = null;
|
||||||
return new FormGroup(controls, validator, asyncValidator);
|
let asyncValidators: AsyncValidatorFn|AsyncValidatorFn[]|null = null;
|
||||||
|
let updateOn: FormHooks|undefined = undefined;
|
||||||
|
|
||||||
|
if (legacyOrOpts != null &&
|
||||||
|
(legacyOrOpts.asyncValidator !== undefined || legacyOrOpts.validator !== undefined)) {
|
||||||
|
// `legacyOrOpts` are legacy form group options
|
||||||
|
validators = legacyOrOpts.validator != null ? legacyOrOpts.validator : null;
|
||||||
|
asyncValidators = legacyOrOpts.asyncValidator != null ? legacyOrOpts.asyncValidator : null;
|
||||||
|
} else if (legacyOrOpts != null) {
|
||||||
|
// `legacyOrOpts` are `AbstractControlOptions`
|
||||||
|
validators = legacyOrOpts.validators != null ? legacyOrOpts.validators : null;
|
||||||
|
asyncValidators = legacyOrOpts.asyncValidators != null ? legacyOrOpts.asyncValidators : null;
|
||||||
|
updateOn = legacyOrOpts.updateOn != null ? legacyOrOpts.updateOn : undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new FormGroup(controls, {asyncValidators, updateOn, validators});
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @description
|
* @description
|
||||||
* Construct a new `FormControl` instance.
|
* Construct a new `FormControl` with the given state, validators and options.
|
||||||
*
|
*
|
||||||
* @param formState Initializes the control with an initial value,
|
* @param formState Initializes the control with an initial state value, or
|
||||||
* or an object that defines the initial value and disabled state.
|
* with an object that contains both a value and a disabled status.
|
||||||
*
|
*
|
||||||
* @param validator A synchronous validator function, or an array of synchronous validator
|
* @param validatorOrOpts A synchronous validator function, or an array of
|
||||||
|
* such functions, or an `AbstractControlOptions` object that contains
|
||||||
|
* validation functions and a validation trigger.
|
||||||
|
*
|
||||||
|
* @param asyncValidator A single async validator or array of async validator
|
||||||
* functions.
|
* functions.
|
||||||
*
|
*
|
||||||
* @param asyncValidator A single async validator or array of async validator functions
|
|
||||||
*
|
|
||||||
* @usageNotes
|
* @usageNotes
|
||||||
*
|
*
|
||||||
* ### Initialize a control as disabled
|
* ### Initialize a control as disabled
|
||||||
|
@ -65,31 +83,33 @@ export class FormBuilder {
|
||||||
* <code-example path="forms/ts/formBuilder/form_builder_example.ts"
|
* <code-example path="forms/ts/formBuilder/form_builder_example.ts"
|
||||||
* linenums="false" region="disabled-control">
|
* linenums="false" region="disabled-control">
|
||||||
* </code-example>
|
* </code-example>
|
||||||
*
|
|
||||||
*/
|
*/
|
||||||
control(
|
control(
|
||||||
formState: any, validator?: ValidatorFn|ValidatorFn[]|null,
|
formState: any, validatorOrOpts?: ValidatorFn|ValidatorFn[]|AbstractControlOptions|null,
|
||||||
asyncValidator?: AsyncValidatorFn|AsyncValidatorFn[]|null): FormControl {
|
asyncValidator?: AsyncValidatorFn|AsyncValidatorFn[]|null): FormControl {
|
||||||
return new FormControl(formState, validator, asyncValidator);
|
return new FormControl(formState, validatorOrOpts, asyncValidator);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @description
|
* Constructs a new `FormArray` from the given array of configurations,
|
||||||
* Construct a new `FormArray` instance.
|
* validators and options.
|
||||||
*
|
*
|
||||||
* @param controlsConfig An array of child controls. The key for each child control is its index
|
* @param controlsConfig An array of child controls or control configs. Each
|
||||||
* in the array.
|
* child control is given an index when it is registered.
|
||||||
*
|
*
|
||||||
* @param validator A synchronous validator function, or an array of synchronous validator
|
* @param validatorOrOpts A synchronous validator function, or an array of
|
||||||
|
* such functions, or an `AbstractControlOptions` object that contains
|
||||||
|
* validation functions and a validation trigger.
|
||||||
|
*
|
||||||
|
* @param asyncValidator A single async validator or array of async validator
|
||||||
* functions.
|
* functions.
|
||||||
*
|
|
||||||
* @param asyncValidator A single async validator or array of async validator functions
|
|
||||||
*/
|
*/
|
||||||
array(
|
array(
|
||||||
controlsConfig: any[], validator?: ValidatorFn|ValidatorFn[]|null,
|
controlsConfig: any[],
|
||||||
|
validatorOrOpts?: ValidatorFn|ValidatorFn[]|AbstractControlOptions|null,
|
||||||
asyncValidator?: AsyncValidatorFn|AsyncValidatorFn[]|null): FormArray {
|
asyncValidator?: AsyncValidatorFn|AsyncValidatorFn[]|null): FormArray {
|
||||||
const controls = controlsConfig.map(c => this._createControl(c));
|
const controls = controlsConfig.map(c => this._createControl(c));
|
||||||
return new FormArray(controls, validator, asyncValidator);
|
return new FormArray(controls, validatorOrOpts, asyncValidator);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** @internal */
|
/** @internal */
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
*/
|
*/
|
||||||
import {fakeAsync, tick} from '@angular/core/testing';
|
import {fakeAsync, tick} from '@angular/core/testing';
|
||||||
import {beforeEach, describe, expect, it} from '@angular/core/testing/src/testing_internal';
|
import {beforeEach, describe, expect, it} from '@angular/core/testing/src/testing_internal';
|
||||||
import {FormBuilder} from '@angular/forms';
|
import {FormBuilder, Validators} from '@angular/forms';
|
||||||
import {of } from 'rxjs';
|
import {of } from 'rxjs';
|
||||||
|
|
||||||
(function() {
|
(function() {
|
||||||
|
@ -119,5 +119,84 @@ import {of } from 'rxjs';
|
||||||
expect(a.value).toEqual(['one', 'two']);
|
expect(a.value).toEqual(['one', 'two']);
|
||||||
expect(a.errors).toEqual({'sync1': true, 'sync2': true});
|
expect(a.errors).toEqual({'sync1': true, 'sync2': true});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('updateOn', () => {
|
||||||
|
it('should default to on change', () => {
|
||||||
|
const c = b.control('');
|
||||||
|
expect(c.updateOn).toEqual('change');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should default to on change with an options obj', () => {
|
||||||
|
const c = b.control('', {validators: Validators.required});
|
||||||
|
expect(c.updateOn).toEqual('change');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should set updateOn when updating on blur', () => {
|
||||||
|
const c = b.control('', {updateOn: 'blur'});
|
||||||
|
expect(c.updateOn).toEqual('blur');
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('in groups and arrays', () => {
|
||||||
|
it('should default to group updateOn when not set in control', () => {
|
||||||
|
const g = b.group({one: b.control(''), two: b.control('')}, {updateOn: 'blur'});
|
||||||
|
|
||||||
|
expect(g.get('one') !.updateOn).toEqual('blur');
|
||||||
|
expect(g.get('two') !.updateOn).toEqual('blur');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should default to array updateOn when not set in control', () => {
|
||||||
|
const a = b.array([b.control(''), b.control('')], {updateOn: 'blur'});
|
||||||
|
|
||||||
|
expect(a.get([0]) !.updateOn).toEqual('blur');
|
||||||
|
expect(a.get([1]) !.updateOn).toEqual('blur');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should set updateOn with nested groups', () => {
|
||||||
|
const g = b.group(
|
||||||
|
{
|
||||||
|
group: b.group({one: b.control(''), two: b.control('')}),
|
||||||
|
},
|
||||||
|
{updateOn: 'blur'});
|
||||||
|
|
||||||
|
expect(g.get('group.one') !.updateOn).toEqual('blur');
|
||||||
|
expect(g.get('group.two') !.updateOn).toEqual('blur');
|
||||||
|
expect(g.get('group') !.updateOn).toEqual('blur');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should set updateOn with nested arrays', () => {
|
||||||
|
const g = b.group(
|
||||||
|
{
|
||||||
|
arr: b.array([b.control(''), b.control('')]),
|
||||||
|
},
|
||||||
|
{updateOn: 'blur'});
|
||||||
|
|
||||||
|
expect(g.get(['arr', 0]) !.updateOn).toEqual('blur');
|
||||||
|
expect(g.get(['arr', 1]) !.updateOn).toEqual('blur');
|
||||||
|
expect(g.get('arr') !.updateOn).toEqual('blur');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should allow control updateOn to override group updateOn', () => {
|
||||||
|
const g = b.group(
|
||||||
|
{one: b.control('', {updateOn: 'change'}), two: b.control('')}, {updateOn: 'blur'});
|
||||||
|
|
||||||
|
expect(g.get('one') !.updateOn).toEqual('change');
|
||||||
|
expect(g.get('two') !.updateOn).toEqual('blur');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should set updateOn with complex setup', () => {
|
||||||
|
const g = b.group({
|
||||||
|
group: b.group(
|
||||||
|
{one: b.control('', {updateOn: 'change'}), two: b.control('')}, {updateOn: 'blur'}),
|
||||||
|
groupTwo: b.group({one: b.control('')}, {updateOn: 'submit'}),
|
||||||
|
three: b.control('')
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(g.get('group.one') !.updateOn).toEqual('change');
|
||||||
|
expect(g.get('group.two') !.updateOn).toEqual('blur');
|
||||||
|
expect(g.get('groupTwo.one') !.updateOn).toEqual('submit');
|
||||||
|
expect(g.get('three') !.updateOn).toEqual('change');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
})();
|
})();
|
||||||
|
|
|
@ -201,11 +201,11 @@ export declare class FormArrayName extends ControlContainer implements OnInit, O
|
||||||
}
|
}
|
||||||
|
|
||||||
export declare class FormBuilder {
|
export declare class FormBuilder {
|
||||||
array(controlsConfig: any[], validator?: ValidatorFn | ValidatorFn[] | null, asyncValidator?: AsyncValidatorFn | AsyncValidatorFn[] | null): FormArray;
|
array(controlsConfig: any[], validatorOrOpts?: ValidatorFn | ValidatorFn[] | AbstractControlOptions | null, asyncValidator?: AsyncValidatorFn | AsyncValidatorFn[] | null): FormArray;
|
||||||
control(formState: any, validator?: ValidatorFn | ValidatorFn[] | null, asyncValidator?: AsyncValidatorFn | AsyncValidatorFn[] | null): FormControl;
|
control(formState: any, validatorOrOpts?: ValidatorFn | ValidatorFn[] | AbstractControlOptions | null, asyncValidator?: AsyncValidatorFn | AsyncValidatorFn[] | null): FormControl;
|
||||||
group(controlsConfig: {
|
group(controlsConfig: {
|
||||||
[key: string]: any;
|
[key: string]: any;
|
||||||
}, extra?: {
|
}, legacyOrOpts?: {
|
||||||
[key: string]: any;
|
[key: string]: any;
|
||||||
} | null): FormGroup;
|
} | null): FormGroup;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue