Element-ui中 元素滚动时el-option超出元素区域的问题

复现场景, 看图

分析缘由

为简单起见, 把选项区域描述为popperEljavascript

  • popperEl的z-index 比较大, 会覆盖在其余元素上面
  • popperEl默认是插入body元素的(能够将popper-append-to-body设为false后不插入到body)
  • popperEl是在mouseup事件里去作隐藏逻辑的, 而按下鼠标, 移动滚动条的时候, 并无触发mouseup事件.
  • popperEl并无监听滚动事件(无法监听, 也不必监听)

解决方案

方案一

我最初想到的解决方案是经过css解决,经过popper-class属性给Select下拉框添加类名,而后用css来作, 试了一下这个方案并不可行(只能在某些特定的场景下起做用),遂放弃,可能最优雅最高性能的方法就是用css来搞定, 有踩过这个坑的朋友请指点一下css

方案二

经过监听$root的scroll事件,利用事件冒泡,只须要在根元素上添加scroll事件的监听就能够了, 测试一番以后, 发现scroll事件根本不支持冒泡, event.bubbles为false)。html

方案三

经过查看element-ui 的select.vue, 发现控制popperEl显隐的是visible 和 emptyText这两个实例属性, 很明显, emptytext是不能动的, 只能在visible上动手脚了. 这里放一小段源码vue

<transition name="el-zoom-in-top" @before-enter="handleMenuEnter" @after-leave="doDestroy">
  <el-select-menu ref="popper" :append-to-body="popperAppendToBody" v-show="visible && emptyText !== false">
    <el-scrollbar tag="ul" wrap-class="el-select-dropdown__wrap" view-class="el-select-dropdown__list" ref="scrollbar" :class="{ 'is-empty': !allowCreate && query && filteredOptionsCount === 0 }" v-show="options.length > 0 && !loading">
      <el-option :value="query" created v-if="showNewOption">
      </el-option>
      <slot></slot>
    </el-scrollbar>
    <p class="el-select-dropdown__empty" v-if="emptyText && (!allowCreate || loading || (allowCreate && options.length === 0 ))">
      {{ emptyText }}
    </p>
  </el-select-menu>
</transition>
复制代码

全局搜索this.visible, 发现了这个方法java

handleClose() {
    this.visible = false;
},
复制代码

这下好办了, 按图索骥, 顺藤摸瓜, 找到这个element-ui

<template>
  <div class="el-select" :class="[selectSize ? 'el-select--' + selectSize : '']" @click.stop="toggleMenu" v-clickoutside="handleClose">
    后面的省略...
复制代码

找到v-clickoutside指令以后, 豁然开朗 原来点击其余区域的时候, popperEl会自动关闭的奥秘在这里, 结合方案二的灵感, 现给出以下代码.app

// src/mixins/fackClickOutSide.js
let lock = true;
let el = null;
const MousedownEvent = new Event('mousedown', {bubbles:true});
const MouseupEvent = new Event('mouseup', {bubbles:true});
const fakeClickOutSide = () => {
  document.dispatchEvent(MousedownEvent);
  document.dispatchEvent(MouseupEvent);
  lock = true; // console.log('dispatchEvent');
};
const mousedownHandle = e => {
  let classList = e.target.classList;
  if(classList.contains('el-select__caret') || classList.contains('el-input__inner')) {
    lock = false;
    return;
  }
  if(lock) return;
  fakeClickOutSide();
};
const mousewheelHandle = e => {
  if(lock || e.target.classList.contains('el-select-dropdown__item') || e.target.parentNode.classList.contains('el-select-dropdown__item')) return;
  fakeClickOutSide();
};
const eventListener = (type) => {
  el[type + 'EventListener']('mousedown', mousedownHandle);
  window[type + 'EventListener']('mousewheel', mousewheelHandle);
  window[type + 'EventListener']('DOMMouseScroll', mousewheelHandle); // fireFox 3.5+ 
}
export default {
  mounted() {
    el = this.$root.$el;
    el.addFakeClickOutSideEventCount = el.addFakeClickOutSideEventCount || 0;
    (! el.addFakeClickOutSideEventCount) && this.$nextTick(() => {
      eventListener('add');
    });
    el.addFakeClickOutSideEventCount += 1;
  },
  destroyed() {
    eventListener('remove');
    el.addFakeClickOutSideEventCount -= 1;
  },
}
复制代码

使用姿式

建议在根组件上混合进去, 固然,你也能够在须要的组件上去混合(不太建议, 这点代码性能损耗应该不大吧, 哈哈哈)ide

// src/App.vue
import fakeClickOutSide from '@/mixins/fakeClickOutSide.js'
export default {
    name: 'App',
    mixins: [fakeClickOutSide],
}
复制代码

测试

常规基础用法 和 自定义模板用法(模板内没有嵌套的标签) 均完美经过.性能

自定义模板内若是嵌套多级标签, 须要在标签上添加标记,而后在mousewheel事件回调里判断是否有这个标记. 测试

总结

依然存在的问题(隐患):

  • 在mousewheel事件回调没有作节流, 考虑到有锁, 且滚轮事件触发的频率也不是很高(相对于mousemove事件来说), 性能消耗并不大, 遂不作节流(主要是懒).
  • 在mousewheel事件回调里,判断event.target 是不是在popperEl元素内部的方法感受不是很靠谱, 且效率不高, 在mousedown 事件里判断是否是el-select元素的方法也存在一样的隐患, 后期再想办法修改(修改是不可能修改的, 又不是不能用).
  • 在自定义模板用法里, 若是有嵌套的标签, 那么在mousewheel事件回调里判断event.target 是否是在popperEl元素内部的方法就崩溃了(这是个雷), 目前的解决办法是手动在嵌套的标签上都加上一个标记, 在事件里,添加这个标记的判断, 可是这种作法对于已经编写完成的模板无效, 只能再次修改, 考虑过使用递归向上查找, 可是效率不高, 性能消耗太大, 且自定义el-option模板这种状况在咱们现阶段的业务中几乎不存在, 因此就没有考虑这个bug.

感谢一位大佬长期以来给予的帮助.

有踩过这个坑的朋友们, 说说大家的解决方案, 或者有优化的方法, 请不吝赐教,谢谢.

相关文章
相关标签/搜索