一个因@click.stop引发的bug的解决


Posted in Javascript onJanuary 08, 2019

问题

在项目页面中使用 element popover,设置trigger='click'时点击外部不会触发自动隐藏,但在 element 官网中是可以正常触发的(官方示例),项目中的菜单是自定义写的,所以怀疑是有黑魔法。

查找原因

  1. 将 popover 写在app.vue根组件内,发现可以正常触发自动隐藏。
  2. 在app.vue的 mounted 钩子中加入window.addEventListener('click', () => console.log('window click===>>>>')),发现只有菜单栏外层能够触发。
  3. 检查菜单栏组件,发现代码中<div class="main" @click.stop="isShowWhole = false">,这里的 click 事件使用了 stop 修饰符(阻止冒泡),可能阻止了 popover 外部点击的事件判断,尝试将 stop 修饰符去掉,发现外部点击事件正常触发。

确认代码修改没有副作用

在修复 bug 时,需要注意不会产生额外的 bug,那就需要了解修改的这段代码的含义

@click.stop="isShowWhole = false"

从代码上看,点击 class 为 main 的 div 将会触发左边侧边栏缩略显示,加上 stop 修饰符是为了防止事件冒泡,所以能否去掉 stop 需要确认是否有这个必要。

// router.js
let routes = [
  {
   path: '/',
   alias: '/admin',
   component: Menu,
   children: [...Pages],
  },
  {
   path: '*',
   name: '404',
   component: NotFound,
  },
 ];

在路由中可以看到,Menu 是作为根路由进行渲染,除了 404 页面都是它的子路由,所以 stop 修饰符是没有必要加上的,去除后经过测试没有其他影响。

深入 element popover 源码分析原因

对 element 组件进行 debug 时,可以直接引入相关组件的源码

import ElPopover from 'element-ui/packages/popover';
export default {
  components: {
    CheckboxFilter,
    ElPopover
  },
  ...
}

然后我们就可以在node_modules的 element 源码进行 debug 操作(危险步骤,debug 后需要复原)。

// node_modules/element-ui/packages/popover/src/main.vue
mounted() {
  ...
  if (this.trigger === 'click') {
   on(reference, 'click', this.doToggle);
   on(document, 'click', this.handleDocumentClick);
  } else if (this.trigger === 'hover') {
   ...
  } else if (this.trigger === 'focus') {
   ...
  }
}

popover 在 mounted 钩子内初始化了trigger='click'的事件绑定,on(document, 'click', this.handleDocumentClick)这里绑定了 document 很可能就是阻止事件冒泡后不能触发外部点击隐藏的判断逻辑。

// node_modules/element-ui/packages/popover/src/main.vue
handleDocumentClick(e) {
 let reference = this.reference || this.$refs.reference;
 const popper = this.popper || this.$refs.popper;

 if (!reference && this.$slots.reference && this.$slots.reference[0]) {
  reference = this.referenceElm = this.$slots.reference[0].elm;
 }
 if (!this.$el ||
  !reference ||
  this.$el.contains(e.target) ||
  reference.contains(e.target) ||
  !popper ||
  popper.contains(e.target)) return;
 this.showPopper = false;
},

这里判断this.$el是否包含 click 的 target,从而是否触发this.showPopper = false,当菜单栏阻止事件冒泡后 document 不能监听到 click 事件,才会无法进行外部点击隐藏的判断逻辑。

延伸v-clickoutside

element 的 select 组件中用到了 v-clickoutside 自定义指令,作用和 popover 的 handleDocumentClick 差不多(倒不如说 handleDocumentClick 是特殊的 clickoutside)

在上面的问题中,我们单独把 v-clickoutside 抽出来使用确实可以的,这是为什么呢?

// node_modules/element-ui/packages/popover/src/utils/clickoutside.js
!Vue.prototype.$isServer && on(document, 'mousedown', e => (startClick = e));

!Vue.prototype.$isServer && on(document, 'mouseup', e => {
 nodeList.forEach(node => node[ctx].documentHandler(e, startClick));
});

答案是 v-clickoutside 使用鼠标事件判断的,所以 click 的 阻止冒泡不会让 clickoutside 无效。

总结

解决 bug 的过程中需要做到不产生额外的 bug,并且深入分析问题的原因有助于能力的提高。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
原生js实现给指定元素的后面追加内容
Apr 10 Javascript
SOSO地图JS画出标注和中心点以html形式运行
Aug 09 Javascript
ANGULARJS中用NG-BIND指令实现单向绑定的例子
Dec 08 Javascript
node.js中的fs.futimes方法使用说明
Dec 17 Javascript
JQuery自适应窗口大小导航菜单附源码下载
Sep 01 Javascript
jquery.cookie.js用法实例详解
Dec 25 Javascript
JavaScript驾驭网页-CSS与DOM
Mar 24 Javascript
再谈Javascript中的基本类型和引用类型(推荐)
Jul 01 Javascript
vue动态生成dom并且自动绑定事件
Apr 19 Javascript
jquery图片放大镜效果
Jun 23 jQuery
解决vue+router路由跳转不起作用的一项原因
Jul 19 Javascript
Node.JS如何实现JWT原理
Sep 18 Javascript
JavaScript学习笔记之图片库案例分析
Jan 08 #Javascript
JavaScript学习笔记之DOM操作实例分析
Jan 08 #Javascript
vue单文件组件lint error自动fix与styleLint报错自动fix详解
Jan 08 #Javascript
说说如何在Vue.js中实现数字输入组件的方法
Jan 08 #Javascript
小试SVG之新手小白入门教程
Jan 08 #Javascript
vue组件通信传值操作示例
Jan 08 #Javascript
利用d3.js力导布局绘制资源拓扑图实例教程
Jan 08 #Javascript
You might like
Session 失效的原因汇总及解决丢失办法
2015/09/30 PHP
php强制下载文件函数
2016/08/24 PHP
php中文乱码问题的终极解决方案汇总
2017/08/01 PHP
PHP实现的自定义图像居中裁剪函数示例【测试可用】
2017/08/11 PHP
PHP实现的各类hash算法长度及性能测试实例
2017/08/27 PHP
js 处理数组重复元素示例代码
2013/12/27 Javascript
Javascript中this的用法详解
2014/09/22 Javascript
jQuery中first()方法用法实例
2015/01/06 Javascript
jQuery animate和CSS3相结合实现缓动追逐效果附源码下载
2016/04/18 Javascript
js验证真实姓名与身份证号,手机号的简单实例
2016/07/18 Javascript
JavaScript DOM 对象深入了解
2016/07/20 Javascript
JavaScript实现清空(重置)文件类型INPUT元素值的方法
2016/11/17 Javascript
详解nodejs微信jssdk后端接口
2017/05/25 NodeJs
vue单页应用中如何使用jquery的方法示例
2017/07/27 jQuery
React中jquery引用的实现方法
2017/09/12 jQuery
详解如何在vscode里面调试js和node.js的方法步骤
2018/12/24 Javascript
javascript中call()、apply()的区别
2019/03/21 Javascript
Vue 动态添加路由及生成菜单的方法示例
2019/06/20 Javascript
解决vue scoped scss 无效的问题
2020/09/04 Javascript
Python 异常处理实例详解
2014/03/12 Python
python通过smpt发送邮件的方法
2015/04/30 Python
python通过socket查询whois的方法
2015/07/18 Python
python实现简单购物商城
2016/05/21 Python
python thrift搭建服务端和客户端测试程序
2018/01/17 Python
python 移除字符串尾部的数字方法
2018/07/17 Python
python运用pygame库实现双人弹球小游戏
2019/11/25 Python
python GUI库图形界面开发之PyQt5布局控件QGridLayout详细使用方法与实例
2020/03/06 Python
Debenhams百货英国官方网站:Debenhams UK
2016/07/12 全球购物
Timberland德国官网:靴子、鞋子、衣服、夹克及配件
2019/12/10 全球购物
IBatis持久层技术
2016/07/18 面试题
网站设计师的岗位职责
2013/11/21 职场文书
初三新学期计划书
2014/05/03 职场文书
教师廉洁自律个人总结
2015/02/10 职场文书
2015少先队大队辅导员工作总结
2015/07/24 职场文书
《分数乘法》教学反思
2016/02/24 职场文书
一文读懂go中semaphore(信号量)源码
2021/04/03 Golang