element UI scrollbar 源码学习和分析

scrollbar 源码解析的时候,我们不是为了弄懂它的每一行代码,但是我们必须学会它这种写scrollbar的思路,以及写法中充分利用vue的语法结构的学习。只有这样我们才能真正知道,写组件,尤其是写可以被广泛使用的公用组件的套路。ok,废话少说,开始进入枯燥的源码学习。请大家怀着一颗仁慈和包容的心态继续往下看,Come on,干起来!!!!

可以页面想象成一个长长的盒子,我们看到的是个视口。视口是不动的,盒子是移动的,当盒子向下滚动时,视口中所看到的盒子正在向下滚动。同样的,滚动块里面的滑块也往下移动同样比例的距离。

image
image

文件夹结构

scrollbar组件在 package/scrollbar/index.js中注册,其中package/scrollbar/src 是代码的核心部分,入口文件是 main.js。

使用如下

image
image

大家详细看一下注释,wrap是视口,view即内容可以滚动,horizontal 和 vertical是自定义滚动条。 下面开始撸代码,大家打起精神。走起!!!!!

main.js

main.js默认导出一个对象,接收一系列配置。


name: 'ElScrollbar',

components: { 
  //  滚动条组件,拥有水平与垂直两种形态
  Bar 
},

props: {
  native: Boolean,    //  是否使用原生滚动条,即不附加自定义滚动条
  wrapStyle: {},      //  wrap的内联样式
  wrapClass: {},      //  wrap的样式名
  viewClass: {},      //  view的样式名
  viewStyle: {},      //  view的内联样式
  noresize: Boolean,  //  当container尺寸发生变化时,自动更新滚动条组件的状态
  tag: {              //  组件最外层的标签属性,默认为 div
    type: String,
    default: 'div'
  }
},

data() {
  return {
    sizeWidth: '0',   //  水平滚动条的宽度
    sizeHeight: '0',  //  垂直滚动条的高度
    moveX: 0,         //  垂直滚动条的移动比例
    moveY: 0          //  水平滚动条的移动比例
  };
},

组件在render函数中生成结构。

tips: 如果在.vue文件中同时存在templete和 render 函数,组件实例会先取 template模板来渲染组件模板,而不是采用render函数

render函数一开始会通过scrollbarWidth 方法来计算当前浏览器的滚动条宽度

render(h) {
    //  获取浏览器的滚动条宽度
    let gutter = scrollbarWidth();
    //  wrap内联样式
    let style = this.wrapStyle;
    
    ...

scrollbarWidth 方法在scrollbar-width.js 中被默认导出

import Vue from 'vue';

//  闭包变量,用于记录滚动条宽度
let scrollBarWidth;

export default function() {
  //  如果在服务端运行,返回 0
  if (Vue.prototype.$isServer) return 0;
  //  如存在滚动条宽度,直接返回
  if (scrollBarWidth !== undefined) return scrollBarWidth;

  //  创建outer标签并隐藏
  const outer = document.createElement('div');
  outer.className = 'el-scrollbar__wrap';
  outer.style.visibility = 'hidden';
  outer.style.width = '100px';
  outer.style.position = 'absolute';
  outer.style.top = '-9999px';
  document.body.appendChild(outer);

  //  记录没有滚动内容的宽度
  const widthNoScroll = outer.offsetWidth;
  //  设置外层div滚动属性
  outer.style.overflow = 'scroll';
  //  创建inner标签,并追加到outer标签中
  const inner = document.createElement('div');
  inner.style.width = '100%';
  outer.appendChild(inner);
  //  此时outer已经可以滚动,记录下inner元素的宽度
  const widthWithScroll = inner.offsetWidth;
  //  销毁outer元素
  outer.parentNode.removeChild(outer);
  //  滚动条宽度 = 没有滚动条时的outer宽度 减去 有滚动条的outer中的inner宽度
  scrollBarWidth = widthNoScroll - widthWithScroll;
  //  返回滚动条宽度
  return scrollBarWidth;
};

得到滚动条方法会进行以下步骤
1、创建outer 容器,并记录outer容器offsetwidth
2、设置outer容器overflow:scroll,并新建inner容器,追加到outer容器下
3、此时outer容器会带着滚动条,记录inner容器的offsetwidth 宽度

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 196,302评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,563评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 143,433评论 0 325
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,628评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,467评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,354评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,777评论 3 387
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,419评论 0 255
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,725评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,768评论 2 314
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,543评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,387评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,794评论 3 300
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,032评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,305评论 1 252
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,741评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,946评论 2 336

推荐阅读更多精彩内容

  • element-ui ScrollBar组件源码深入分析 scrollbar组件根目录下包括index.js文件和...
    luichooy阅读 3,146评论 3 5
  • 前几天美化博客时发现滚动条在window下实在太难看,所以在基于vue的技术上寻找美化滚动条的方法。记得Eleme...
    onaug6th阅读 2,730评论 0 7
  • 第一部分 HTML&CSS整理答案 1. 什么是HTML5? 答:HTML5是最新的HTML标准。 注意:讲述HT...
    kismetajun阅读 27,347评论 1 45
  • ¥开启¥ 【iAPP实现进入界面执行逐一显】 〖2017-08-25 15:22:14〗 《//首先开一个线程,因...
    小菜c阅读 6,331评论 0 17
  • 问答题47 /72 常见浏览器兼容性问题与解决方案? 参考答案 (1)浏览器兼容问题一:不同浏览器的标签默认的外补...
    _Yfling阅读 13,709评论 1 92