2016-06-23 09:47:54 -07:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2016-10-19 13:42:39 -07:00
|
|
|
import {isBlank, isPresent, looseIdentical} from '../facade/lang';
|
2016-06-25 13:27:29 -07:00
|
|
|
import {FormArray, FormControl, FormGroup} from '../model';
|
2016-06-08 15:36:24 -07:00
|
|
|
import {Validators} from '../validators';
|
2016-06-08 16:38:52 -07:00
|
|
|
|
|
|
|
import {AbstractControlDirective} from './abstract_control_directive';
|
2016-06-12 16:37:42 -07:00
|
|
|
import {AbstractFormGroupDirective} from './abstract_form_group_directive';
|
2016-06-08 16:38:52 -07:00
|
|
|
import {CheckboxControlValueAccessor} from './checkbox_value_accessor';
|
|
|
|
import {ControlContainer} from './control_container';
|
2016-06-08 15:36:24 -07:00
|
|
|
import {ControlValueAccessor} from './control_value_accessor';
|
|
|
|
import {DefaultValueAccessor} from './default_value_accessor';
|
2016-06-08 16:38:52 -07:00
|
|
|
import {NgControl} from './ng_control';
|
|
|
|
import {normalizeAsyncValidator, normalizeValidator} from './normalize_validator';
|
2016-06-08 15:36:24 -07:00
|
|
|
import {NumberValueAccessor} from './number_value_accessor';
|
|
|
|
import {RadioControlValueAccessor} from './radio_control_value_accessor';
|
2016-10-19 20:12:13 +03:00
|
|
|
import {RangeValueAccessor} from './range_value_accessor';
|
2016-08-02 09:40:42 -07:00
|
|
|
import {FormArrayName} from './reactive_directives/form_group_name';
|
2016-06-08 16:38:52 -07:00
|
|
|
import {SelectControlValueAccessor} from './select_control_value_accessor';
|
2016-06-27 00:24:27 +02:00
|
|
|
import {SelectMultipleControlValueAccessor} from './select_multiple_control_value_accessor';
|
2016-08-29 11:33:49 -07:00
|
|
|
import {AsyncValidatorFn, Validator, ValidatorFn} from './validators';
|
2016-06-08 15:36:24 -07:00
|
|
|
|
|
|
|
|
|
|
|
export function controlPath(name: string, parent: ControlContainer): string[] {
|
2016-10-19 13:42:39 -07:00
|
|
|
return [...parent.path, name];
|
2016-06-08 15:36:24 -07:00
|
|
|
}
|
|
|
|
|
2016-06-10 11:15:59 -07:00
|
|
|
export function setUpControl(control: FormControl, dir: NgControl): void {
|
2016-09-30 09:26:53 -07:00
|
|
|
if (!control) _throwError(dir, 'Cannot find control with');
|
|
|
|
if (!dir.valueAccessor) _throwError(dir, 'No value accessor for form control with');
|
2016-06-08 15:36:24 -07:00
|
|
|
|
|
|
|
control.validator = Validators.compose([control.validator, dir.validator]);
|
|
|
|
control.asyncValidator = Validators.composeAsync([control.asyncValidator, dir.asyncValidator]);
|
|
|
|
dir.valueAccessor.writeValue(control.value);
|
|
|
|
|
|
|
|
// view -> model
|
|
|
|
dir.valueAccessor.registerOnChange((newValue: any) => {
|
|
|
|
dir.viewToModelUpdate(newValue);
|
|
|
|
control.markAsDirty();
|
2016-08-05 13:35:17 -07:00
|
|
|
control.setValue(newValue, {emitModelToViewChange: false});
|
2016-06-08 15:36:24 -07:00
|
|
|
});
|
|
|
|
|
2016-08-29 11:33:49 -07:00
|
|
|
// touched
|
|
|
|
dir.valueAccessor.registerOnTouched(() => control.markAsTouched());
|
|
|
|
|
2016-07-12 15:02:25 -07:00
|
|
|
control.registerOnChange((newValue: any, emitModelEvent: boolean) => {
|
|
|
|
// control -> view
|
|
|
|
dir.valueAccessor.writeValue(newValue);
|
|
|
|
|
|
|
|
// control -> ngModel
|
|
|
|
if (emitModelEvent) dir.viewToModelUpdate(newValue);
|
|
|
|
});
|
2016-06-08 15:36:24 -07:00
|
|
|
|
2016-08-24 16:58:43 -07:00
|
|
|
if (dir.valueAccessor.setDisabledState) {
|
|
|
|
control.registerOnDisabledChange(
|
|
|
|
(isDisabled: boolean) => { dir.valueAccessor.setDisabledState(isDisabled); });
|
|
|
|
}
|
|
|
|
|
2016-08-29 11:33:49 -07:00
|
|
|
// re-run validation when validator binding changes, e.g. minlength=3 -> minlength=4
|
|
|
|
dir._rawValidators.forEach((validator: Validator | ValidatorFn) => {
|
2016-09-09 14:09:11 -07:00
|
|
|
if ((<Validator>validator).registerOnValidatorChange)
|
|
|
|
(<Validator>validator).registerOnValidatorChange(() => control.updateValueAndValidity());
|
2016-08-29 11:33:49 -07:00
|
|
|
});
|
|
|
|
|
|
|
|
dir._rawAsyncValidators.forEach((validator: Validator | ValidatorFn) => {
|
2016-09-09 14:09:11 -07:00
|
|
|
if ((<Validator>validator).registerOnValidatorChange)
|
|
|
|
(<Validator>validator).registerOnValidatorChange(() => control.updateValueAndValidity());
|
2016-08-29 11:33:49 -07:00
|
|
|
});
|
2016-06-08 15:36:24 -07:00
|
|
|
}
|
|
|
|
|
2016-08-25 14:37:57 -07:00
|
|
|
export function cleanUpControl(control: FormControl, dir: NgControl) {
|
|
|
|
dir.valueAccessor.registerOnChange(() => _noControlError(dir));
|
|
|
|
dir.valueAccessor.registerOnTouched(() => _noControlError(dir));
|
2016-09-09 14:09:11 -07:00
|
|
|
dir._rawValidators.forEach((validator: Validator) => validator.registerOnValidatorChange(null));
|
|
|
|
dir._rawAsyncValidators.forEach(
|
|
|
|
(validator: Validator) => validator.registerOnValidatorChange(null));
|
2016-08-25 14:37:57 -07:00
|
|
|
if (control) control._clearChangeFns();
|
|
|
|
}
|
|
|
|
|
2016-06-25 13:27:29 -07:00
|
|
|
export function setUpFormContainer(
|
|
|
|
control: FormGroup | FormArray, dir: AbstractFormGroupDirective | FormArrayName) {
|
2016-07-13 14:13:02 -07:00
|
|
|
if (isBlank(control)) _throwError(dir, 'Cannot find control with');
|
2016-06-08 15:36:24 -07:00
|
|
|
control.validator = Validators.compose([control.validator, dir.validator]);
|
|
|
|
control.asyncValidator = Validators.composeAsync([control.asyncValidator, dir.asyncValidator]);
|
|
|
|
}
|
|
|
|
|
2016-08-25 14:37:57 -07:00
|
|
|
function _noControlError(dir: NgControl) {
|
|
|
|
return _throwError(dir, 'There is no FormControl instance attached to form control element with');
|
|
|
|
}
|
|
|
|
|
2016-06-08 15:36:24 -07:00
|
|
|
function _throwError(dir: AbstractControlDirective, message: string): void {
|
2016-07-13 14:13:02 -07:00
|
|
|
let messageEnd: string;
|
|
|
|
if (dir.path.length > 1) {
|
|
|
|
messageEnd = `path: '${dir.path.join(' -> ')}'`;
|
|
|
|
} else if (dir.path[0]) {
|
|
|
|
messageEnd = `name: '${dir.path}'`;
|
|
|
|
} else {
|
|
|
|
messageEnd = 'unspecified name attribute';
|
|
|
|
}
|
2016-08-25 00:50:16 -07:00
|
|
|
throw new Error(`${message} ${messageEnd}`);
|
2016-06-08 15:36:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
export function composeValidators(validators: /* Array<Validator|Function> */ any[]): ValidatorFn {
|
|
|
|
return isPresent(validators) ? Validators.compose(validators.map(normalizeValidator)) : null;
|
|
|
|
}
|
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
export function composeAsyncValidators(validators: /* Array<Validator|Function> */ any[]):
|
|
|
|
AsyncValidatorFn {
|
2016-06-08 15:36:24 -07:00
|
|
|
return isPresent(validators) ? Validators.composeAsync(validators.map(normalizeAsyncValidator)) :
|
|
|
|
null;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function isPropertyUpdated(changes: {[key: string]: any}, viewModel: any): boolean {
|
2016-09-19 17:15:57 -07:00
|
|
|
if (!changes.hasOwnProperty('model')) return false;
|
|
|
|
const change = changes['model'];
|
2016-06-08 15:36:24 -07:00
|
|
|
|
|
|
|
if (change.isFirstChange()) return true;
|
|
|
|
return !looseIdentical(viewModel, change.currentValue);
|
|
|
|
}
|
|
|
|
|
2016-10-19 13:42:39 -07:00
|
|
|
const BUILTIN_ACCESSORS = [
|
|
|
|
CheckboxControlValueAccessor,
|
|
|
|
RangeValueAccessor,
|
|
|
|
NumberValueAccessor,
|
|
|
|
SelectControlValueAccessor,
|
|
|
|
SelectMultipleControlValueAccessor,
|
|
|
|
RadioControlValueAccessor,
|
|
|
|
];
|
|
|
|
|
2016-08-24 16:58:43 -07:00
|
|
|
export function isBuiltInAccessor(valueAccessor: ControlValueAccessor): boolean {
|
2016-10-19 13:42:39 -07:00
|
|
|
return BUILTIN_ACCESSORS.some(a => valueAccessor.constructor === a);
|
2016-08-24 16:58:43 -07:00
|
|
|
}
|
|
|
|
|
2016-06-08 15:36:24 -07:00
|
|
|
// TODO: vsavkin remove it once https://github.com/angular/angular/issues/3011 is implemented
|
2016-06-08 16:38:52 -07:00
|
|
|
export function selectValueAccessor(
|
|
|
|
dir: NgControl, valueAccessors: ControlValueAccessor[]): ControlValueAccessor {
|
2016-09-30 09:26:53 -07:00
|
|
|
if (!valueAccessors) return null;
|
2016-06-08 15:36:24 -07:00
|
|
|
|
|
|
|
var defaultAccessor: ControlValueAccessor;
|
|
|
|
var builtinAccessor: ControlValueAccessor;
|
|
|
|
var customAccessor: ControlValueAccessor;
|
|
|
|
valueAccessors.forEach((v: ControlValueAccessor) => {
|
2016-10-19 13:42:39 -07:00
|
|
|
if (v.constructor === DefaultValueAccessor) {
|
2016-06-08 15:36:24 -07:00
|
|
|
defaultAccessor = v;
|
|
|
|
|
2016-08-24 16:58:43 -07:00
|
|
|
} else if (isBuiltInAccessor(v)) {
|
2016-10-19 13:42:39 -07:00
|
|
|
if (builtinAccessor)
|
2016-07-13 14:13:02 -07:00
|
|
|
_throwError(dir, 'More than one built-in value accessor matches form control with');
|
2016-06-08 15:36:24 -07:00
|
|
|
builtinAccessor = v;
|
|
|
|
|
|
|
|
} else {
|
2016-10-19 13:42:39 -07:00
|
|
|
if (customAccessor)
|
2016-07-13 14:13:02 -07:00
|
|
|
_throwError(dir, 'More than one custom value accessor matches form control with');
|
2016-06-08 15:36:24 -07:00
|
|
|
customAccessor = v;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2016-10-19 13:42:39 -07:00
|
|
|
if (customAccessor) return customAccessor;
|
|
|
|
if (builtinAccessor) return builtinAccessor;
|
|
|
|
if (defaultAccessor) return defaultAccessor;
|
2016-06-08 15:36:24 -07:00
|
|
|
|
2016-07-13 14:13:02 -07:00
|
|
|
_throwError(dir, 'No valid value accessor for form control with');
|
2016-06-08 15:36:24 -07:00
|
|
|
return null;
|
|
|
|
}
|