本文章从如下图所示的最基本的table入手,分析table组件源代码。本人已经对table组件原来的源码进行削减,源码点击这里下载。本文只对重要的代码片段进行讲解,推荐下载代码把项目运行起来,跟着文章的思路阅读。

思路

```<template>
<div class="el-table">
<!-- 隐藏列: slot里容纳table-column -->
<div class="hidden-columns" ref="hiddenColumns">
<slot></slot>
</div>

&lt;div class="el-table__header-wrapper"
ref="headerWrapper"&gt;
&lt;table-header ref="tableHeader"
:store="store"&gt;
&lt;/table-header&gt;
&lt;/div&gt; &lt;div class="el-table__body-wrapper"
ref="bodyWrapper"&gt;
&lt;table-body :context="context"
:store="store"&gt;
&lt;/table-body&gt;
&lt;/div&gt;

</div>

</template>


<p>table、table-header、table-body、table-column之间通过table-store进行状态管理。table-header、table-body对table-store数据进行监听,每当table改变table-store数据时触发table-header、table-body重新渲染。</p>
<p>table-column为列数据column绑定相应的renderCell函数,供table-body渲染时使用。table-column这个组件自身不做任何渲染。所以会看到模板将其隐藏。还有就是table-header、table-body通过render函数进行渲染。</p>
<h2>初始化顺序</h2> ![](https://img2018.cnblogs.com/blog/1557854/201812/1557854-20181220201602169-596863631.jpg) <h2>table</h2>
<ol>
<li>
<p>初始化store</p>

data() {

const store = new TableStore(this);

return {

store,

};

}


</li>
<li>
<p>将store共享给table-header、table-body</p>
``` &lt;div class="el-table__header-wrapper"
ref="headerWrapper"&gt;
&lt;table-header :store="store"&gt;&lt;/table-header&gt;
&lt;/div&gt; &lt;div class="el-table__body-wrapper"
ref="bodyWrapper"&gt;
&lt;table-body :store="store"&gt;&lt;/table-body&gt;
&lt;/div&gt;
  • 将数据存储到store,供table-body获取data将其渲染


    watch: {
    data: {
    immediate: true,
    handler(value) {
    // 供 table-body computed.data 使用
    this.store.commit('setData', value);
    // ......
    }
    },
    },
  • 设置tableId


    created() {
    //.....
    this.tableId = `el-table_${tableIdSeed}`;
    //.....
    }
  • 调用 updateColumns 触发 table-header、table-body 二次render更新,标记mounted完成


    mounted() {
    // .....
    this.store.updateColumns();
    // .....
    this.$ready = true;
    }
  • table-column

    1. 生成column,并为column绑定renderCell函数供table-body使用


      created(){
      // .........
      let column = getDefaultColumn(type, {
      id: this.columnId,
      columnKey: this.columnKey,
      label: this.label,
      property: this.prop || this.property,// 旧版element ui为property,现在的版本是prop
      type, // selection、index、expand
      renderCell: null,
      renderHeader: this.renderHeader, // 提供给table-column, table-column.js line 112
      width,
      formatter: this.formatter,
      context: this.context,
      index: this.index,
      });
      // ......... // 提table-body使用, table-body.js line 69
      column.renderCell = function (createElement, data) {
      if (_self.$scopedSlots.default) {
      renderCell = () =&gt; _self.$scopedSlots.default(data);
      //&lt;template slot-scope="{row}"&gt;
      //&lt;span&gt;{{row.frequentlyUsed | formatBoolean}}&lt;/span&gt;
      //&lt;/template&gt;
      } if (!renderCell) {// table-header不渲染index列的走这里,
      /*&lt;div className="cell"&gt;王小虎&lt;/div&gt;*/
      renderCell = DEFAULT_RENDER_CELL;
      } // &lt;ElTableColumn
      // type="index"
      // width="50"/&gt;
      return &lt;div className="cell"&gt;{renderCell(createElement, data)}&lt;/div&gt;;
      }; }
    2. 给store.state._columns数组填充数据


      mounted() {
      // ......
      owner.store.commit('insertColumn', this.columnConfig, columnIndex, this.isSubColumn ? parent.columnConfig : null);
      }

    table-store

    table-store有两个很重要的属性_columns、data,_columns保存列的相关信息,data则保存开发者传入的表格数据。还有两个重要的函数insertColumn与updateColumns。

    1. insertColumn为_columns填充数据


      TableStore.prototype.mutations = {
      insertColumn(states, column, index, parent) {
      let array = states._columns;
      // ...... if (typeof index !== 'undefined') {
      // 在index的位置插入column
      array.splice(index, 0, column);
      } else {
      array.push(column);
      } // .....
      },
      }
    2. updateColumns 对_columns进行过滤得到columns


      TableStore.prototype.updateColumns = function() {
      const states = this.states;
      const _columns = states._columns || []; const notFixedColumns = _columns.filter(column =&gt; !column.fixed);
      // .....
      const leafColumns = doFlattenColumns(notFixedColumns);
      // ..... states.columns = [].concat(leafColumns);
      // ....
      }

    table-header、table-body

    table-header、table-body都拥有以下属性


    props: {
    store: {
    required: true
    },
    } computed: {
    columns() {
    return this.store.states.columns;
    },
    }, render(){
    // 渲染columns的数据
    }

    这两个组件的工作原理是监听columns数据变化以触发render渲染。在table组件的mounted阶段会调用 updateColumns 更新 columns,从而触发 table-header、table-body 重新渲染。

    另外table-body还会监听data变化,触发render。例如当组件加载后发送请求,待请求响应赋值data,重新渲染table-body。


    computed: {
    data() {
    // table.vue watch.data 中 调用 setData 在store 中存储 data
    return this.store.states.data;
    },
    },

    参考

    Element UI table官方文档

    来源:https://segmentfault.com/a/1190000015756225

    Element UI table组件源码分析的更多相关文章

    1. 开源MyBatisGenerator组件源码分析

      开源MyBatisGenerator组件源码分析 看源码前,先了解Generator能做什么? MyBatisGenerator是用来生成mybatis的Mapper接口和xml文件的工具,提供多种启 ...

    2. Django-restframework 源码之认证组件源码分析

      Django-restframework 源码之认证组件源码分析 一 前言 之前在 Django-restframework 的流程分析博客中,把最重要的关于认证.权限和频率的方法找到了.该方法是 A ...

    3. element-ui 组件源码分析整理笔记目录

      element-ui button组件 radio组件源码分析整理笔记(一) element-ui switch组件源码分析整理笔记(二) element-ui inputNumber.Card .B ...

    4. ceph-csi组件源码分析(1)-组件介绍与部署yaml分析

      更多ceph-csi其他源码分析,请查看下面这篇博文:kubernetes ceph-csi分析目录导航 ceph-csi组件源码分析(1)-组件介绍与部署yaml分析 基于tag v3.0.0 ht ...

    5. element-ui button组件 radio组件源码分析整理笔记(一)

      Button组件 button.vue <template> <button class="el-button" @click="handleClick ...

    6. element-ui input组件源码分析整理笔记(六)

      input 输入框组件 源码: <template> <div :class="[ type === 'textarea' ? 'el-textarea' : 'el-in ...

    7. element-ui MessageBox组件源码分析整理笔记(十二)

      MessageBox组件源码,有添加部分注释 main.vue <template> <transition name="msgbox-fade"> < ...

    8. Django REST framework —— 权限组件源码分析

      在上一篇文章中我们已经分析了认证组件源码,我们再来看看权限组件的源码,权限组件相对容易,因为只需要返回True 和False即可 代码 class ShoppingCarView(ViewSetMix ...

    9. Django REST framework —— 认证组件源码分析

      我在前面的博客里已经讲过了,我们一般编写API的时候用的方式 class CoursesView(ViewSetMixin,APIView): pass 这种方式的有点是,灵活性比较大,可以根据自己的 ...

    随机推荐

    1. elasticsearch query 和 filter 的区别

      Query查询器 与 Filter 过滤器 尽管我们之前已经涉及了查询DSL,然而实际上存在两种DSL:查询DSL(query DSL)和过滤DSL(filter DSL).过滤器(filter)通常 ...

    2. property中ref、value、name的区别

      转载: 版权声明:本文为CSDN博主「qq_36098284」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明.原文链接:https://blog.csdn.net ...

    3. RN 环境搭建 运行demo App

      1.环境搭建 1.1 JDK 1.2Android JDK  需要安装c++环境(我选择安装android studio) 1.3Node npm config set registry https: ...

    4. Ionic cordova-plugin-splashscreen

      1.添加插件 cordova plugin add https://github.com/apache/cordova-plugin-splashscreen.git 2.设置启动画面 在根目录下面r ...

    5. 利用PHP获取访客IP、地区位置、浏览器及来源页面等信息

      这篇文章主要介绍了利用PHP获取访客IP.地区位置.浏览器及来源页面等信息的相关资料,文中给出了详细的示例代码供大家参考学习,对大家具有一定的参考借鉴价值,需要的朋友们下面来一起看看吧. 前言 本文中 ...

    6. mysql分区与分表的区别

      分区 分区就是把一个数据表的文件和索引分散存储在不同的物理文件中. mysql支持的分区类型包括Range.List.Hash.Key,其中Range比较常用: RANGE分区:基于属于一个给定连续区 ...

    7. jvm 分代回收算法通俗理解

      jvm区域总体分两类,heap区和非heap区.heap区又分:Eden Space(伊甸园).Survivor Space(幸存者区).Tenured Gen(老年代-养老区). 非heap区又分: ...

    8. 《DSP using MATLAB》Problem 7.34

      代码: %% ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ %% Output In ...

    9. Luogu P1462 通往奥格瑞玛的道路(最短路+二分)

      P1462 通往奥格瑞玛的道路 题面 题目背景 在艾泽拉斯大陆上有一位名叫歪嘴哦的神奇术士,他是部落的中坚力量 有一天他醒来后发现自己居然到了联盟的主城暴风城 在被众多联盟的士兵攻击后,他决定逃回自己 ...

    10. Leetcode455.Assign Cookies分发饼干

      假设你是一位很棒的家长,想要给你的孩子们一些小饼干.但是,每个孩子最多只能给一块饼干.对每个孩子 i ,都有一个胃口值 gi ,这是能让孩子们满足胃口的饼干的最小尺寸:并且每块饼干 j ,都有一个尺寸 ...