[Angular 基础] - 自定义指令,深入学习 directive

news2025/1/12 1:06:26

[Angular 基础] - 自定义指令,深入学习 directive

这篇笔记的前置笔记为 [Angular 基础] - 数据绑定(databinding),对 Angular 的 directives 不是很了解的可以先过一下这篇笔记

后面也会拓展一下项目,所以感兴趣的也可以补一下文后对应的项目:

  • 第一个 Angular 项目 - 静态页面
  • 第一个 Angular 项目 - 动态页面

创建新 directive

directive 的创建方式和 component 类似,这里选择用指令生成:

❯ ng g d directives/test --skip-tests
CREATE src/app/directives/test.directive.ts (137 bytes)
UPDATE src/app/app.module.ts (757 bytes)

运行这个指令就会在 src/app/directives 下创建一个新的 directive:

在这里插入图片描述

目前项目结构如下,这里 V 和 VM 层暂时不用去管,directive 会一个个过一遍

⚠️:如果手动生成 directive,同样需要在 app.module.ts 中导入对应的 directive:

@NgModule({
  declarations: [
    AppComponent,
    BasicHighlightDirective,
    BetterHighlightDirective,
  ],
  imports: [BrowserModule, FormsModule],
  providers: [],
  bootstrap: [AppComponent],
})
export class AppModule {}

实现一个 attribute directive

一个空白的 directive 结构如下:

import { Directive } from '@angular/core';

@Directive({
  selector: '[appBasicHighlight]',
})
export class BasicHighlightDirective {
  constructor() {}
}

首先分析一下 directive 的结构,这里的 @Directive@Component 一样都是装饰器,不过这里的使用比较简单,只是放入了一个 selector

selector 中的内容为绑定对应的 HTML Template 中的 attribute,也就是说 HTML template:

  • ✅ 使用 appBasicHighlight,就能够绑定对应的 directive
  • ❌ 不使用 [appBasicHighlight] 去进行属性绑定

这时候修改 app 的 VM 层:

<p appBasicHighlight>Style me with basic directive</p>
`

简单的 attribute directive

directive 的构造函数是比较重要的,它总共会提供 3 个参数用来操控对应的 DOM

使用 ElementRef

这里的 ElementRef 是对当前绑定指令的 HTML 元素的引用值,这里也就是 p 标签中的内容

通过直接修改 ElementRef 也是一种可以直接修改 DOM 元素的方式,使用方法如下:

@Directive({
  selector: '[appBasicHighlight]',
})
export class BasicHighlightDirective implements OnInit {
  constructor(private elementRef: ElementRef) {}

  ngOnInit() {
    console.log(this.elementRef);
    this.elementRef.nativeElement.style.backgroundColor = 'pink';
  }
}

效果如下:

在这里插入图片描述

这里简述一下修改:

  • implements OnInit 这一段算是补充吧

    尽管说 TS 的 implements 执行力比较差,不过我看了下官方文档都有用,就稍微标准化一下好了

  • ngOnInit 中执行对于样式的修改

    本案例使用 ngOnInit 或者直接在构造函数里修改样式其实没有什么特别大的区别,不过对于其他的情况,例如 HTML 中的内容是动态生成的情况下,直接在构造函数里就会造成内容的缺失。

通过这种方式就能够创建一个 attribute directive 了,不过直接使用 elementRef 并不是一个推荐的做法,下面会创建另一个 attribute directive,并使用推荐的方式去修改属性

复杂一些的 attribute directive

这里新建一个 directive,并将其命名为 better-highlight.directive,同时在 app 的 V 层新建一个 p 标签,并添加对应的 attribute directive:

<p appBetterHighlight>Style me with better directive</p>
renderer
import { Directive } from '@angular/core';

@Directive({
  selector: '[appBetterHighlight]',
})
export class BetterHighlightDirective {
  constructor(private renderer: Renderer2) {}
}

这里要修改样式的方法是通过这个 renderer 去实现的

renderer: Renderer2 是 Angular 对 DOM 操作的一个 service 封装,其主要的优点在于提供统一的 API 使得其在浏览器坏境、SSR 环境以及 web worker 中都会有同样的表现。另外它也会对一些 HTML 元素进行清理,这样可以更觉有效的防止 XSS 攻击。

对于同样修改样式,这里依旧在 ngOnInit 中实现,实现的方式是通过调用 setStyle 进行:

@Directive({
  selector: '[appBetterHighlight]',
})
export class BetterHighlightDirective implements OnInit {
  // @Input
  constructor(private elementRef: ElementRef, private renderer: Renderer2) {}

  ngOnInit() {
    this.renderer.setStyle(
      this.elementRef.nativeElement,
      'background-color',
      'lightblue'
    );
  }
}

显示效果如下:

在这里插入图片描述

@HostListener

@HostListener 是 Angular 提供的,对当前元素所提供的事件绑定的一个装饰器,其语法如下:

@HostListener('event_name', ['$event'])
methodName(event: EventType): void {}

其中 :

  • event_name 为事件名称,如 click, mouseenter

  • $event 为对应的事件

    事件前添加 $ 算是 Angular 约定俗成的一种规范了

  • methodName,如其名,函数名

  • EventType,事件类型,可以不传,TS 用来做规范的

这里用 mouseentermouseleave 为例,对背景颜色进行修改,修改后代码如下:

@Directive({
  selector: '[appBetterHighlight]',
})
export class BetterHighlightDirective implements OnInit {
  constructor(private elementRef: ElementRef, private renderer: Renderer2) {}

  ngOnInit() {
    this.renderer.setStyle(
      this.elementRef.nativeElement,
      'background-color',
      'lightblue'
    );
  }

  @HostListener('mouseenter', ['$event']) mouseover(eventData: MouseEvent) {
    console.log(eventData);

    this.renderer.setStyle(
      this.elementRef.nativeElement,
      'background-color',
      'lightgreen'
    );
  }

  @HostListener('mouseleave') mouseleave(eventData: Event) {
    console.log(eventData);

    this.renderer.setStyle(
      this.elementRef.nativeElement,
      'background-color',
      'lightblue'
    );
  }
}

效果如下:

在这里插入图片描述

这里主要注意的是这么几个点:

  • 如果不传递 ['$event'],那么函数也不会自动监听到对应的事件

    这也是为什么 mouseenter 的时候能抓到 $event,但是 mouseleave 的时候抓不到的原因

  • Event 只是类型检查

    不是说没用,相反,如果确定类型的话,那么 TS 将会提供更好的类型检查和 intelligence 提示

    但是如果要写比较 generic 的方案,可能还是直接用 Event 比较好

💡:我添加了 CSS transition,让背景色过渡的稍微自然点,不过这个不是什么重点

@HostBinding

这个时候看到组件内出现了很多的重复代码:

this.renderer.setStyle(
  this.elementRef.nativeElement,
  'background-color',
  `${color}`
);

可以看到,这里唯一产生变化的只有需要被修改的颜色。

要解决这个问题,可以使用 Angular 提供的 @HostBinding 装饰器,它的语法为:

@HostBinding('property') propertyName: Type = value;

其中:

  • property 为想要绑定的元素属性

    这个案例中就是 style.backgroundColor

  • propertyName 为变量名

  • Type 为类型

    本案例为 string,其他的案例可能会出现 number, boolean,如果是可选项的话也可以为 undefined

  • value

使用如下:

export class BetterHighlightDirective implements OnInit {
  @HostBinding('style.backgroundColor') backgroundColor: string = 'lightblue';

  constructor(private elementRef: ElementRef, private renderer: Renderer2) {}

  ngOnInit() {
    // 已经有了默认值,这下面的代码也可以注释掉了
    // this.backgroundColor = 'lightblue';
  }

  @HostListener('mouseenter', ['$event']) mouseover(eventData: MouseEvent) {
    console.log(eventData);

    this.backgroundColor = 'lightgreen';
  }

  @HostListener('mouseleave') mouseleave(eventData: Event) {
    console.log(eventData);

    this.backgroundColor = 'lightblue';
  }
}

最后展示的效果依旧是一样的

动态添加属性

这个时候可以注意到,现在唯一要修改的地方就是颜色,这个情况也可以使用变量去存储这个修改的颜色,同时前面可以添加 @Input,这样可以让父元素动态重写颜色:

export class BetterHighlightDirective implements OnInit {
  @Input() defaultColor: string = 'lightblue';
  @Input() highlightColor: string = 'lightgreen';
  @HostBinding('style.backgroundColor') backgroundColor: string =
    this.defaultColor;

  constructor(private elementRef: ElementRef, private renderer: Renderer2) {}

  @HostListener('mouseenter', ['$event']) mouseover(eventData: MouseEvent) {
    console.log(eventData);
    this.backgroundColor = this.highlightColor;
  }

  @HostListener('mouseleave') mouseleave(eventData: Event) {
    console.log(eventData);

    this.backgroundColor = this.defaultColor;
  }
}

在不重写默认颜色时,效果是一样的。

但是父元素也可以选择重写默认值:

<p
  appBetterHighlight
  class="bg-transition"
  [defaultColor]="'lightyellow'"
  highlightColor="violet"
>
  Style me with better directive
</p>

效果如下:

在这里插入图片描述

这里又有两个点需要注意的:

  • highlightColor="violet"

    这是一个特殊的语法缩写,本质上还是一个 property binding,而不是 HTML 所有的原生属性

    我这里特地用了两种写法,只是为了添加一下 note。为了更好的阅读性和理解,还是推荐使用 [customPropertyName]="'value'" 的写法

  • 背景颜色默认为蓝色

    这就是前面提到的 ngOnInit 的作用,这个情况下 Angular 的组件需要经历一个初始化的状态,在这个初始化的状态,它会绑定对应的属性——包括来自外部的属性

    这一段代码里我特地把 ngOnInit 注释掉了,没有了这个初始化的状态,那么当前组件依旧接受默认值,也就是 lightblue,把 ngOnInit 加回去,并添加对应的修改:

    ngOnInit() {
      this.backgroundColor = this.defaultColor;
    }
    

    才能将默认的背景色重写为父元素传进来的 lightyellow

实现一个 structural directive

官方文档有一个实现了 unless 的 structural directive,也可以参考一下,我这里就写一个 loading spinner 了。

实现如下:

@Directive({
  selector: '[appLoading]',
})
export class LoadingDirective {
  private loadingSpinner: HTMLElement;

  constructor(
    private templateRef: TemplateRef<any>,
    private viewContainer: ViewContainerRef,
    private renderer: Renderer2,
    private el: ElementRef
  ) {
    this.loadingSpinner = this.renderer.createElement('div');
    this.renderer.addClass(this.loadingSpinner, 'spinner');
  }

  @Input() set appLoading(isLoading: boolean) {
    this.viewContainer.clear();
    if (isLoading) {
      this.renderer.appendChild(
        this.el.nativeElement.parentElement,
        this.loadingSpinner
      );
    } else {
      this.renderer.removeChild(
        this.el.nativeElement.parentElement,
        this.loadingSpinner
      );
      this.viewContainer.createEmbeddedView(this.templateRef);
    }
  }
}

app 的 V 层:

<div *appLoading="isLoading">
  some random syntax showing only when loading is false
</div>

VM 层修改变量,以及 CSS 我就不贴代码了

效果如下:

在这里插入图片描述

下面进入实现的分析部分

TemplateRef

即绑定 structural directive 的元素,在这个情况下,就是:

<div *appLoading="isLoading">
  some random syntax showing only when loading is false
</div>

简单的理解就是,当满足特定条件时,这里需要渲染的内容

ViewContainerRef

ViewContainerRef 就是管理渲染内容的容器

Angular 没有 virtual DOM,但是又不想直接暴露 DOM 进行操作,因此就像 ElementRef 一样,它对将整个 视图(view) 进行了一个抽象,创建了 ViewContainerRef 以方便管理与 directive 绑定的,整个 DOM 的 view/template

setter

set 是一个 JavaScript 的语法糖,这也是 ES6 后出现的语法,与 Angular 无关。

这里 Angular 动态的将其 setter 和 @Input 进行绑定,并且提供一个更加直观且简洁明了的方式对当前与 @Input 绑定的值进行变化管理。如果不使用 setter 的话,也可以在对应的 ngOnChangesngOnInit 中监听值的变化,并且进行对应的操作。

整个 setter 中做的操作就分为两步:

  1. isLoading = true

    这个情况下需要渲染一个 loading spinner——这在构造函数中就已经创建好了,并且使用 renderer 去进行渲染

    当前的 nativeElement 指向的是 <div *appLoading="isLoading"></div> 这个具体的元素

    因此这里的操作就是在 nativeElement 的父元素下,新增一个 loading spinner

  2. isLoading = false

    这个情况下 loading spinner 被移除,原本的 template view 被渲染

💡:至于选用 renderer 就是因为在 attribute directive 部分已经讲过了,而且实现起来比较方便。我找了一下不用 renderer 渲染的方式,需要用到 ComponentFactoryResolver,这个暂时就还没学上,等之后学上了再说

structural directive 幕后

如果看了官方文档就会知道,<div *ngIf="hero" class="name">{{hero.name}}</div> 这样的语法是一个缩写,本质上它的实现方法如下:

<ng-template [ngIf]="hero">
  <div class="name">{{hero.name}}</div>
</ng-template>

随后就像上面写过的 structural directive 的实现一样,ngIf 通过属性绑定被 @Input 监听到,structural component 再根据业务逻辑进行制定渲染。

ng-template 本身也是一个 view 的 placeholder,它是不会被渲染的

下面是官方文档关于 ngFor 的实现:

<div
  *ngFor="let hero of heroes; let i=index; let odd=odd; trackBy: trackById"
  [class.odd]="odd"
>
  ({{i}}) {{hero.name}}
</div>

<ng-template
  ngFor
  let-hero
  [ngForOf]="heroes"
  let-i="index"
  let-odd="odd"
  [ngForTrackBy]="trackById"
>
  <div [class.odd]="odd">({{i}}) {{hero.name}}</div>
</ng-template>

也就是鉴于这样的转化,所以 structural directive 的语法一定是 *[directive_name]="[variable]" 这样的实现

第一个 Angular 项目——实现下拉框

项目里面没导入对应的 js 文件,所以现在 bootstrap 的 dropdown 是没办法被触发的。这里就是用 directive 去解决这个问题

新建 directive

❯ ng generate directive directives/dropdown --skip-tests
CREATE src/app/directives/dropdown.directive.ts (145 bytes)
UPDATE src/app/app.module.ts (1200 bytes)

实现 dropdown directive

这里需要了解一下 bootstrap 的 dropdown 是怎么被展开的——实际上是通过 open 这样一个 class 去实现的。因此,当 class 为 btn-group open 时,下拉框时展开的,当 class 为 btn-group 时,下拉框时关闭的。

所以这里需要实现一个 attribute directive,去管理 class 即可

最初的实现方法为:

export class DropdownDirective {
  constructor(private el: ElementRef, private renderer: Renderer2) {}

  @HostListener('click')
  onClick() {
    if (this.el.nativeElement.classList.contains('open')) {
      this.renderer.removeClass(this.el.nativeElement, 'open');
    } else {
      this.renderer.addClass(this.el.nativeElement, 'open');
    }
  }
}

不过一个简化的方法是使用 @HostBinding 去进行操作:

export class DropdownDirective {
  @HostBinding('class.open') isOpen = false;

  @HostListener('click')
  onClick() {
    this.isOpen = !this.isOpen;
  }
}

二者实现的效果是一样的:

在这里插入图片描述


补充一下点击 HTML 任何地方关闭 dropdown 的实现:

export class DropdownDirective {
  @HostBinding('class.open') isOpen = false;

  @HostListener('document:click', ['$event'])
  onClick(evemt: MouseEvent) {
    this.isOpen = this.el.nativeElement.contains(evemt.target)
      ? !this.isOpen
      : false;
  }

  constructor(private el: ElementRef) {}
}

在这里插入图片描述

reference

没有特殊标注的都是来自官方文档的内容

  • Structural directives
  • TemplateRef
  • ViewContainerRef

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1459690.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

齐次方程是否有非零解,和它的系数矩阵行列式的关系 (done)

视频来源&#xff1a;https://www.bilibili.com/video/BV1vY4y1J7gd/?spm_id_from333.337.search-card.all.click&vd_source7a1a0bc74158c6993c7355c5490fc600 4:22 有这么一句话&#xff0c;如下图 对于齐次方程&#xff0c;若系数矩阵的行列式为零&#xff0c;则方程…

常见消息中间件分享

文章目录 概念核心角色作用&使用场景应用解耦异步通信削峰填谷大数据流处理 使用模型点对点模型发布-订阅模型 常见消息中间件介绍一、kafka二、RabbitMQ三、RocketMQ 比较一、Kafka如何实现高吞吐量二、RocketMQ如何实现事务消息 概念 消息中间件是基于队列与消息传递技术…

14. rk3588自带的RKNNLite检测yolo模型(python)

首先将文件夹~/rknpu2/runtime/RK3588/Linux/librknn_api/aarch64/下的文件librknnrt.so复制到文件夹/usr/lib/下&#xff08;该文件夹下原有的文件librknnrt.so是用来测试resnet50模型的&#xff0c;所以要替换成yolo模型的librknnrt.so&#xff09;&#xff0c;如下图所示&am…

【办公类-16-07-03】“2023下学期 周计划-户外游戏 每班1周五天相同场地,6周一次循环、有场地、贴墙版”(python 排班表系列)

作品展示——有场地说明 背景需求&#xff1a; 前期做了一份“贴周计划”用的班主任版的户外游戏安排表&#xff08;中X班19周&#xff0c;没有场地&#xff09; 【办公类-16-07-02】“2023下学期 周计划-户外游戏 每班1周五天相同场地&#xff0c;6周一次循环”&#xff08;…

击败.helper勒索病毒:恢复被加密的数据文件的方法

导言: 近年来&#xff0c;勒索病毒成为网络安全领域的一大威胁&#xff0c;其中.helper勒索病毒更是备受关注。该类型的勒索软件以其高效的加密算法&#xff0c;能够将用户的文件加密&#xff0c;迫使用户支付赎金才能解密数据。本文将介绍.helper勒索病毒的特点、恢复被加密数…

unity学习(30)——跳转到角色选择界面(跳转新场景)

1.在scene文件夹中&#xff08;[siːn]&#xff09;&#xff0c;右键->create->scene&#xff0c;名字叫SelectMenu&#xff08;选择角色场景&#xff09;。 2.把新建场景拖拽到hierarchy[ˈhaɪərɑːki]中。 3.此时才能在file->build setting中Add open scene&…

vue页面基本增删改查

练手项目vue页面 新手前端轻喷&#xff1a; 效果如下 1、2两个部分组成&#xff1a; 对应代码中 element-ui中的 el-form 和 el-table 照着抄呗&#xff0c;硬着头皮来&#xff01; 建议&#xff1a;认真读一遍你用的组件 这篇文章烂尾了&#xff0c;对不起大家

MySQL安装教程(详细版)

今天分享的是Win10系统下MySQL的安装教程&#xff0c;打开MySQL官网&#xff0c;按步骤走呀~ 宝们安装MySQL后&#xff0c;需要简单回顾一下关系型数据库的介绍与历史&#xff08;History of DataBase&#xff09; 和 常见关系型数据库产品介绍 呀&#xff0c;后面就会进入正式…

计算机视觉基础:【矩阵】矩阵选取子集

OpenCV的基础是处理图像&#xff0c;而图像的基础是矩阵。 因此&#xff0c;如何使用好矩阵是非常关键的。 下面我们通过一个具体的实例来展示如何通过Python和OpenCV对矩阵进行操作&#xff0c;从而更好地实现对图像的处理。 示例 示例&#xff1a;选取矩阵中指定的行和列的…

总结Rabbitmq的六种模式

RabbitMQ六种工作模式 RabbitMQ是由erlang语言开发&#xff0c;基于AMQP&#xff08;Advanced Message Queue 高级消息队列协议&#xff09;协议实现的消息队列&#xff0c;它是一种应用程序之间的通信方法&#xff0c;消息队列在分布式系统开发中应用非常广泛。 RabbitMQ有六…

请介绍一下美国德克萨斯州历史地理与环境经济文化与社会教育与科研结论请介绍一下罗格斯大学(Rutgers University)校园与分校学术项目研究与创新学

目录 请介绍一下美国德克萨斯州 历史 地理与环境 经济 文化与社会 教育与科研 结论 请介绍一下罗格斯大学&#xff08;Rutgers University&#xff09; 校园与分校 学术项目 研究与创新 学生生活 社会贡献 请介绍一下德州A&M大学系统 组成成员 教育与研究 …

板块一 Servlet编程:第六节 HttpSession对象全解 来自【汤米尼克的JAVAEE全套教程专栏】

板块一 Servlet编程&#xff1a;第六节 HttpSession对象全解 一、什么是HttpSessionSession的本质 二、创建Seesion及常用方法三、Session域对象四、Session对象的销毁 在上一节中&#xff0c;我们学习了Servlet五大对象里的第三个Cookie对象&#xff0c;但Cookie是有大小限制和…

关于开放系统互联的一些笔记

最近几天就发几篇计算机方面的基础知识 属于个人归纳整理&#xff0c;便于理解希望对大家有帮助 原文地址&#xff1a;关于开放系统互联的一些笔记 - Pleasure的博客 下面是正文内容&#xff1a; 前言 最近在恶补一些计算机方面的基础知识…… 正文 首先为了能够更透彻的理…

VSCode C/C++无法跳转到定义(又是你 clangd !)

原博客&#xff1a;VSCode C/C无法跳转到定义、自动补全、悬停提示功能_c/c:edit configurations(json)-CSDN博客 我在此基础上加一点&#xff1a; 首先确保自己有这个插件&#xff1a; 点击 齿轮⚙ 符号&#xff0c;进入 配置设置&#xff0c;找到 把 C_cpp : Intelli Sens…

【实现100个unity特效之4】Unity ShaderGraph使用教程与各种特效案例(2023/12/1更新)

文章目录 一、前言二、ShaderGraph1.什么是ShaderGraph2.在使用ShaderGraph时需要注意以下几点&#xff1a;3.优势4.项目 三、实例效果边缘发光进阶&#xff1a;带方向的菲涅尔边缘光效果裁剪进阶 带边缘色的裁剪溶解进阶 带边缘色溶解卡通阴影水波纹积雪效果不锈钢效果、冰晶效…

AR智能眼镜主板硬件设计_AR眼镜光学方案

AR眼镜凭借其通过导航、游戏、聊天、翻译、音乐、电影和拍照等交互方式&#xff0c;将现实与虚拟进行无缝融合的特点&#xff0c;实现了更加沉浸式的体验。然而&#xff0c;要让AR眼镜真正成为便捷实用的智能设备&#xff0c;需要解决一系列技术难题&#xff0c;如显示、散热、…

钉钉小程序无法关联应用

钉钉小程序无法关联应用 problem 后台创建了新的应用钉钉小程序开发者工具>企业内部应用>关联应用 没有下拉列表 无法关联&#xff0c;只能点击新增按钮&#xff0c;重新进入后台很奇怪&#xff0c;明明创建好了应用&#xff0c;为什么关联下拉列表没有这个应用呢&…

在VS里使用C#制作窗口应用

新建项目 创建项目的时候搜索net&#xff0c;选择这个。 打开应该是这样 第一个控件 选择公共控件 - PictureBox - 拖入Form 在Image处选择上传本地资源&#xff0c;建议上传一个小一点的图片。 修改一下尺寸。 ctrls 保存 从“属性”切换到“事件” 双击Click事件…

STM32-开发工具

开发过程中可能用到的工具 1、烧录下载调试工具ST-LINK ST-LINK&#xff0c;是ST(意法半导体)推出的调试编程工具&#xff0c;适用于STM32系列芯片的USB接口的下载及在线仿真器。 2、串口调试工具/串口下载工具 串口调试工具是一种用于通过串口通信协议与目标设备进行数据交…

Linux——进程替换

&#x1f4d8;北尘_&#xff1a;个人主页 &#x1f30e;个人专栏:《Linux操作系统》《经典算法试题 》《C》 《数据结构与算法》 ☀️走在路上&#xff0c;不忘来时的初心 文章目录 一、进程程序替换1、替换原理2、替换函数3、函数解释4、命名理解 二、用例测试1、execl测试2、…