Angular4.x Event (DOM事件和自定义事件详解)


Posted in Javascript onOctober 09, 2018

Angular组件和DOM元素通过事件与外部进行通信,两者中的事件绑定语法是相同的-(eventName)="expression":

<button (click)="onClick()">Click</button>

DOM元素

DOM 元素触发的一些事件通过 DOM 层级结构传播,事件首先由最内层的元素开始,然后传播到外部元素,直到它们到根元素,这种传播过程称为事件冒泡。

DOM事件冒泡与Angular可以无缝工作,具体示例如下:

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

@Component({
 selector: 'exe-app',
 template: `
 <div (click)="onClick()">
 <button>Click</button>
 </div>
 `
})
export class AppComponent {
 onClick() {
 console.log('Click');
 }
}

以上代码成功运行后,当用户点击 Click 按钮,浏览器控制台将会输出:

Click

即表示 <div> 元素上设置的监听函数被执行,也间接证明了事件冒泡能正常工作。

Angular Component

Angular 允许开发者通过 @Output() 装饰器和 EventEmitter 自定义事件。它不同于 DOM 事件,因为它不支持事件冒泡。

首先来看一下自定义组件如何监听DOM事件:

event-bubbling.component.ts

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

@Component({
 selector: 'event-bubbling',
 template: `
 <div>
 <button>Click</button>
 </div>
 `
})
export class EventBubblingComponent { }

app.component.ts

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

@Component({
 selector: 'exe-app',
 template: `
 <div>
 <event-bubbling (click)="onClick()"></event-bubbling>
 </div>
 `
})
export class AppComponent {
 onClick() {
 console.log('Click');
 }
}

以上代码成功运行后,当用户点击 Click 按钮,浏览器控制台也会输出 Click。表示我们的自定义组件,也是可以正常处理组件内元素触发的 click 事件。但当我们在自定义组件中创建自定义事件时,事件名称也是使用 click ,那么会不会有问题呢?我们马上来实践一下。

Custom Events

event-bubbling.component

import { Component, Output, EventEmitter } from '@angular/core';

@Component({
 selector: 'event-bubbling',
 template: `
 <div>
 <button (click)="onClick('Button 1')">Button 1</button>
 <button (click)="onClick('Button 2')">Button 2</button>
 </div>
 `
})
export class EventBubblingComponent {
 @Output() click = new EventEmitter();

 onClick(button: string) {
 this.click.next(button);
 }
}

app.component.ts

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

@Component({
 selector: 'exe-app',
 template: `
 <div>
 <event-bubbling (click)="onClick($event)"></event-bubbling>
 </div>
 `
})
export class AppComponent {
 onClick(event: any) {
 console.log(event);
 }
}

以上代码成功运行后,当用户点击 Button 1 按钮时,浏览器控制台将会输出:

Button 1 
MouseEvent {isTrusted: true, screenX: 69, screenY: 161, clientX: 43, clientY: 22…}

我们看到控制台输出了两个信息,说明在这种情况下,Angular 可以处理自定义事件和 DOM (click) 事件。当 click 事件触发后,onClick() 方法接收的 event 参数是 MouseEvent 对象。

那么要如何修复这个问题呢?我们可以利用 DOM 事件提供的防止冒泡的机制,即 stopPropagation() 方法。具体示例如下:

event-bubbling.component.ts

import { Component, Output, EventEmitter } from '@angular/core';

@Component({
 selector: 'event-bubbling',
 template: `
 <div>
 <button (click)="onClick($event, 'Button 1')">Button 1</button>
 <button (click)="onClick($event, 'Button 2')">Button 2</button>
 </div>
 `
})
export class EventBubblingComponent {
 @Output() click = new EventEmitter();

 onClick(event: Event, button: string) {
 event.stopPropagation();
 this.click.next(button);
 }
}

以上代码成功运行后,当用户点击 Button 1 按钮时,只会输出 Button 1 ,即问题我们已经修复了。

现在我们来总结一下:

DOM 事件冒泡机制,允许在父元素监听由子元素触发的 DOM 事件

Angular 支持 DOM 事件冒泡机制,但不支持自定义事件的冒泡。

自定义事件名称与 DOM 事件的名称如 (click,change,select,submit) 同名,可能会导致问题。虽然可以使用 stopPropagation() 方法解决问题,但实际工作中,不建议这样使用。

Event Modifiers

在实际项目中,我们经常需要在事件处理器中调用 preventDefault() 或 stopPropagation() 方法:

preventDefault() - 如果事件可取消,则取消该事件,意味着该事件的所有默认动作都不会发生。需要注意的是该方法不会阻止该事件的冒泡

stopPropagation() - 阻止当前事件在 DOM 树上冒泡

对于 preventDefault() 方法,有一个经典的应用场景。即当我们希望点击链接在新窗口打开页面,但不希望当前页面跳转。这个时候可以使用 preventDefault() 阻止后面将要执行的浏览器默认动作。

<a id="link" href="https://i.cnblogs.com/EditPosts.aspx?postid=7792556" rel="external nofollow" >Angular 4.x 事件冒泡</a>
<script>
 document.getElementById('link').onclick = function(ev) {
 ev.preventDefault(); // 阻止浏览器默认动作 (页面跳转)
 window.open(this.href); // 在新窗口打开页面
 };
</script>

在 Angular 中阻止 DOM 事件冒泡,我们可以使用以下两种方式:

方式一:

<div>
 <button (click)="$event.stopPropagation(); doSomething()">Click me</button>
</div>

方式二:

@Component({
 selector: 'exe-app',
 template: `
 <div>
 <button (click)="doSomething($event)">Click me</button>
 </div>`
})
export class AboutComponent {
 doSomething($event: Event) {
 $event.stopPropagation();
 // your logic
 }
}

是不是感觉很麻烦,每次都得显式地调用 $event.stopPropagation() 方法。有没有更简便的方法呢?能不能使用声明式的语法?在 Vue 中可以通过声明式的方式,解决我们上面的问题。具体如下:

<!-- the click event's propagation will be stopped -->
<a v-on:click.stop="doThis"></a>

<!-- the submit event will no longer reload the page -->
<form v-on:submit.prevent="onSubmit"></form>

<!-- the click event will be triggered at most once -->
<a v-on:click.once="doThis"></a>

接下来我们也来基于 Angular 的指令系统,实现上述的功能。最终的使用示例如下:

<div class="parent" (click)="fromParent()">
 <button class="child" (click.stop)="fromChild()">Click me</button>
</div>

自定义 [click.stop] 指令

@Directive({
 selector: '[click.stop]'
})
export class StopPropagationDirective {
 @Output("click.stop") stopPropEvent = new EventEmitter();
 unsubscribe: () => void;

 constructor(
 private renderer: Renderer2, // Angular 2.x导入Renderer
 private element: ElementRef) {
 }

 ngOnInit() {
 this.unsubscribe = this.renderer.listen(
  this.element.nativeElement, 'click', event => {
  event.stopPropagation();
  this.stopPropEvent.emit(event);
  });
 }

 ngOnDestroy() {
 this.unsubscribe();
 }
}

[click.stop] 指令应用

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

@Component({
 selector: 'exe-app',
 template: `
 <div class="parent" (click)="fromParent()">
 <button class="child" (click.stop)="fromChild()">Click me</button>
 </div>
 `
})
export class AppComponent {
 fromChild() {
 console.log('I am Child');
 }

 fromParent() {
 console.log('I am Parent');
 }
}

以上代码成功运行后,当用户点击 Click me 按钮时,浏览器控制台只会输出 I am Child。若把 (click.stop) 改为 (click) ,当用户再次点击 Click me 按钮时,控制台就会输出两条信息。有兴趣的读者,可以亲自验证一下哈。

这篇Angular4.x Event (DOM事件和自定义事件详解)就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
精选的10款用于构建良好易用性网站的jQuery插件
Jan 23 Javascript
js网页侧边随页面滚动广告效果实现
Apr 14 Javascript
JavaScript实现复制功能各浏览器支持情况实测
Jul 18 Javascript
Function.prototype.bind用法示例
Sep 16 Javascript
js实现瀑布流的一种简单方法实例分享
Nov 04 Javascript
JavaScript数值数组排序示例分享
May 27 Javascript
vue Render中slots的使用的实例代码
Jul 19 Javascript
gulp教程_从入门到项目中快速上手使用方法
Sep 14 Javascript
Vue与Node.js通过socket.io通信的示例代码
Jul 25 Javascript
深入了解JavaScript 的 WebAssembly
Jun 15 Javascript
Layui表格监听行单双击事件讲解
Nov 14 Javascript
js+css3实现简单时钟特效
Sep 13 Javascript
Vue官方推荐AJAX组件axios.js使用方法详解与API
Oct 09 #Javascript
vue router 源码概览案例分析
Oct 09 #Javascript
Angular4 Select选择改变事件的方法
Oct 09 #Javascript
对angular 监控数据模型变化的事件方法$watch详解
Oct 09 #Javascript
JS判断用户用的哪个浏览器实例详解
Oct 09 #Javascript
vue发送ajax请求详解
Oct 09 #Javascript
AngularJS 监听变量变化的实现方法
Oct 09 #Javascript
You might like
smarty内置函数capture用法分析
2015/01/22 PHP
基于CI框架的微信网页授权库示例
2016/11/25 PHP
HTA版JSMin(省略修饰语若干)基于javascript语言编写
2009/12/24 Javascript
jquery 倒计时效果实现秒杀思路
2013/09/11 Javascript
jquery自动填充勾选框即把勾选框打上true
2014/03/24 Javascript
原生javascript模仿win8等待提示圆圈进度条
2014/04/24 Javascript
javascript生成随机颜色示例代码
2014/05/05 Javascript
删除javascript中注释语句的正则表达式
2014/06/11 Javascript
JavaScript实现删除,移动和复制文件的方法
2015/08/05 Javascript
jQuery实现点击小图显示大图代码分享
2015/08/25 Javascript
基于Flowplayer打造一款免费的WEB视频播放器附源码
2015/09/06 Javascript
js中作用域的实例解析
2017/03/16 Javascript
JSON 数据格式详解
2017/09/13 Javascript
JavaScript强制类型转换和隐式类型转换操作示例
2019/05/01 Javascript
原生js实现日历效果
2020/03/02 Javascript
[01:00]选手抵达华西村 整装待发备战2016国际邀请赛中国区预选赛
2016/06/25 DOTA
Python多进程机制实例详解
2015/07/02 Python
Python黑帽编程 3.4 跨越VLAN详解
2016/09/28 Python
python使用SMTP发送qq或sina邮件
2017/10/21 Python
Python判断两个list是否是父子集关系的实例
2018/05/04 Python
python 循环读取txt文档 并转换成csv的方法
2018/10/26 Python
pytorch中tensor张量数据类型的转化方式
2019/12/31 Python
python 基于UDP协议套接字通信的实现
2021/01/22 Python
详解css3 mask遮罩实现一些特效
2018/10/24 HTML / CSS
详解canvas drawImage()方法绘制图片不显示的问题
2018/10/08 HTML / CSS
H5混合开发app如何升级的方法
2018/01/10 HTML / CSS
美国名牌太阳镜折扣网站:Eyedictive
2017/05/15 全球购物
益模软件Java笔试题
2012/03/27 面试题
英文简历中的自荐信范文
2013/12/14 职场文书
小学语文教学反思
2014/02/10 职场文书
大学计划书范文800字
2014/08/14 职场文书
党员学习党的群众路线思想汇报(5篇)
2014/09/10 职场文书
医院办公室主任岗位职责
2015/04/01 职场文书
公司宣传语大全
2015/07/13 职场文书
学生病假条范文
2015/08/17 职场文书
Python  lambda匿名函数和三元运算符
2022/04/19 Python