Пользовательский angular2 компонент ввода формы с двусторонней привязкой и проверкой внутри компонента

Есть ли способ сделать входной компонент двустороннего связывания, который также может иметь валидацию внутри компонента?

То, что я пытаюсь достичь, состоит в том, чтобы иметь компоненты, которые я мог бы выстроить в своих формах следующим образом:

<form #f="ngForm">
            <my-form-input [(inputModel)]="name" [inputField]="myFormInputName"></my-form-input>
            <my-form-input [(inputModel)]="name2" [inputField]="myFormInputName2"></my-form-input>
...
            <my-form-input [(inputModel)]="lastItem" [inputField]="lastItemName"></my-form-input>

</form>

У меня есть следующая настройка и не могу понять, как сделать это правильно:

Компонент:

import {Component,Input, Output,EventEmitter} from 'angular2/core'
import {FORM_DIRECTIVES}    from 'angular2/common';

@Component({
  selector: 'my-form-input',
  directives: [FORM_DIRECTIVES],
  template:
    `
    <input type="text" class="form-control"  id="i1" required [ngModel]="inputModel" (ngModelChange)="onChangeInput($event)" ngControl="ctrl" #ctrl="ngForm"/>
    <p>{{"Is field valid? I would like to make some decisions here depending on that: "+ctrl.valid}}</p>

  `
})
export class InputComponent {

  constructor(){};

  @Input()  inputField:string;
  @Input()  inputModel: Object;
  @Output() inputModelChange = new EventEmitter();

  onChangeInput(event){
    this.inputModel=event;
    this.inputModelChange.emit(event);
  }
}

Приложение:

//our root app component
import {Component} from 'angular2/core'
import {FORM_DIRECTIVES}    from 'angular2/common';
import {InputComponent} from './my.input'


@Component({
  selector: 'my-app',
  providers: [],
  template: `
    <div>
      <p>Is there a way to make a custom 2 way binding form input component having also validation?</p>
      <form #f="ngForm">

        <my-form-input [(inputModel)]="name" [inputField]="myFormInputName"></my-form-input>

        <p>{{name}}</p>
      </form>
    </div>
  `,
  directives: [InputComponent,FORM_DIRECTIVES]
})
export class App {
  constructor() {
    this.name = 'Angular2'
  }
}

Я также сделал Plunker для иллюстрации моей проблемы: http://plnkr.co/edit/0vXjHbQmv7v7EKQcLWaa?p=preview

Ответы

Ответ 1

Вы можете передать элемент управления формы своему компоненту, чтобы создать выделенный элемент управления для ввода. На основе этого нового элемента управления отображать ошибки, если их атрибуты valid false:

@Component({
  selector: 'my-form-input',
  directives: [FORM_DIRECTIVES],
  template: `
    <input type="text" class="form-control"  id="i1"   
       [ngModel]="inputModel"
       (ngModelChange)="onChangeInput($event)"
       [ngFormControl]="formCtrl.controls[inputField]"/>
    <p>Is field valid? I would like to make some decisions
       here depending on that: {{formCtrl.controls[inputField].valid}}
    </p>
  `
})
export class InputComponent implements OnInit {
  @Input()  inputField:string;
  @Input()  formCtrl;
  @Input()  inputModel: Object;
  @Output() inputModelChange = new EventEmitter(); 

  ngOnInit() {
    this.formCtrl.control.addControl(
        this.inputField, new Control('', Validators.required));
  }

  onChangeInput(event){
    this.inputModel=event;
    this.inputModelChange.emit(event);
  }
}

Вам нужно использовать метод addControl, чтобы состояние всей формы соответствовало элементам управления, созданным в ваших входных компонентах.

В вашем случае вы определяете свои встроенные элементы управления с помощью директивы ngControl. Я сделал несколько тестов, но я не могу заставить его работать таким образом...

Вот как использовать этот компонент из родительского:

@Component({
  selector: 'my-app', 
  template: `
    <form #f="ngForm">
      <my-form-input [(inputModel)]="name" [inputField]="myFormInputName" [formCtrl]="f"></my-form-input>
    </form>
  `,
  directives: [ FORM_DIRECTIVES, InputComponent ]
})
export class AppComponent {
}

Он оставался проблемой проверки сразу после запуска. Глобальное состояние формы не обновлялось, и произошла следующая ошибка:

Expression '
    Valid : {{f.valid}} in [email protected]:24' has changed after it was checked. Previous value: '
    Valid : true'. Current value: '
    Valid : false'

Чтобы исправить этот аспект, посмотрите на ответ Жюльена, основанный на классе ChangeDetectorRef и его методе detectChanges...

Вот пример plunkr: https://plnkr.co/edit/Z4uOUq2q4iXdpo0J6R1o?p=preview.

Ответ 2

Ответ Thierry частично хорош, так как при включении ProdMode вы просто скрываете проблему: вы не обновляете свой компонент.

После инициализации компонента формы вы должны вызвать "detectChange":

export class AppComponent {
    constructor(private cdr: ChangeDetectorRef) {}
    ngOnInit() {
        this.cdr.detectChanges();
    }
}