【问题标题】:Angular: composite ControlValueAccessor to implement nested formAngular:复合 ControlValueAccessor 实现嵌套表单
【发布时间】:2018-04-28 02:06:24
【问题描述】:

在 Angular Connect 2017 演示文稿中介绍了实现嵌套表单的 ControlValueAccessor 组合。

https://docs.google.com/presentation/d/e/2PACX-1vTS20UdnMGqA3ecrv7ww_7CDKQM8VgdH2tbHl94aXgEsYQ2cyjq62ydU3e3ZF_BaQ64kMyQa0INe2oI/pub?slide=id.g293d7d2b9d_1_1532

在本次演讲中,演讲者展示了一种实现自定义表单控件的方法,该控件具有多个值(不仅是单个字符串值,而且具有两个字符串字段,如街道和城市)。我想实现它,但我被卡住了。示例应用程序在这里,有人知道我应该更正什么吗?

https://stackblitz.com/edit/angular-h2ehwx

父组件

@Component({
  selector: 'my-app',
  template: `
    <h1>Form</h1>
    <form [formGroup]="form" (ngSubmit)="onSubmit(form.value)" novalidate>
      <label>name</label>
      <input formControlName="name">
      <app-address-form formControlName="address"></app-address-form>
      <button>submit</button>
    </form>
  `,
})
export class AppComponent  {
  @Input() name: string;
  submitData = '';
  form: FormGroup;

  constructor(private fb: FormBuilder) {
    this.form = fb.group({
      name: 'foo bar',
      address: fb.group({
        city: 'baz',
        town: 'qux',
      })
    });
  }

  onSubmit(v: any) {
    console.log(v);
  }
}

嵌套表单组件

@Component({
  selector: 'app-address-form',
  template: `
    <div [formGroup]="form">
      <label>city</label>
      <input formControlName="city" (blur)="onTouched()">
      <label>town</label>
      <input formControlName="town" (blur)="onTouched()">
    </div>
  `,
  providers: [
    {
      provide: NG_VALUE_ACCESSOR,
      multi: true,
      useExisting: forwardRef(() => AddressFormComponent)
    }
  ]
})
export class AddressFormComponent implements ControlValueAccessor {
  form: FormGroup;

  onTouched: () => void = () => {};

  writeValue(v: any) {
    this.form.setValue(v, { emitEvent: false });
  }

  registerOnChange(fn: (v: any) => void) {
    this.form.valueChanges.subscribe(fn);
  }

  setDisabledState(disabled: boolean) {
    disabled ? this.form.disable() : this.form.enable();
  }

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

我收到的错误消息

ERROR TypeError: Cannot read property 'setValue' of undefined
at AddressFormComponent.writeValue (address-form.component.ts:32)
at setUpControl (shared.js:47)
at FormGroupDirective.addControl (form_group_directive.js:125)
at FormControlName._setUpControl (form_control_name.js:201)
at FormControlName.ngOnChanges (form_control_name.js:114)
at checkAndUpdateDirectiveInline (provider.js:249)
at checkAndUpdateNodeInline (view.js:472)
at checkAndUpdateNode (view.js:415)
at debugCheckAndUpdateNode (services.js:504)
at debugCheckDirectivesFn (services.js:445)

我认为 FormGroup 实例应该以某种方式注入到嵌套的表单组件中......

【问题讨论】:

  • 您可以在问题中发布您的代码吗?你到底有什么问题?
  • @roelofs 谢谢,我更新了帖子。

标签: forms angular


【解决方案1】:

我们(在工作中)遇到了这个问题并尝试了几个月的不同方法:如何正确处理嵌套表单。

确实,ControlValueAccessor 似乎是要走的路,但我们发现它非常冗长,构建嵌套表单的时间很长。由于我们在我们的应用程序中大量使用该模式,因此我们最终花费了一些时间进行调查并尝试提出更好的解决方案。我们称它为ngx-sub-form,它是NPM 上的一个repo(+Github 上的源代码)。

基本上,要创建子表单,您只需扩展我们提供的类并传递您的 FormControls。就是这样。

我们已经更新了我们的代码库以使用它,我们对此非常满意,所以您可能想尝试一下,看看它对您有何帮助:)

README on github 中解释了一切。

PS:我们还有一个完整的演示在这里运行https://cloudnc.github.io/ngx-sub-form

【讨论】:

  • 我可以投反对票,但很高兴能解释一下为什么你认为我写了一个错误的答案。
【解决方案2】:

几个问题,在您的AppComponent 上将您的FormBuilder 更改为:

this.form = fb.group({
  name: 'foo bar',
  address: fb.control({ //Not using FormGroup
    city: 'baz',
    town: 'qux',
  })
});

在您的AddressFormComponent 上,您需要像这样初始化您的 FormGroup:

form: FormGroup = new FormGroup({
    city: new FormControl,
    town: new FormControl
});

这是您的示例的分支:https://stackblitz.com/edit/angular-np38bi

【讨论】:

  • 哇,我必须使用 fb.control 而不是 fb.group。我觉得很奇怪,因为它的实例被引用为 FormGroup,但它正在工作!谢谢@12seconds。
  • fmm,如果 FormArray 是嵌套的,我得到了错误。问题是什么? stackblitz.com/edit/angular-krhrcz
  • @adwd 我认为自定义控件是一个控件是很自然的。 :) 自定义控件的值只是复杂的对象。
  • @penleychan 验证有问题。如果我们将嵌套表单视为 FormControl,则主表单不知道嵌套表单的有效性。这是演示stackblitz.com/edit/angular-pmfq5z?file=app%2Fapp.component.ts。尝试清除必填字段。我们如何解决这个问题?
  • @TarasHupalo,如果您stackoverflow.com/a/53053333/6736888,我建议您看看我的其他答案以实现此目的
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2011-03-08
  • 2020-07-07
  • 2023-02-05
  • 2019-06-06
  • 1970-01-01
  • 2019-07-25
相关资源
最近更新 更多