【问题标题】:Angular Material Datepicker with ControlValueAccessor and formControlName [duplicate]带有 ControlValueAccessor 和 formControlName 的 Angular Material Datepicker [重复]
【发布时间】:2020-03-24 15:29:34
【问题描述】:

我正在尝试将 ControlValueAccessor 与 Angular Material 日期选择器一起使用。出于某种原因,它不适用于 formControlName 和反应形式。有人可以解决这个问题吗?正在为某些样式和属性构建包装器。这些值被发送到反应形式等

https://material.angular.io/components/datepicker/overview

打字稿:

import { Component, OnInit, Input, Output, EventEmitter, forwardRef } from '@angular/core';
import { NG_VALUE_ACCESSOR } from '@angular/forms';

@Component({
  selector: 'app-date-picker',
  templateUrl: './date-picker.component.html',
  styleUrls: ['./date-picker.component.scss'],
  providers: [
    {
      provide: NG_VALUE_ACCESSOR,
      useExisting: forwardRef(() => DatePickerComponent),
      multi: true
    }
  ], host: {
    '(change)': 'onChange($event)',
    '(blur)': 'onTouch()'
  }
})

export class DatePickerComponent implements OnInit {
  @Input() Value: Date = new Date();
  @Input() PlaceHolder: string;
  @Input() Label:string;
  @Output() dateValueAction = new EventEmitter();

  onChange: any = () => { };
  onTouch: any = () => { };
  constructor() { }

  ngOnInit() {
  }

  set valueUpdated(val) {
    this.Value = val;
    this.onChange(val);
    this.onTouch(val);
  }

  writeValue(obj: any): void {
    this.Value = obj;
    this.valueUpdated = obj;
  }
  registerOnChange(fn: any): void {
    this.onChange = fn;
  }
  registerOnTouched(fn: any): void {
    this.onTouch = fn;
  }

  dateValueChanged() {
    this.valueUpdated = this.Value;
    this.dateValueAction.emit(this.Value);
  }
}

HTML:

<mat-form-field appearance="outline">
    <mat-label>{{Label}}</mat-label>   
    <input matInput [matDatepicker]="picker" placeholder={{PlaceHolder}} [(value)]="Value" (dateChange)="dateValueChanged()">
    <mat-datepicker-toggle matSuffix [for]="picker">
        <mat-icon matDatepickerToggleIcon>calendar_today</mat-icon>
    </mat-datepicker-toggle>
    <mat-datepicker #picker></mat-datepicker>
</mat-form-field>

想用formControlName等实现

<app-date-picker formControlName="startDate">

【问题讨论】:

    标签: angular typescript angular-material angular-reactive-forms angular8


    【解决方案1】:

    在实现ControlValueAccessor 时,您不应使用标准输入/输出通信机制。
    输入值由writeValue提供,控件变化通知(输出值)通过调用registerOnChange提供的回调函数完成。

    请尝试以下代码:

    export class DatePickerComponent implements ControlValueAccessor {
      private value: Date = new Date();
      @Input() placeHolder: string;
      @Input() label: string;
    
      private onChange: Function;
      private onTouch: Function;
    
      writeValue(obj: Date): void {
        this.value = obj;
      }
      registerOnChange(fn: Function): void {
      this.onChange = fn;
      }
      registerOnTouched(fn: Function): void {
        this.onTouch = fn;
      }
      dateValueChanged(): void {
        this.onChange(this.value)
      }
    }
    

    另外,您直接在组件装饰器的描述符中指定了一个类。这样做你不应该使用forwardRef

     providers: [
        {
          provide: NG_VALUE_ACCESSOR,
          useExisting: DatePickerComponent,
          multi: true
        }
      ]
    

    【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2017-03-19
    • 1970-01-01
    • 2018-03-06
    • 2019-07-25
    • 2020-08-17
    • 1970-01-01
    • 1970-01-01
    • 2015-02-06
    相关资源
    最近更新 更多