简单实现angular2组件双向绑定(转)

直接献上代码

父组件

import { Component, OnInit } from ‘@angular/core‘;

@Component({
  selector: ‘ngx-input‘,
  templateUrl: `
    <p>双向绑定</p>
    {{pvalue}} <input type="radio" [attr.value]="pvalue" [checked]="pvalue" (change)="pvalue=!pvalue">
    <br>
    <p>双向子组件</p>
    <ngx-two-way [(ngModel)]="pvalue"></ngx-two-way>
`
})
export class InputComponent {

  pvalue = false;

  constructor() { }
}

子组件

import {Component, forwardRef} from ‘@angular/core‘;
import {ControlValueAccessor, NG_VALUE_ACCESSOR} from ‘@angular/forms‘

@Component({
  selector: ‘ngx-two-way‘,
  templateUrl:`
    <div class="toggle-button-container">
      <a href="javascript:;"  (click)="toggle()" class="text-close" *ngIf="!model">关</a>
      <a href="javascript:;"  (click)="toggle()" class="text-open" *ngIf="model">开</a>
  </div>
  `,
  providers: [{
        provide: NG_VALUE_ACCESSOR,
        useExisting: forwardRef(() => TwoWayComponent),
        multi: true
    }]
})
export class TwoWayComponent implements ControlValueAccessor {

  constructor() { }

  private model: any;

  public onModelChange: Function = () => {};
  public onModelTouched: Function = () => {};
  writeValue(value: any) {
      this.model = value;
  }
  registerOnChange(fn: Function): void {
      this.onModelChange = fn;
  }
  registerOnTouched(fn: Function): void {
      this.onModelTouched = fn;
  }

  // setDisabledState(val: boolean): void {}

  toggle(event) {
    this.model = !this.model;
    this.onModelChange(this.model);
  }
}

简单说明:

NG_VALUE_ACCESSOR

原文说明:

Used to provide a ControlValueAccessor for form controls.

为ControlValueAccessor提供注入

ControlValueAccessor:

这是实现双向绑定的核心

原文说明:

A bridge between a control and a native element.

A ControlValueAccessor abstracts the operations of writing a new value to a DOM element representing an input control.

大意是让一个DOM元素像input空间一样去展示数据

其中registerOnChange和registerOnTouched是必须实现的方法。

useExisting: forwardRef(() => TwoWayComponent),

如果少了这一段,会报错

image.png

 Argument of type ‘{ selector: string; templateUrl: string; styleUrls: string[]; providers: { provide: InjectionToke...‘ is not assignable to parameter of type ‘Component‘.
  Types of property ‘providers‘ are incompatible.
    Type ‘{ provide: InjectionToken<ControlValueAccessor>; }[]‘ is not assignable to type ‘Provider[]‘.
      Type ‘{ provide: InjectionToken<ControlValueAccessor>; }‘ is not assignable to type ‘Provider‘.
        Type ‘{ provide: InjectionToken<ControlValueAccessor>; }‘ is not assignable to type ‘FactoryProvider‘.
          Property ‘useFactory‘ is missing in type ‘{ provide: InjectionToken<ControlValueAccessor>; }‘.

大意是InjectionToke(在NG_VALUE_ACCESSOR中import的)必须被指派一个component

InjectionToke

原文说明:

Creates a token that can be used in a DI Provider.

Use an InjectionToken whenever the type you are injecting is not reified (does not have a runtime representation) such as when injecting an interface, callable type, array or parametrized type.

InjectionToken is parametrize on T which is the type of object which will be returned by the Injector. This provides additional level of type safety.

大意是让你在接口、类在还没实例化之前调用

forwardRef

原文说明:

Allows to refer to references which are not yet defined.

For instance, forwardRef is used when the token which we need to refer to for the purposes of DI is declared, but not yet defined. It is also used when the token which we use when creating a query is not yet defined.

为InjectionToke服务的,让有DI未定义的token能够使用

multi: true

少了会报错

image.png

应该是为NG_VALUE_ACCESSOR能让forEach而服务的

作者:琢磨先生lf
链接:https://www.jianshu.com/p/75b59181a5d9
來源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。

原文地址:https://www.cnblogs.com/gynStrive/p/9715250.html

时间: 2024-10-16 05:24:41

简单实现angular2组件双向绑定(转)的相关文章

Angular中父子组件双向绑定传值

下面为大家展示一个较为简单的ng父子组件双向绑定传值,下面是父组件页面 这个页面的大概功能就是父组件(红色)通过输入框输入内容反映到子组件上进行展示,并且进行了投影, 子组件(橙黄色)通过Input输入框输入内容反映到父组件上,并且使用了@Output传值给父组件 下面是父组件的页面 父组件ts 子组件页面 子组件ts 我主要来说下需要注意的几个地方,第一个就是当我利用子组件的@Output发射数据给父组件的时候, import { EventEmitter } from '@angular/c

vue组件双向绑定.sync修饰符的一个坑

我们知道组件是单项的,但是有时候需要双向,这时候我们可以使用.sync修饰符,但今天遇到一个坑,一直不成功,花了半小时试出来的.... 在编程的时候我们很习惯冒号后面跟着空格.而.sync双向绑定需要子组件显性触发 this.$emit('update:foo', newValue) 这里的updata:foo可以说是一个名字,不能加空格,不能加空格,不能加空格!!!这就是坑,由于编程习惯,也许会有人加空格,我也是,所以写出来当作一个笔记. 根据文档 <comp :foo.sync="ba

react 父组件于子组件双向绑定

在项目中我们可能会遇到类似这样的场景,也就是父子组件的双向数据绑定 首先,先把在head中引入react.js.react-dom.js和可选择的babel.js(这里需要注意引用的顺序,react.js必须在react-dom.js之前) <head> <script src="react.js"></script> <script src="react-dom.js"></script> <scr

自定义一个双向绑定组件

父组件: <k-input v-model="model.username"></k-input> 子组件KInput: <template> <div> <!--实现自定义组件双向绑定 v-model是语法糖,实现自定义组件双绑定需要指定 :value和@input即可--> <input type="text" :value="value" @input="onInp

vue的双向绑定原理及实现

前言 使用vue也好有一段时间了,虽然对其双向绑定原理也有了解个大概,但也没好好探究下其原理实现,所以这次特意花了几晚时间查阅资料和阅读相关源码,自己也实现一个简单版vue的双向绑定版本,先上个成果图来吸引各位: 代码:                                                                    效果图:   是不是看起来跟vue的使用方式差不多?接下来就来从原理到实现,从简到难一步一步来实现这个SelfVue.由于本文只是为了学习和分享

vue中数据双向绑定的实现原理

vue中最常见的属v-model这个数据双向绑定了,很好奇它是如何实现的呢?尝试着用原生的JS去实现一下. 首先大致学习了解下Object.defineProperty()这个东东吧! * Object.defineProperty() * 对对象的属性进行 定义/修改 * */ let obj = {x:10} // 这两种方式都相对来说比较简单,直接,但是有些时候我们需要对对象的属性的修改和增加进行必要的干预 // obj.y = 20; // obj.x = 100; // obj.x =

VUE JS 使用组件实现双向绑定

1.VUE 前端简单介绍  VUE JS是一个简洁的双向数据绑定框架,他的性能超过ANGULARJS,原因是实现的机制和ANGULARJS 不同,他在初始化时对数据增加了get和set方法,在数据set时,在数据属性上添加监控,这样数据发生改变时,就会触发他上面的watcher,而ANGULARJS 是使用脏数据检查来实现的. 另外VUEJS 入门比ANGULARJS 简单,中文文档也很齐全. 2.组件实现    在使用vue开发过程中,我们会需要扩展一些组件,在表单中使用,比如一个用户选择器.

Angular2-无法给元素的属性做双向绑定,除非这个属性是指令或者组件

呈现一个实体对象的DOM结构,这个DOM有个自定义属性是需要动态赋值的,比如说 item.data 要绑到自定义属性 data 上:你可能会这么写: data="{{item.data}}" ,但直接用插值语法会报错的. Angular2无法给非指令或者组件做双向绑定. 如果非要这么做,这里有个这种方案:自定义一个指令:"data"专门接收"item.data"数据,然后在指令内部做操作,将data数据绑上去. 示例如下: @Directive(

vue 自定义组件 v-model双向绑定、 父子组件同步通信

父子组件通信,都是单项的,很多时候需要双向通信.方法如下: 1.父组件使用:msg.sync="aa"  子组件使用$emit('update:msg', 'msg改变后的值xxx') 2.父组件传值直接传对象,子组件收到对象后可随意改变对象的属性,但不能改变对象本身. 3.父组件使用: v-model 第一种曾经被废除过,由于维护成本的原因被删掉,但经过证实,确实有存在的意义,又被加上. 第一种: 父组件: <template> <div> <aa cl