Vue 应该说是很火的一款前端库了,和 React 一样的高热度,今天就来用它写一个轻量的滚动条组件;

知识储备:要开发滚动条组件,需要知道知识点是如何计算滚动条的大小和位置,还有一个问题是如何监听容器大小的改变,然后更新滚动条的位置;

先把样式贴出来:

.disable-selection {
  -webkit-touch-callout: none;
  -webkit-user-select: none;
  -khtml-user-select: none;
  -moz-user-select: none;
  -ms-user-select: none;
  user-select: none;
}

.resize-trigger {
  position: absolute;
  display: block;
  top: 0;
  left: 0;
  height: 100%;
  width: 100%;
  overflow: hidden;
  pointer-events: none;
  z-index: -1;
  opacity: 0;
}

.scrollbar-container {
  position: relative;
  overflow-x: hidden!important;
  overflow-y: hidden!important;
  width: 100%;
  height: 100%;
}

.scrollbar-container--auto {
  overflow-x: visible!important;
  overflow-y: visible!important;
}

.scrollbar-container .scrollbar-view {
  width: 100%;
  height: 100%;
  -webkit-overflow-scrolling: touch;
}

.scrollbar-container .scrollbar-view-x {
  overflow-x: scroll!important;
}

.scrollbar-container .scrollbar-view-y {
  overflow-y: scroll!important;
}

.scrollbar-container .scrollbar-vertical,
.scrollbar-container .scrollbar-horizontal {
  position: absolute;
  opacity: 0;
  cursor: pointer;
  transition: opacity 0.25s linear;
  background: rgba(0, 0, 0, 0.2);
}

.scrollbar-container .scrollbar-vertical {
  top: 0;
  left: auto;
  right: 0;
  width: 12px;
}

.scrollbar-container .scrollbar-horizontal {
  top: auto;
  left: 0;
  bottom: 0;
  height: 12px;
}

.scrollbar-container:hover .scrollbar-vertical,
.scrollbar-container:hover .scrollbar-horizontal,
.scrollbar-container .scrollbar-vertical.scrollbar-show,
.scrollbar-container .scrollbar-horizontal.scrollbar-show {
  opacity: 1;
}

.scrollbar-container.cssui-scrollbar--s .scrollbar-vertical {
  width: 6px;
}

.scrollbar-container.cssui-scrollbar--s .scrollbar-horizontal {
  height: 6px;
}

 然后,把模板贴出来:

<template>
  <div
    :style="containerStyle"
    :class="containerClass"
    @mouseenter="quietUpdate"
    @mouseleave="quietOff"
  >
    <div
      ref="scroll"
      :style="scrollStyle"
      :class="scrollClass"
      @scroll.stop.prevent="realUpdate"
    >
      <div
        ref="content"
        v-resize="resizeHandle"
      >
        <slot />
      </div>
    </div>
    <div
      v-if="yBarShow"
      :style="yBarStyle"
      :class="yBarClass"
      @mousedown="downVertical"
    />
    <div
      v-if="xBarShow"
      :style="xBarStyle"
      :class="xBarClass"
      @mousedown="downHorizontal"
    />
  </div>
</template>

上面的代码中,我用到了 v-resize 这个指令,这个指令就是封装容器大小改变时,向外触发事件的,看到网上有通过 MutationObserver 来监听的,这个问题是监听所有的属性变化,好像还有兼容问题,还有一种方案是用 GitHub 的这个库:resize-observer-polyfill,上面的这些方法都可以,我也是尝试了一下,但我觉得始终是有点小题大做了,不如下面这个方法好,就是创建一个看不见的 object 对象,然后使它的绝对定位,相对于滚动父容器,和滚动条容器的大小保持一致,监听 object 里面 window 对象的 resize 事件,这样就可以做到实时响应高度变化了,贴上代码:

import Vue from 'vue';
import { throttle, isFunction } from 'lodash';

Vue.directive('resize', {
  inserted(el, { value: handle }) {
    if (!isFunction(handle)) { return; }

    const aimEl = el;
    const resizer = document.createElement('object');

    resizer.type = 'text/html';
    resizer.data = 'about:blank';
    resizer.setAttribute('tabindex', '-1');
    resizer.setAttribute('class', 'resize-trigger');
    resizer.onload = () => {
      const win = resizer.contentDocument.defaultView;
      win.addEventListener('resize', throttle(() => {
         const rect = el.getBoundingClientRect();
          handle(rect);
      }, 500));
    };

    aimEl.style.position = 'relative';
    aimEl.appendChild(resizer);
    aimEl.resizer = resizer;
  },

  unbind(el) {
    const aimEl = el;

    if (aimEl.resizer) {
      aimEl.style.position = '';
      aimEl.removeChild(aimEl.resizer);
      delete aimEl.resizer;
    }
  },
});

还有用到 tools js中的工具方法:

if (!Date.now) { Date.now = function () { return new Date().getTime(); }; }

const vendors = ['webkit', 'moz'];

if (!window.requestAnimationFrame) {
  for (let i = 0; i < vendors.length; ++i) {
    const vp = vendors[i];
    window.requestAnimationFrame = window[`${vp}RequestAnimationFrame`];
    window.cancelAnimationFrame = (window[`${vp}CancelAnimationFrame`] || window[`${vp}CancelRequestAnimationFrame`]);
  }
}

if (!window.requestAnimationFrame || !window.cancelAnimationFrame) {
  let lastTime = 0;

  window.requestAnimationFrame = callback => {
    const now = Date.now();
    const nextTime = Math.max(lastTime + 16, now);
    return setTimeout(() => { callback(lastTime = nextTime); }, nextTime - now);
  };

  window.cancelAnimationFrame = clearTimeout;
}

let scrollWidth = 0;

// requestAnimationFrame 封装
export const ref = (fn) => { window.requestAnimationFrame(fn); };

// 检测 class
export const hasClass = (el = null, cls = '') => {
  if (!el || !cls) { return false; }
  if (cls.indexOf(' ') !== -1) { throw new Error('className should not contain space.'); }
  if (el.classList) { return el.classList.contains(cls); }
  return ` ${el.className} `.indexOf(` ${cls} `) > -1;
};

// 添加 class
export const addClass = (element = null, cls = '') => {
  const el = element;
  if (!el) { return; }
  let curClass = el.className;
  const classes = cls.split(' ');

  for (let i = 0, j = classes.length; i < j; i += 1) {
    const clsName = classes[i];
    if (!clsName) { continue; }

    if (el.classList) {
      el.classList.add(clsName);
    } else if (!hasClass(el, clsName)) {
      curClass += ' ' + clsName;
    }
  }
  if (!el.classList) {
    el.className = curClass;
  }
};

// 获取滚动条宽度
export const getScrollWidth = () => {
  if (scrollWidth > 0) { return scrollWidth; }

  const block = docu.createElement('div');
  block.style.cssText = 'position:absolute;top:-1000px;width:100px;height:100px;overflow-y:scroll;';
  body.appendChild(block);
  const { clientWidth, offsetWidth } = block;
  body.removeChild(block);
  scrollWidth = offsetWidth - clientWidth;

  return scrollWidth;
};

下面是 js 功能的部分,代码还是不少,有一些方法做了节流处理,用了一些 lodash 的方法,主要还是上面提到的滚动条计算的原理,大小的计算,具体看  toUpdate  这个方法,位置的计算,主要是 horizontalHandler,verticalHandler,实际滚动距离的计算,看mouseMoveHandler 这个方法:

import { raf, addClass, removeClass, getScrollWidth } from 'src/tools';

const SCROLLBARSIZE = getScrollWidth();

/**
 * ----------------------------------------------------------------------------------
 * UiScrollBar Component
 * ----------------------------------------------------------------------------------
 *
 * @author zhangmao
 * @change 2019/4/15
 */
export default {
  name: 'UiScrollBar',

  props: {
    size: { type: String, default: 'normal' }, // small
    // 主要是为了解决在 dropdown 隐藏的情况下无法获取当前容器的真实 width height 的问题
    show: { type: Boolean, default: false },
    width: { type: Number, default: 0 },
    height: { type: Number, default: 0 },
    maxWidth: { type: Number, default: 0 },
    maxHeight: { type: Number, default: 0 },
  },

  data() {
    return {
      enter: false,
      yRatio: 0,
      xRatio: 0,
      lastPageY: 0,
      lastPageX: 0,
      realWidth: 0,
      realHeight: 0,
      yBarTop: 0,
      yBarHeight: 0,
      xBarLeft: 0,
      xBarWidth: 0,
      scrollWidth: 0,
      scrollHeight: 0,
      containerWidth: 0,
      containerHeight: 0,
      cursorDown: false,
    };
  },

  computed: {
    xLimit() { return this.width > 0 || this.maxWidth > 0; },
    yLimit() { return this.height > 0 || this.maxHeight > 0; },
    yBarShow() { return this.getYBarShow(); },
    xBarShow() { return this.getXBarShow(); },
    yBarStyle() { return { top: `${this.yBarTop}%`, height: `${this.yBarHeight}%` }; },
    yBarClass() { return ['scrollbar-vertical', { 'scrollbar-show': this.cursorDown }]; },
    xBarStyle() { return { left: `${this.xBarLeft}%`, width: `${this.xBarWidth}%` }; },
    xBarClass() { return ['scrollbar-horizontal', { 'scrollbar-show': this.cursorDown }]; },
    scrollClass() {
      return ['scrollbar-view', {
        'scrollbar-view-x': this.xBarShow,
        'scrollbar-view-y': this.yBarShow,
      }];
    },
    scrollStyle() {
      const hasWidth = this.yBarShow && this.scrollWidth > 0;
      const hasHeight = this.xBarShow && this.scrollHeight > 0;
      return {
        width: hasWidth ? `${this.scrollWidth}px` : '',
        height: hasHeight ? `${this.scrollHeight}px` : '',
      };
    },
    containerClass() {
      return ['scrollbar-container', {
        'cssui-scrollbar--s': this.size === 'small',
        'scrollbar-container--auto': !this.xBarShow && !this.yBarShow,
      }];
    },
    containerStyle() {
      const showSize = this.xBarShow || this.yBarShow;
      const styleObj = {};

      if (showSize) {
        if (this.containerWidth > 0) { styleObj.width = `${this.containerWidth}px`; }
        if (this.containerHeight > 0) { styleObj.height = `${this.containerHeight}px`; }
      }

      return styleObj;
    },
  },

  watch: {
    show: 'showChange',
    width: 'initail',
    height: 'initail',
    maxWidth: 'initail',
    maxHeight: 'initail',
  },

  created() {
    this.dftData();
    this.initEmiter();
  },

  mounted() { this.$nextTick(this.initail); },

  methods: {

    // ------------------------------------------------------------------------------

    // 外部调用方法
    refresh() { this.initail(); }, // 手动更新滚动条
    scrollX(x) { this.$refs.scroll.scrollLeft = x; },
    scrollY(y) { this.$refs.scroll.scrollTop = y; },
    scrollTop() { this.$refs.scroll.scrollTop = 0; },
    getScrollEl() { return this.$refs.scroll; },
    scrollBottom() { this.$refs.scroll.scrollTop = this.$refs.content.offsetHeight; },

    // --------------------------------------------------------------------------

    quietOff() { this.enter = false; },

    // ------------------------------------------------------------------------------

    quietUpdate() {
      this.enter = true;
      this.scrollUpdate();
    },

    // ------------------------------------------------------------------------------

    realUpdate() {
      this.quietOff();
      this.scrollUpdate();
    },

    // ------------------------------------------------------------------------------

    resizeHandle() { this.initail(); },

    // ------------------------------------------------------------------------------

    // 默认隐藏 异步展示的情况
    showChange(val) { if (val) { this.initail(); } },

    // ------------------------------------------------------------------------------

    // 组件渲染成功后的入口
    initail() {
      this.setContainerSize();
      this.setScrollSize();
      this.setContentSize();
      this.realUpdate();
    },

    // ------------------------------------------------------------------------------

    // 设置整个容器的大小
    setContainerSize() {
      this.setContainerXSize();
      this.setContainerYSize();
    },

    // ------------------------------------------------------------------------------

    // 设置滚动容器的大小
    setScrollSize() {
      this.scrollWidth = this.containerWidth + SCROLLBARSIZE;
      this.scrollHeight = this.containerHeight + SCROLLBARSIZE;
    },

    // ------------------------------------------------------------------------------

    // 设置内容区域的大小
    setContentSize() {
      const realElement = this.$refs.content.firstChild;

      if (realElement) {
        const { offsetWidth = 0, offsetHeight = 0 } = realElement;

        this.realWidth = this.lodash.round(offsetWidth);
        this.realHeight = this.lodash.round(offsetHeight);
      }
    },

    // ------------------------------------------------------------------------------

    setContainerXSize() {
      if (this.xLimit) {
        this.containerWidth = this.width || this.maxWidth;
        return;
      }
      if (this.yLimit) { this.containerWidth = this.lodash.round(this.$el.offsetWidth); }
    },

    // ------------------------------------------------------------------------------

    setContainerYSize() {
      if (this.yLimit) {
        this.containerHeight = this.height || this.maxHeight;
        return;
      }
      if (this.xLimit) { this.containerHeight = this.lodash.round(this.$el.offsetHeight); }
    },

    // ------------------------------------------------------------------------------

    downVertical(e) {
      this.lastPageY = e.pageY;
      this.cursorDown = true;
      addClass(document.body, 'disable-selection');
      document.addEventListener('mousemove', this.moveVertical, false);
      document.addEventListener('mouseup', this.upVertical, false);
      document.onselectstart = () => false;
      return false;
    },

    // ------------------------------------------------------------------------------

    downHorizontal(e) {
      this.lastPageX = e.pageX;
      this.cursorDown = true;
      addClass(document.body, 'disable-selection');
      document.addEventListener('mousemove', this.moveHorizontal, false);
      document.addEventListener('mouseup', this.upHorizontal, false);
      document.onselectstart = () => false;
      return false;
    },

    // ------------------------------------------------------------------------------

    moveVertical(e) {
      const delta = e.pageY - this.lastPageY;
      this.lastPageY = e.pageY;

      raf(() => { this.$refs.scroll.scrollTop += delta / this.yRatio; });
    },

    // ------------------------------------------------------------------------------

    moveHorizontal(e) {
      const delta = e.pageX - this.lastPageX;
      this.lastPageX = e.pageX;

      raf(() => { this.$refs.scroll.scrollLeft += delta / this.xRatio; });
    },

    // ------------------------------------------------------------------------------

    upVertical() {
      this.cursorDown = false;
      removeClass(document.body, 'disable-selection');
      document.removeEventListener('mousemove', this.moveVertical);
      document.removeEventListener('mouseup', this.upVertical);
      document.onselectstart = null;
    },

    // ------------------------------------------------------------------------------

    upHorizontal() {
      this.cursorDown = false;
      removeClass(document.body, 'disable-selection');
      document.removeEventListener('mousemove', this.moveHorizontal);
      document.removeEventListener('mouseup', this.upHorizontal);
      document.onselectstart = null;
    },

    // ------------------------------------------------------------------------------

    scrollUpdate() {
      const {
        clientWidth = 0,
        scrollWidth = 0,
        clientHeight = 0,
        scrollHeight = 0,
      } = this.$refs.scroll;

      this.yRatio = clientHeight / scrollHeight;
      this.xRatio = clientWidth / scrollWidth;

      raf(() => {
        if (this.yBarShow) {
          this.yBarHeight = Math.max(this.yRatio * 100, 1);
          this.yBarTop = this.lodash.round((this.$refs.scroll.scrollTop / scrollHeight) * 100, 2);

          // 只更新不触发事件
          if (this.enter) { return; }

          const top = this.$refs.scroll.scrollTop;
          const left = this.$refs.scroll.scrollLeft;
          const cHeight = this.$refs.scroll.clientHeight;
          const sHeight = this.$refs.scroll.scrollHeight;

          // trigger event
          this.debounceScroll({ top, left });
          if (top === 0) {
            this.debounceTop();
          } else if (top + cHeight === sHeight) {
            this.debounceBottom();
          }
        }

        if (this.xBarShow) {
          this.xBarWidth = Math.max(this.xRatio * 100, 1);
          this.xBarLeft = this.lodash.round((this.$refs.scroll.scrollLeft / scrollWidth) * 100, 2);

          // 只更新不触发事件
          if (this.enter) { return; }

          const top = this.$refs.scroll.scrollTop;
          const left = this.$refs.scroll.scrollLeft;
          const cWidth = this.$refs.scroll.clientWidth;
          const sWidth = this.$refs.scroll.scrollWidth;

          // trigger event
          this.debounceScroll({ top, left });
          if (left === 0) {
            this.debounceLeft();
          } else if (left + cWidth === sWidth) {
            this.debounceRight();
          }
        }
      });
    },

    // ------------------------------------------------------------------------------

    dftData() {
      this.debounceLeft = null;
      this.debounceRight = null;
      this.debounceTop = null;
      this.debounceBottom = null;
      this.debounceScroll = null;
    },

    // ------------------------------------------------------------------------------

    // 初始化触发事件
    initEmiter() {
      this.turnOn('winResize', this.initail);
      this.debounceTop = this.lodash.debounce(() => this.$emit('top'), 500);
      this.debounceLeft = this.lodash.debounce(() => this.$emit('left'), 500);
      this.debounceRight = this.lodash.debounce(() => this.$emit('right'), 500);
      this.debounceBottom = this.lodash.debounce(() => this.$emit('bottom'), 500);
      this.debounceScroll = this.lodash.debounce(obj => this.$emit('scroll', obj), 250);
    },

    // ------------------------------------------------------------------------------

    // 是否展示垂直的滚动条
    getYBarShow() {
      if (this.yLimit) {
        if (this.height > 0) { return this.realHeight > this.height; }
        if (this.maxHeight > 0) { return this.realHeight > this.maxHeight; }
        return this.realHeight > this.containerHeight;
      }
      return false;
    },

    // ------------------------------------------------------------------------------

    // 是否展示横向的滚动条
    getXBarShow() {
      if (this.xLimit) {
        if (this.width > 0) { return this.realWidth > this.width; }
        if (this.maxWidth > 0) { return this.realWidth > this.maxWidth; }
        return this.realWidth > this.containerWidth;
      }
      return false;
    },

    // ------------------------------------------------------------------------------

  },
};