最新文章专题视频专题问答1问答10问答100问答1000问答2000关键字专题1关键字专题50关键字专题500关键字专题1500TAG最新视频文章推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37视频文章20视频文章30视频文章40视频文章50视频文章60 视频文章70视频文章80视频文章90视频文章100视频文章120视频文章140 视频2关键字专题关键字专题tag2tag3文章专题文章专题2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章专题3
问答文章1 问答文章501 问答文章1001 问答文章1501 问答文章2001 问答文章2501 问答文章3001 问答文章3501 问答文章4001 问答文章4501 问答文章5001 问答文章5501 问答文章6001 问答文章6501 问答文章7001 问答文章7501 问答文章8001 问答文章8501 问答文章9001 问答文章9501
当前位置: 首页 - 科技 - 知识百科 - 正文

对ElementUItable组件的源码的详细分析

来源:懂视网 责编:小采 时间:2020-11-27 19:33:13
文档

对ElementUItable组件的源码的详细分析

对ElementUItable组件的源码的详细分析:本文章从如下图所示的最基本的table入手,分析table组件源代码。本人已经对table组件原来的源码进行削减。本文只对重要的代码片段进行讲解,推荐下载代码把项目运行起来,跟着文章的思路阅读。思路<template> <p class="el-t
推荐度:
导读对ElementUItable组件的源码的详细分析:本文章从如下图所示的最基本的table入手,分析table组件源代码。本人已经对table组件原来的源码进行削减。本文只对重要的代码片段进行讲解,推荐下载代码把项目运行起来,跟着文章的思路阅读。思路<template> <p class="el-t
本文章从如下图所示的最基本的table入手,分析table组件源代码。本人已经对table组件原来的源码进行削减。本文只对重要的代码片段进行讲解,推荐下载代码把项目运行起来,跟着文章的思路阅读。

思路

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

 <p class="el-table__header-wrapper"
 ref="headerWrapper">
 <table-header ref="tableHeader"
 :store="store">
 </table-header>
 </p>

 <p class="el-table__body-wrapper"
 ref="bodyWrapper">
 <table-body :context="context"
 :store="store"> 
 </table-body>
 </p>
 </p>
</template>

table、table-header、table-body、table-column之间通过table-store进行状态管理。table-header、table-body对table-store数据进行监听,每当table改变table-store数据时触发table-header、table-body重新渲染。

table-column为列数据column绑定相应的renderCell函数,供table-body渲染时使用。table-column这个组件自身不做任何渲染。所以会看到模板将其隐藏。还有就是table-header、table-body通过render函数进行渲染。

初始化顺序

3596660871-5b57ca0dcc1eb_articlex.jpeg

table

  1. 初始化store

    data() {
     const store = new TableStore(this);
     return {
     store,
     };
    }
  2. 将store共享给table-header、table-body

     <p class="el-table__header-wrapper"
     ref="headerWrapper">
     <table-header :store="store"></table-header>
     </p>
    
     <p class="el-table__body-wrapper"
     ref="bodyWrapper">
     <table-body :store="store"></table-body>
     </p>
  3. 将数据存储到store,供table-body获取data将其渲染

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

    created() {
     //.....
     this.tableId = `el-table_${tableIdSeed}`;
     //.....
     }
  5. 调用 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 = () => _self.$scopedSlots.default(data);
     //<template slot-scope="{row}">
     //<span>{{row.frequentlyUsed | formatBoolean}}</span>
     //</template>
     }
     
     if (!renderCell) {// table-header不渲染index列的走这里,
     /*<p className="cell">王小虎</p>*/
     renderCell = DEFAULT_RENDER_CELL;
     }
     
     // <ElTableColumn
     // type="index"
     // width="50"/>
     return <p className="cell">{renderCell(createElement, data)}</p>;
     };
     
    }
  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 => !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;
 },
 },

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

文档

对ElementUItable组件的源码的详细分析

对ElementUItable组件的源码的详细分析:本文章从如下图所示的最基本的table入手,分析table组件源代码。本人已经对table组件原来的源码进行削减。本文只对重要的代码片段进行讲解,推荐下载代码把项目运行起来,跟着文章的思路阅读。思路<template> <p class="el-t
推荐度:
标签: 代码 的代码 分析
  • 热门焦点

最新推荐

猜你喜欢

热门推荐

专题
Top