如何在 Angular 中触摸自定义控件组件中的内部控件?

Pla*_*Rov 3 typescript angular2-forms angular angular-forms

我有一个带有我自己的自定义控件组件的表单:

@Component({
  selector: "my-app",
  template: `
    <form [formGroup]="form">
      <app-custom-control formControlName="customControl"></app-custom-control>
    </form>

    <button (click)="touch()">
      Touch!
    </button>
  `,
  styleUrls: ["./app.component.css"]
})
export class AppComponent {
  form: FormGroup;

  constructor(private fb: FormBuilder) {}

  ngOnInit() {
    this.form = this.fb.group({
      customControl: "1"
    });
  }

  touch() {
    this.form.markAllAsTouched();
  }
}
Run Code Online (Sandbox Code Playgroud)

我的自定义控制组件内部又包含另一个自定义控制组件(在我的真实应用程序中需要它,因为外部控制组件有两种模式 - 读取和编辑):

@Component({
  selector: "app-custom-control",
  template: `
    <ng-select [ngModel]="value" [items]="items"></ng-select>
  `,
  styleUrls: ["./custom-control.component.css"],
  providers: [
    {
      provide: NG_VALUE_ACCESSOR,
      useExisting: CustomControlComponent,
      multi: true
    }
  ]
})
export class CustomControlComponent implements ControlValueAccessor, OnInit {
  items = ["1", "2"];

  value = null;
  onChange = (value: any) => {};
  onTouched = () => {};

  constructor() {}

  registerOnChange(fn: any) {
    this.onChange = fn;
  }

  registerOnTouched(fn: () => {}): void {
    this.onTouched = fn;
  }

  writeValue(outsideValue: number) {
    this.value = outsideValue;
  }
}
Run Code Online (Sandbox Code Playgroud)

可以添加NG_VALIDATORSCustomControlComponent实现组件的验证方式(它只会引发 ng-select 错误)。this.form.markAllAsTouched()但我真的不知道在包装表单的组件中执行操作时如何触摸内部组件(ng-select) 。

我尝试在ngDoCheck生命周期方法中进行操作,但它似乎运行了太多次,这是不可接受的解决方案。

游乐场:https://stackblitz.com/edit/angular-ivy-u3kdfj

yur*_*zui 5

github上有几个相关的问题:

您可以做的是获取 ControlValueAccessor 控件(outerControl)和内部 NgModel 控件(innerControl)的 NgControl:

自定义控件.component.ts

export class CustomControlComponent implements ControlValueAccessor, AfterViewInit {
  @ViewChild(NgControl) innerNgControl: NgControl;

  constructor(private inj: Injector) {}

  ...

  ngAfterViewInit() { 
    const outerControl = this.inj.get(NgControl).control;
    ...
  }
}
Run Code Online (Sandbox Code Playgroud)

然后你可以使用这个技巧绕过触摸状态到内部控制:

const prevMarkAsTouched = outerControl.markAsTouched;
outerControl.markAsTouched = (...args: any) => { 
  this.innerNgControl.control.markAsTouched();
  prevMarkAsTouched.bind(outerControl)(...args);
};
Run Code Online (Sandbox Code Playgroud)

叉式堆栈闪电战