首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

如何订阅Angular ControlValueAccessor指令中的FormControl值更改

在Angular中,ControlValueAccessor是一个接口,它允许自定义表单控件与Angular的响应式表单系统进行交互。要订阅FormControl值的更改,你可以使用valueChanges observable,这是AbstractControl类(FormControl继承自它)提供的一个属性。

基础概念

  • ControlValueAccessor: 这是一个接口,用于创建自定义表单控件,使其能够与Angular的模板驱动表单和响应式表单系统无缝集成。
  • FormControl: 这是响应式表单中的一个基本构建块,代表表单控件的状态和值。
  • valueChanges: 这是一个observable,每当控件的值发生变化时,它都会发出新的值。

相关优势

  • 实时反馈: 通过订阅valueChanges,你可以实时获取表单控件的最新值,这对于实时验证或数据预处理非常有用。
  • 解耦: 使用ControlValueAccessor可以将表单控件的实现与其表单逻辑分离,使得组件更加模块化和可重用。

类型与应用场景

  • 自定义表单控件: 当你需要创建一个不在Angular标准控件库中的表单控件时,可以使用ControlValueAccessor
  • 第三方控件集成: 如果你要将第三方库的控件集成到Angular表单中,也可以通过实现ControlValueAccessor来实现。

示例代码

假设你有一个自定义的表单控件,并且你想订阅它的值更改。以下是如何实现的示例:

代码语言:txt
复制
import { Component, forwardRef } from '@angular/core';
import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';

@Component({
  selector: 'app-custom-input',
  template: `<input type="text" (input)="onChange($event.target.value)" (blur)="onTouched()">`,
  providers: [
    {
      provide: NG_VALUE_ACCESSOR,
      useExisting: forwardRef(() => CustomInputComponent),
      multi: true
    }
  ]
})
export class CustomInputComponent implements ControlValueAccessor {
  private onChange: (_: any) => void = () => {};
  private onTouched: () => void = () => {};

  writeValue(value: any): void {
    // 这里可以设置输入框的值
  }

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

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

  setDisabledState?(isDisabled: boolean): void {
    // 这里可以设置输入框的禁用状态
  }
}

在你的组件中订阅值更改:

代码语言:txt
复制
import { Component } from '@angular/core';
import { FormControl } from '@angular/forms';

@Component({
  selector: 'app-my-form',
  template: `<app-custom-input formControlName="customInput"></app-custom-input>`
})
export class MyFormComponent {
  customInputControl = new FormControl();

  constructor() {
    this.customInputControl.valueChanges.subscribe(value => {
      console.log('Value changed:', value);
      // 在这里处理值更改
    });
  }
}

遇到问题及解决方法

如果你在订阅valueChanges时遇到问题,比如没有收到更新,在检查以下几点:

  1. 确保ControlValueAccessor实现正确: 确认你的自定义控件正确实现了ControlValueAccessor接口的所有必要方法。
  2. 检查变更检测: 如果你的组件或服务不在Angular的变更检测策略范围内,可能会导致值更改不被检测到。确保你的组件是在Angular的变更检测范围内。
  3. 异步操作: 如果你在valueChanges的处理函数中执行了异步操作,确保你正确处理了这些操作,以免丢失更新。

通过以上步骤,你应该能够成功订阅并响应FormControl的值更改。

页面内容是否对你有帮助?
有帮助
没帮助

相关·内容

(转) 别再对 Angular 表单的 ControlValueAccessor 感到迷惑

首先我解释下为啥需要 ControlValueAccessor 接口以及它在 Angular 中是如何使用的。...FormControl 和 ControlValueAccessor 如果你之前使用过 Angular 表单,你可能会熟悉 FormControl ,Angular 官方文档将它描述为追踪单个表单控件值和有效性的实体对象...这个对象桥接原生表单控件和 formControl 指令,并同步两者的值。...下图是 Angular 表单控件 如何通过 ControlValueAccessor 来和原生表单控件交互的(译者注:formControl 和你写的或者 Angular 提供的 CustomControlValueAccessor...= new FormControl(3); } 所有表单指令,包括上面代码中的 formControl 指令,都会调用 setUpControl 函数来让表单控件和DefaultValueAccessor

3.8K20

Angular6自定义表单控件方式集成Editormd

曾经找到过“Editor.md”,看之心喜,一直想在Angular中集成下这款markdownpad编辑器玩,在网上也只找到一篇通过指令集成的,虽然可以实现,但还是希望能做成组件形式的,之后看到一篇自定义组件的文章...ControlValueAccessor 这是自定义表单组件的核心,只有继承这个接口,才有被 Angular的formControl识别的资格。...(isDisabled: boolean): void; } writeValue:在初始化的时候将formControl的值传递给原生表单控件(即,将模型中的新值写入视图或 DOM 属性中); registerOnChange...formControl指令的实现: // https://github.com/angular/angular/blob/master/packages/forms/src/directives/reactive_directives...指令调用了setUpControl函数来实现formControl和ControlValueAccessor之间的交互。

5.2K20
  • Angular 从入坑到挖坑 - 表单控件概览

    一、Overview angular 入坑记录的笔记第三篇,介绍 angular 中表单控件的相关概念,了解如何在 angular 中创建一个表单,以及如何针对表单控件进行数据校验。...在使用 ngModel 进行模板绑定时,angular 在 form 标签上自动附加了一个 NgForm 指令,因为 NgForm 指令会控制表单中带有 ngModel 指令和 name 属性的元素,而...,然后将控件组中的每一个控件作为属性值添加到实例中 import { Component, OnInit } from '@angular/core'; // 引入 FormControl 和 FormGroup...在模板驱动表单中,因为不是直接使用的 FormControl 实例,因此这里应该在模板上添加一个自定义的指令来完成对于控件数据的校验 使用 angular cli 创建一个用来进行表单验证的指令 ng...g directive direactives/hero-validate 在创建完成指令之后,我们需要将这个指令将该验证器添加到已经存在的验证器集合中,同时为了使这个指令可以与 angular 表单集成在一起

    18.9K20

    Angular系列教程-第四节

    1.表单 Angular 提供了两种不同的方法来通过表单处理用户输入:响应式表单和模板驱动表单。...两者都从视图中捕获用户输入事件、验证用户输入、创建表单模型、修改数据模型,并提供跟踪这些更改的途径 使用’@angular/forms’库中的FormGroup, FormControl,FormArray...,FormBuilder 等类构建出的数据对象就是响应式的表单,在响应式的表单中,我们会在数据源里面进行各种操作,像添加校验等,在html文件中使用 formGroup,formGroupName,formControlName...总结 响应式表单是动态的,模板驱动表单是固定的 2.响应式表单使用 注册ReactiveFormsModule 组件导入FormControl 模板中注册组件 控件的值(获取setvalue...使用ngModel实现双向绑定 使用模板变量来获取表单 4.内置验证器 min 此验证器要求控件的值大于或等于指定的数字 max 此验证器要求控件的值小于等于指定的数字 required 此验证器要求控件具有非空值

    2.8K50

    Angular2 :从 beta 到 release4.0 版本升级总结

    Angular 模块是带有 @NgModule 装饰器函数的类。 @NgModule 接收一个元数据对象,该对象告诉 Angular 如何编译和运行模块代码。...它标记出该模块拥有的组件、指令和管道, 并把它们的一部分公开出去,以便外部组件使用它们。 它可以向应用的依赖注入器中添加服务提供商。 具体请参考官方文档。.../commom'; => import {FormControl} from '@angular/forms'; 原使用[ngFormModel]属性 更改表单属性 [ngFormModel] 为 [...组件迁移后,无法正确订阅事件 原因:angular(v4.0.0)中依赖注入,若在不同地方声明provider,则会创建不同的实例。...原因:angular(v4.1.1)中,使用ActivatedRoute的API获取路由信息。

    8.2K00

    Angular v18 现已推出!

    '); handleClick() { this.name.set('Zoneless Angular'); }}在上面的示例中,单击按钮将调用该方法,该方法将更新信号值并更新 UI。...开发者预览版中的信号 API在 Angular 版本 17.1 和 17.2 中,我们宣布了新的信号输入、基于信号的查询和新的输出语法。在我们的信号指南中了解如何使用 API。...作为收敛努力的结果,Wiz 将 Angular Signals 深度集成到他们的渲染模型中。在 ng-conf 上,我们分享了 YouTube 现在如何使用 Angular Signals。...我们目前正在与合作伙伴合作,评估数据触发器的重要性,例如传递接收属性或更改绑定值的组件。...FormControl,Angular 窗体中的类现在公开一个名为 的属性,该属性允许您订阅此窗体控件的事件流。

    28010

    Angular 自定义属性指令

    本文将使用 UltimateAngular/angular-pro-src 中的示例,来一步步介绍自定义属性指令的相关知识。...要实现该需求,前提是我们能监听输入框的 input 事件,然后获取该输入框的值,在对输入的数字进行格式化处理。...此时,我们的 TooltipDirective 指令,已经包含了控制 tooltip 元素显示和隐藏的方法。那么现在的问题是,我们要如何访问 TooltipDirective 指令的实例。...表单模块中,也大量使用了 exportAs 属性,比如 ngModel、ngForm、ngModelGroup 及 formControl 指令等。...本文通过 CreditCardDirective 和 TooltipDirective 两个指令,介绍了 Angular 自定义属性指令所涉及的相关的基础知识,若想继续深入学习的话,可以阅读 Angular

    2K30

    AngularDart 4.0 高级-生命周期钩子 顶

    ngOnDestroy 在Angular摧毁指令/组件之前进行清理。 取消订阅observables并分离事件处理程序以避免内存泄漏。 在Angular摧毁指令/组件之前调用。...OnChanges 看看每次组件输入属性发生变化时,Angular如何用变更对象调用ngOnChanges钩子。 显示如何解释更改对象。...构造函数不应仅仅将初始局部变量设置为简单值。 ngOnInit是组件获取其初始数据的好地方。 教程和HTTP章节显示了如何。 还要记住,指令的数据绑定输入属性在构建之后才会设置。...日志条目显示为power属性更改的字符串值。 但ngOnChanges并没有捕捉到hero.name的变化,这一开始令人惊讶。 当输入属性的值改变时,Angular只会调用钩子。...以下AfterContent挂钩根据内容子代(只能通过使用@ContentChild注解的属性查询它)中的值进行更改。

    6.2K10

    Angular的12个经典问题,看看你能答对几个?(文末附带Angular测试)

    @angular/core会创建组件,渲染它,创建并呈现它的后代。当@angular/core的数据绑定属性更改时,处理就会更改,在从DOM中删除其模板之前,就会销毁掉它。...ngOnChanges:当Angular设置其接收当前和上一个对象值的数据绑定属性时响应。 ngOnInit:在第一个ngOnChange触发器之后,初始化组件/指令。...ngOnDestroy:在Angular销毁指令/组件之前清除。取消订阅可观察的对象并脱离事件处理程序,以避免内存泄漏。...在Angular2中,组件中发生的任何改变总是从当前组件传播到其所有子组件中。如果一个子组件的更改需要反映到其父组件的层次结构中,我们可以通过使用事件发射器api来发出事件。...这通常用在setter中,当类中的值被更改完成时。 可以通过模块的任何一个组件,使用订阅方法来实现事件发射的订阅。

    17.4K80

    Angular 18 引入了 Zoneless 变更检测

    开发人员可以通过在其应用程序的引导程序中添加如下的提供程序来尝试实验性的 zoneless 支持: bootstrapApplication(App, { providers: [ provideExperimentalZonelessChangeDetection...Angular.dev 是 Angular 文档的官方网站。其中包含了动手入门之旅、互动游乐场、更新的指南和简化的导航。所有对 angular.io 的请求现在都重定向到了 angular.dev。...现在可以在 Angular 18 中为 ng-content 指定默认的内容。这允许开发人员在他们的组件中提供回退内容。...表单现在公开了一个名为 events 的属性,允许开发人员订阅表单控件的事件流。...例如: const nameControl = new FormControl('name', Validators.required); nameControl.events.subscribe(event

    26410

    AngularDart4.0 指南- 模板语法二 顶

    谁可以记住要设置哪个元素属性以及哪个元素事件发出用户更改? 如何从输入框中提取当前显示的文本,以便更新数据属性? 谁想每一次都看看?...Angular为所有基本的HTML表单元素提供值访问器,Forms指南展示了如何绑定到它们。...这些元素的所有组件都保留在内存中,Angular可能会继续检查更改。 您的应用可能会占用相当可观的计算资源,会降低用户不可见的性能。...您可以定义一个HTML块来定义应该如何显示单个项目。 您告诉Angular将该块用作呈现列表中每个项目的模板。...heroForm的值是什么? heroForm是一个Angular NgForm指令的引用,可以跟踪表单中每个控件的值和有效性。 原生元素没有form属性。

    30K20

    使用.net core ABP和Angular模板构建博客管理系统(实现编辑页面功能)

    编辑模块 我们看看作者的新建页面用到了什么, 用到了ngx-bootstrap的弹出层。...返回列表后也没有自动更新 预览处也没实现实时预览 自动更新也没有实现 发布功能还没有实现 界面不够美观 返回列表更新 这个就要用到angular的父子页面传值。...测试父子页面传值 实现实时预览 我们之前设计是使用markdown语法来制作这个编辑功能。...预览有了,但是没有实时同步 要实现实时同步,我们使用angular的FormControl来帮忙 import { FormControl } from '@angular/forms'; import...1.gif 好项目是慢慢优化出来的,一口是吃不出一个大胖子来的,慢慢优化,一步步行动起来,才能遇见更好的自己。 在操作等待的时候没有遮罩层,这种体验很不好。 操作成功或者失败也没有提示。

    1K30

    AngularDart Material Design 输入 顶

    它有可选的标签。注意:客户端必须在其指令列表中声明materialInputDirectives而不是MaterialInputComponent。...将此设置为true会更改行为,以便在更改选项或选项时:       1.选择中的第一个选定值在选项中有效       2.如果选择没有选定值,则选项中没有任何活动 inputText String...请参阅Filterable中的过滤方法。默认为10。 loading bool  打开时没有可用的建议,请在建议下拉列表中显示加载指示符。...使用materialNumberInputDirectives获取一组与输入一起使用以提供验证的指令。...Accessor始终设置从输入设置的原始String值,但仅在可以解析输入时设置Control的值。 keypressUpdate属性在每个按键上都有值更新,而默认值是仅在模糊事件上更新的值。

    5.3K40

    【17】进大厂必须掌握的面试题-50个Angular面试

    Angular中的过滤器用于格式化表达式的值,以便将其显示给用户。这些过滤器可以添加到模板,指令,控制器或服务中。不仅如此,您还可以创建自己的自定义过滤器。...Angular中的提供程序是什么? 提供程序是Angular中的可配置服务。这是对依赖关系注入系统的一条指令,它提供有关获取依赖关系值的方式的信息。...您可以使用此钩子来取消订阅可观察对象并分离事件处理程序,以避免发生任何类型的内存泄漏。 31.通过对Angular进行脏检查,您了解什么? 在Angular中,摘要过程称为脏检查。...之所以调用它,是因为它扫描整个范围以进行更改。换句话说,它将所有新的作用域模型值与以前的作用域值进行比较。...被监视的变量处于单个循环(摘要循环)中,任何变量的任何值更改都会在DOM中重新分配其他被监视变量的值 32.区分DOM和BOM。

    41.5K51

    VUE2.0如何追踪数据变化?

    Angular 1 中,采用脏检查机制,缺点是:当watcher越来越多时,作用域内每一次变化,所有watcher都要重新计算。...Watcher:某个属性数据的监听者/订阅者,一旦数据有变化,它会通知指令(directive)重新编译模板并渲染UI。...数据对象的每个属性,都包含一个Dep实例对象,用于存储关心该属性变化的watchers。 在model--->UI渲染过程中,通过数据属性的get函数,可以添加相对应的watcher到Dep对象中。...当触发UI更新操作(比如,input框输入某些内容),即UI--->Model--->UI这个过程中,首先触发对应数据属性的set函数,然后订阅者容器Dep对象发布消息通知notify,随后,所有订阅者...$el.textContent === 'new message' // true 在下一个Tick中,DOM节点才会更新 }) 小结 一句话总结Vue.js如何实现数据双向绑定:通过ES5新特性Object.defineProperty

    1.2K20
    领券