2015-11-18 15:55:43 -08:00
|
|
|
import {Injectable} from 'angular2/core';
|
2015-11-06 17:34:07 -08:00
|
|
|
import {StringMapWrapper} from 'angular2/src/facade/collection';
|
|
|
|
import {isPresent, isArray, CONST_EXPR, Type} from 'angular2/src/facade/lang';
|
2015-03-09 17:41:49 +01:00
|
|
|
import * as modelModule from './model';
|
2016-02-19 11:49:31 -08:00
|
|
|
import {ValidatorFn, AsyncValidatorFn} from './directives/validators';
|
2015-03-10 18:12:50 -07:00
|
|
|
|
|
|
|
|
2015-03-31 22:47:11 +00:00
|
|
|
/**
|
2015-04-10 11:15:01 -07:00
|
|
|
* Creates a form object from a user-specified configuration.
|
|
|
|
*
|
2015-09-29 12:59:56 -07:00
|
|
|
* ### Example ([live demo](http://plnkr.co/edit/ENgZo8EuIECZNensZCVr?p=preview))
|
2015-05-11 15:20:04 +02:00
|
|
|
*
|
2015-09-29 12:59:56 -07:00
|
|
|
* ```typescript
|
2015-04-29 10:33:20 -05:00
|
|
|
* @Component({
|
2015-09-29 12:59:56 -07:00
|
|
|
* selector: 'my-app',
|
|
|
|
* viewBindings: [FORM_BINDINGS]
|
2015-04-29 10:33:20 -05:00
|
|
|
* template: `
|
2015-11-23 16:02:19 -08:00
|
|
|
* <form [ngFormModel]="loginForm">
|
|
|
|
* <p>Login <input ngControl="login"></p>
|
|
|
|
* <div ngControlGroup="passwordRetry">
|
|
|
|
* <p>Password <input type="password" ngControl="password"></p>
|
|
|
|
* <p>Confirm password <input type="password" ngControl="passwordConfirmation"></p>
|
2015-04-29 10:33:20 -05:00
|
|
|
* </div>
|
|
|
|
* </form>
|
2015-09-29 12:59:56 -07:00
|
|
|
* <h3>Form value:</h3>
|
|
|
|
* <pre>{{value}}</pre>
|
2015-04-29 10:33:20 -05:00
|
|
|
* `,
|
2015-09-01 20:59:43 -07:00
|
|
|
* directives: [FORM_DIRECTIVES]
|
2015-04-29 10:33:20 -05:00
|
|
|
* })
|
2015-09-29 12:59:56 -07:00
|
|
|
* export class App {
|
2015-04-29 10:33:20 -05:00
|
|
|
* loginForm: ControlGroup;
|
2015-05-11 15:20:04 +02:00
|
|
|
*
|
2015-04-29 10:33:20 -05:00
|
|
|
* constructor(builder: FormBuilder) {
|
|
|
|
* this.loginForm = builder.group({
|
|
|
|
* login: ["", Validators.required],
|
|
|
|
* passwordRetry: builder.group({
|
|
|
|
* password: ["", Validators.required],
|
2015-11-02 10:00:56 -08:00
|
|
|
* passwordConfirmation: ["", Validators.required, asyncValidator]
|
2015-04-29 10:33:20 -05:00
|
|
|
* })
|
|
|
|
* });
|
|
|
|
* }
|
2015-04-10 11:15:01 -07:00
|
|
|
*
|
2015-09-29 12:59:56 -07:00
|
|
|
* get value(): string {
|
|
|
|
* return JSON.stringify(this.loginForm.value, null, 2);
|
|
|
|
* }
|
|
|
|
* }
|
2015-04-10 11:15:01 -07:00
|
|
|
* ```
|
2015-03-31 22:47:11 +00:00
|
|
|
*/
|
2015-07-17 13:21:37 -07:00
|
|
|
@Injectable()
|
2015-03-10 18:12:50 -07:00
|
|
|
export class FormBuilder {
|
2015-09-29 12:59:56 -07:00
|
|
|
/**
|
|
|
|
* Construct a new {@link ControlGroup} with the given map of configuration.
|
|
|
|
* Valid keys for the `extra` parameter map are `optionals` and `validator`.
|
|
|
|
*
|
|
|
|
* See the {@link ControlGroup} constructor for more details.
|
|
|
|
*/
|
2015-10-02 16:47:54 -07:00
|
|
|
group(controlsConfig: {[key: string]: any},
|
|
|
|
extra: {[key: string]: any} = null): modelModule.ControlGroup {
|
2015-03-10 18:12:50 -07:00
|
|
|
var controls = this._reduceControls(controlsConfig);
|
2016-02-19 11:49:31 -08:00
|
|
|
var optionals = <{[key: string]: boolean}>(
|
|
|
|
isPresent(extra) ? StringMapWrapper.get(extra, "optionals") : null);
|
|
|
|
var validator: ValidatorFn = isPresent(extra) ? StringMapWrapper.get(extra, "validator") : null;
|
|
|
|
var asyncValidator: AsyncValidatorFn =
|
|
|
|
isPresent(extra) ? StringMapWrapper.get(extra, "asyncValidator") : null;
|
2015-11-02 10:00:56 -08:00
|
|
|
return new modelModule.ControlGroup(controls, optionals, validator, asyncValidator);
|
2015-03-10 18:12:50 -07:00
|
|
|
}
|
2015-09-29 12:59:56 -07:00
|
|
|
/**
|
2015-11-02 10:00:56 -08:00
|
|
|
* Construct a new {@link Control} with the given `value`,`validator`, and `asyncValidator`.
|
2015-09-29 12:59:56 -07:00
|
|
|
*/
|
2016-02-19 11:49:31 -08:00
|
|
|
control(value: Object, validator: ValidatorFn = null,
|
|
|
|
asyncValidator: AsyncValidatorFn = null): modelModule.Control {
|
2015-11-02 10:00:56 -08:00
|
|
|
return new modelModule.Control(value, validator, asyncValidator);
|
2015-03-10 18:12:50 -07:00
|
|
|
}
|
|
|
|
|
2015-09-29 12:59:56 -07:00
|
|
|
/**
|
|
|
|
* Construct an array of {@link Control}s from the given `controlsConfig` array of
|
2015-11-02 10:00:56 -08:00
|
|
|
* configuration, with the given optional `validator` and `asyncValidator`.
|
2015-09-29 12:59:56 -07:00
|
|
|
*/
|
2016-02-19 11:49:31 -08:00
|
|
|
array(controlsConfig: any[], validator: ValidatorFn = null,
|
|
|
|
asyncValidator: AsyncValidatorFn = null): modelModule.ControlArray {
|
2015-10-06 18:00:42 -07:00
|
|
|
var controls = controlsConfig.map(c => this._createControl(c));
|
2015-11-02 10:00:56 -08:00
|
|
|
return new modelModule.ControlArray(controls, validator, asyncValidator);
|
2015-03-25 10:51:05 -07:00
|
|
|
}
|
|
|
|
|
2015-10-09 17:21:25 -07:00
|
|
|
/** @internal */
|
2016-02-11 17:01:17 -08:00
|
|
|
_reduceControls(controlsConfig: {[k: string]:
|
|
|
|
any}): {[key: string]: modelModule.AbstractControl} {
|
2015-10-02 17:33:21 -07:00
|
|
|
var controls: {[key: string]: modelModule.AbstractControl} = {};
|
2016-02-11 17:01:17 -08:00
|
|
|
StringMapWrapper.forEach(controlsConfig, (controlConfig: any, controlName: string) => {
|
2015-03-10 18:12:50 -07:00
|
|
|
controls[controlName] = this._createControl(controlConfig);
|
|
|
|
});
|
|
|
|
return controls;
|
|
|
|
}
|
|
|
|
|
2015-10-09 17:21:25 -07:00
|
|
|
/** @internal */
|
2015-05-20 18:10:30 -07:00
|
|
|
_createControl(controlConfig: any): modelModule.AbstractControl {
|
2015-07-10 11:29:41 +02:00
|
|
|
if (controlConfig instanceof modelModule.Control ||
|
|
|
|
controlConfig instanceof modelModule.ControlGroup ||
|
|
|
|
controlConfig instanceof modelModule.ControlArray) {
|
2015-03-10 18:12:50 -07:00
|
|
|
return controlConfig;
|
|
|
|
|
2015-06-11 19:32:55 +02:00
|
|
|
} else if (isArray(controlConfig)) {
|
2015-08-28 11:29:19 -07:00
|
|
|
var value = controlConfig[0];
|
2016-02-19 11:49:31 -08:00
|
|
|
var validator: ValidatorFn = controlConfig.length > 1 ? controlConfig[1] : null;
|
|
|
|
var asyncValidator: AsyncValidatorFn = controlConfig.length > 2 ? controlConfig[2] : null;
|
2015-11-02 10:00:56 -08:00
|
|
|
return this.control(value, validator, asyncValidator);
|
2015-03-10 18:12:50 -07:00
|
|
|
|
|
|
|
} else {
|
|
|
|
return this.control(controlConfig);
|
|
|
|
}
|
|
|
|
}
|
2016-02-19 11:49:31 -08:00
|
|
|
}
|