问题
在项目页面中使用 element popover,设置trigger='click'时点击外部不会触发自动隐藏,但在 element 官网中是可以正常触发的(官方示例),项目中的菜单是自定义写的,所以怀疑是有黑魔法。
查找原因
- 将 popover 写在app.vue根组件内,发现可以正常触发自动隐藏。
- 在app.vue的 mounted 钩子中加入
window.addEventListener('click', () => console.log('window click==='))
,发现只有菜单栏外层能够触发。 - 检查菜单栏组件,发现代码中
<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,并且深入分析问题的原因有助于能力的提高。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
稳了!魔兽国服回归的3条重磅消息!官宣时间再确认!
昨天有一位朋友在大神群里分享,自己亚服账号被封号之后居然弹出了国服的封号信息对话框。
这里面让他访问的是一个国服的战网网址,com.cn和后面的zh都非常明白地表明这就是国服战网。
而他在复制这个网址并且进行登录之后,确实是网易的网址,也就是我们熟悉的停服之后国服发布的暴雪游戏产品运营到期开放退款的说明。这是一件比较奇怪的事情,因为以前都没有出现这样的情况,现在突然提示跳转到国服战网的网址,是不是说明了简体中文客户端已经开始进行更新了呢?
更新日志
- 小骆驼-《草原狼2(蓝光CD)》[原抓WAV+CUE]
- 群星《欢迎来到我身边 电影原声专辑》[320K/MP3][105.02MB]
- 群星《欢迎来到我身边 电影原声专辑》[FLAC/分轨][480.9MB]
- 雷婷《梦里蓝天HQⅡ》 2023头版限量编号低速原抓[WAV+CUE][463M]
- 群星《2024好听新歌42》AI调整音效【WAV分轨】
- 王思雨-《思念陪着鸿雁飞》WAV
- 王思雨《喜马拉雅HQ》头版限量编号[WAV+CUE]
- 李健《无时无刻》[WAV+CUE][590M]
- 陈奕迅《酝酿》[WAV分轨][502M]
- 卓依婷《化蝶》2CD[WAV+CUE][1.1G]
- 群星《吉他王(黑胶CD)》[WAV+CUE]
- 齐秦《穿乐(穿越)》[WAV+CUE]
- 发烧珍品《数位CD音响测试-动向效果(九)》【WAV+CUE】
- 邝美云《邝美云精装歌集》[DSF][1.6G]
- 吕方《爱一回伤一回》[WAV+CUE][454M]