欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

angular4自定義表單控件[(ngModel)]的實(shí)現(xiàn)

 更新時(shí)間:2018年11月23日 09:53:32   作者:去StackOverflow  
這篇文章主要介紹了angular4自定義表單控件[(ngModel)]的實(shí)現(xiàn),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧

[(ngModel)]拆分

[(ngModel)][]輸入()輸出組合起來(lái),進(jìn)行雙向數(shù)據(jù)綁定。拆分開(kāi)來(lái)

  • 輸入屬性
  • [ngModel](ngModelChange)輸出監(jiān)聽(tīng)元素值的變化,并同步view value與model value。

復(fù)制代碼 代碼如下:
<input type="text" id="modelInner" [ngModel]="model" (ngModelChange)="getModelChange($event)">

 model: string;
  constructor() {
    this.model = 'model init';
  }

  getModelChange(event: string) {
    this.model = event; // view value 與 model value 同步
  }

自定義組件上使用 [(ngModel)]

我們不能把[(ngModel)]用到非表單類(lèi)的原生元素或第三方自定義組件上,除非寫(xiě)一個(gè)合適的值訪問(wèn)器,這種技巧超出了本章的范圍。

Angular文檔中描述到這里,就中止了。剛好我要定制一個(gè)模擬radio的組件,只能如文檔所說(shuō),依葫蘆畫(huà)瓢實(shí)現(xiàn) ControlValueAccessor。

ControlValueAccessor接口

ControlValueAccessor acts as a bridge between the Angular forms API and a native element in the DOM.
Implement this interface if you want to create a custom form control directive that integrates with Angular forms.

簡(jiǎn)而言之,實(shí)現(xiàn)了這個(gè)接口的組件,就可以使用 Angular forms API,比如[(ngModel)] 。

interface ControlValueAccessor { 
 writeValue(obj: any): void
 registerOnChange(fn: any): void
 registerOnTouched(fn: any): void
 setDisabledState(isDisabled: boolean)?: void
}

實(shí)現(xiàn)ControlValueAccessor步驟

模仿primeng中的自定義radio組件,寫(xiě)了一個(gè)簡(jiǎn)單的自定義radio組件。

  • 創(chuàng)建一個(gè)RADIO_VALUE_ACCESSOR常量用來(lái)在組件中注冊(cè)NG_VALUE_ACCESSOR
  • 實(shí)現(xiàn)ControlValueAccessor中的3+1個(gè)方法

完整demo代碼如下:

import { NgModule, Component, Input, Output, ElementRef, OnInit, EventEmitter, forwardRef, ViewChild, ChangeDetectorRef } from '@angular/core';
import { CommonModule } from '@angular/common';
import { NG_VALUE_ACCESSOR, ControlValueAccessor } from '@angular/forms';

const RADIO_VALUE_ACCESSOR: any = {
 provide: NG_VALUE_ACCESSOR,
 useExisting: forwardRef(() => PRadioComponent),
 multi: true
};

@Component({
 selector: 'app-p-radio',
 template: `
    <div class="p-radio">
      <label class="radio-label" (click)="select()" *ngIf="label">
        <div class="name" [class.checked-name]="rb.checked">{{label}}</div>
      </label>
      <div class="helper-hidden-accessible">
        <input #rb type="radio" [attr.name]="name" [attr.value]="value" [checked]="checked">
      </div>
      <div class="radio-md" (click)="handleClick()">
        <div class="radio-icon " [class.radio-checked]="rb.checked">
           <div class="radio-inner"></div>
        </div>
      </div>
    </div>
  `,
 styleUrls: ['./p-radio.component.scss'],
 providers: [RADIO_VALUE_ACCESSOR]
})
export class PRadioComponent implements ControlValueAccessor {

 @Input() name: string;
 @Input() label: string;
 @Input() value: string;
 checked: boolean;

 @ViewChild('rb') inputViewChild: ElementRef;
 @Output() pRadioChange: EventEmitter<any> = new EventEmitter();
 onModelChange: Function = () => { };

 constructor(
  private cd: ChangeDetectorRef
 ) { }

 // model view -> view value
 writeValue(value: any): void {
  if (value) {
   this.checked = (value === this.value);
   if (this.inputViewChild.nativeElement) {
    this.inputViewChild.nativeElement.checked = this.checked;
   }
   this.cd.markForCheck();
  }
 }

 // view value ->model value
 registerOnChange(fn: Function): void {
  this.onModelChange = fn;
 }

 registerOnTouched(fn: Function): void { }

 handleClick() {
  this.select();
 }

 select() {
  this.inputViewChild.nativeElement.checked = !this.inputViewChild.nativeElement.checked;
  this.checked = !this.checked;
  if (this.checked) {
   this.onModelChange(this.value); // 同步view value 和 model value
  } else {
   this.onModelChange(null);
  }
  this.pRadioChange.emit(null);
 }

}

@NgModule({
 imports: [CommonModule],
 exports: [PRadioComponent],
 declarations: [PRadioComponent]
})

export class RadioButtonModule { }

方法何時(shí)被調(diào)用?

writeValue(obj: any): void

API中提到 (model -> view) 時(shí),writeValue() 會(huì)被調(diào)用。
model value 和 view value分別指什么?
舉個(gè)調(diào)用PRadioComponent的例子:

復(fù)制代碼 代碼如下:
<app-p-radio [value]="'1'" [label]="'text1'" [(ngModel)]="checkedValue"></app-p-radio>

這里checkedValue屬性就是model value,view value 為PRadioComponent內(nèi)部的某個(gè)屬性(PRadioComponent中定義為this.value)。

當(dāng)model view(checkedValue)發(fā)生改變時(shí),PRadioComponent中的writeValue(obj: any)就會(huì)被調(diào)用,參數(shù)為當(dāng)前model value(checkedValue)的值,在函數(shù)中將參數(shù)值賦給內(nèi)部的view value,從而實(shí)現(xiàn)(model -> view)。接受到model value的值后,改變PRadioComponent的UI顯示。

registerOnChange(fn: any): void

這個(gè)方法的作用是同步 view value 和 model value (view -> model),

 registerOnChange(fn: Function): void {
  this.onModelChange = fn;
 }

調(diào)用this.onModelChange()時(shí)候,將view value當(dāng)作參數(shù)傳入此方法中,即完成了同步,此例子中this.onModelChange(this.value);。

上面兩種方法是相對(duì)的:

  • writeValue(obj: any): model value發(fā)生改變 ,完成后UI發(fā)生改變(model value-> view value)
  • registerOnChange(fn: any): 觸發(fā)事件(比如click),view value和UI發(fā)生改變,完成調(diào)用后model value與view value同步(view value-> model value)

registerOnTouched(fn: any): void

setDisabledState(isDisabled: boolean)?: void

目的只為在控件中簡(jiǎn)單的使用[(ngModel)],所以這兩個(gè)方法沒(méi)有用到。registerOnTouched(fn: any)必須實(shí)現(xiàn),所以定義了一個(gè)空函數(shù)。

實(shí)際效果

初始值為'a',點(diǎn)擊改變view value,在Angury調(diào)試工具中看到值改為'b'。然后在調(diào)試工具中將checkedValue改為'a',視圖發(fā)生了改變??梢?jiàn),完成了數(shù)據(jù)的雙向綁定。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Angular8 實(shí)現(xiàn)table表格表頭固定效果

    Angular8 實(shí)現(xiàn)table表格表頭固定效果

    這篇文章主要介紹了Angular8 實(shí)現(xiàn)table表格表頭固定效果,表頭固定,內(nèi)部實(shí)現(xiàn)滾動(dòng)條效果,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-01-01
  • 深入理解Angular4訂閱(Subscribe)與取消

    深入理解Angular4訂閱(Subscribe)與取消

    這篇文章主要介紹了深入理解Angular4訂閱(Subscribe)與取消,詳細(xì)的介紹了訂閱與取消的使用,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-11-11
  • Angularjs之如何在跨域請(qǐng)求中傳輸Cookie的方法

    Angularjs之如何在跨域請(qǐng)求中傳輸Cookie的方法

    跨域傳輸Cookie是需要后臺(tái)和前臺(tái)同時(shí)做相關(guān)處理才能解決的,這篇文章主要介紹了Angularjs之如何在跨域請(qǐng)求中傳輸Cookie的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-06-06
  • 如何利用AngularJS打造一款簡(jiǎn)單Web應(yīng)用

    如何利用AngularJS打造一款簡(jiǎn)單Web應(yīng)用

    如果大家希望在應(yīng)用程序的創(chuàng)建工作中采取各類(lèi)最佳實(shí)踐,那么AngularJS也能夠帶來(lái)極大的助益??偠灾?,這套框架的強(qiáng)大功能與特性永遠(yuǎn)不會(huì)讓有著應(yīng)用開(kāi)發(fā)需求的朋友們失望
    2015-12-12
  • angular實(shí)現(xiàn)頁(yè)面打印局部功能的思考與方法

    angular實(shí)現(xiàn)頁(yè)面打印局部功能的思考與方法

    這篇文章主要給大家介紹了關(guān)于angular實(shí)現(xiàn)頁(yè)面打印局部功能的思考與方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。
    2018-04-04
  • 淺析Angular2子模塊以及異步加載

    淺析Angular2子模塊以及異步加載

    本篇文章主要介紹了淺析Angular2子模塊以及異步加載,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-04-04
  • angular源碼學(xué)習(xí)第一篇 setupModuleLoader方法

    angular源碼學(xué)習(xí)第一篇 setupModuleLoader方法

    這篇文章主要介紹了angular源碼學(xué)習(xí)第一篇,setupModuleLoader方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-10-10
  • Angularjs中的事件廣播 —全面解析$broadcast,$emit,$on

    Angularjs中的事件廣播 —全面解析$broadcast,$emit,$on

    下面小編就為大家?guī)?lái)一篇Angularjs中的事件廣播 —全面解析$broadcast,$emit,$on。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考,一起跟隨小編過(guò)來(lái)看看吧
    2016-05-05
  • angularJS 中input示例分享

    angularJS 中input示例分享

    這篇文章主要介紹了angularJS 中input示例分享,需要的朋友可以參考下
    2015-02-02
  • AngularJS控制器controller給模型數(shù)據(jù)賦初始值的方法

    AngularJS控制器controller給模型數(shù)據(jù)賦初始值的方法

    這篇文章主要介紹了AngularJS控制器controller給模型數(shù)據(jù)賦初始值的方法,涉及AngularJS控制器controller簡(jiǎn)單賦值操作實(shí)現(xiàn)技巧,需要的朋友可以參考下
    2017-01-01

最新評(píng)論