Home > Web Front-end > JS Tutorial > Detailed analysis of the source code of the Element UI table component

Detailed analysis of the source code of the Element UI table component

不言
Release: 2018-07-25 10:23:39
Original
4003 people have browsed it

This article starts with the most basic table as shown in the figure below and analyzes the table component source code. I have cut down the original source code of the table component. This article only explains the important code snippets. It is recommended to download the code to run the project and read along with the article.

Ideas

<template>
  <p>
    <!-- 隐藏列: slot里容纳table-column -->
    </p>
<p>
      <slot></slot>
    </p>

    <p>
      <table-header>
      </table-header>
    </p>

    <p>
      <table-body>                  
      </table-body>
    </p>
  
</template>
Copy after login

State management between table, table-header, table-body and table-column is done through table-store. table-header and table-body monitor table-store data, and trigger table-header and table-body to re-render whenever the table changes table-store data.

table-column binds the corresponding renderCell function to the column data column for use when rendering the table-body. The table-column component itself does not do any rendering. So you'll see that the template hides it. There is also table-header and table-body that are rendered through the render function.

Initialization sequence

Detailed analysis of the source code of the Element UI table component

##table

  1. Initialize store

    data() {
      const store = new TableStore(this);
      return {
        store,
      };
    }
    Copy after login
  2. Share the store to table-header and table-body

        <p>
          <table-header></table-header>
        </p>
    
        <p>
          <table-body></table-body>
        </p>
    Copy after login
  3. Store data to the store for table-body gets data and renders it

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

    created() {
          //.....
          this.tableId = `el-table_${tableIdSeed}`;
          //.....
      }
    Copy after login
  5. Call updateColumns to trigger table-header and table-body secondary render updates , mark mounted completed

    mounted() {
        // .....
        this.store.updateColumns();
        // .....
        this.$ready = true;
    }
    Copy after login
##table-column

    Generate column and bind column
  1. renderCell Function

    For table-body use<div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:php;toolbar:false">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&gt;           //&lt;span&gt;{{row.frequentlyUsed | formatBoolean}}&lt;/span&gt;           //&lt;/template&gt;         }            if (!renderCell) {// table-header不渲染index列的走这里,           /*&lt;p&gt;王小虎&lt;/p&gt;*/           renderCell = DEFAULT_RENDER_CELL;         }            //  &lt;eltablecolumn&gt;&lt;/eltablecolumn&gt;         return &lt;p&gt;{renderCell(createElement, data)}&lt;/p&gt;;       };    }</pre><div class="contentsignin">Copy after login</div></div>

  2. Fill the store.state._columns array with data
  3. mounted() {
        // ...... 
        owner.store.commit('insertColumn', this.columnConfig, columnIndex, this.isSubColumn ? parent.columnConfig : null);
    }
    Copy after login

table-store

table-store has two very important attributes _columns and data. _columns saves the relevant information of the columns, and data saves the table data passed in by the developer. There are also two important functions insertColumn and updateColumns.

    insertColumn fills data for _columns
  1. 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);
        }
    
        // .....
      },
    }
    Copy after login

  2. updateColumns filters _columns to get columns
  3. 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);
      // ....
    }
    Copy after login

table-header, table-body

table-header, table-body all have the following attributes

props: {
    store: {
      required: true
    },
}

computed: {
    columns() {
      return this.store.states.columns;
    },
},

render(){
    // 渲染columns的数据
}
Copy after login

The working principle of these two components is to monitor column data Change to trigger render. In the mounted phase of the table component, updateColumns will be called to update the columns, thereby triggering table-header and table-body to be re-rendered.

In addition, table-body will also monitor data changes and trigger render. For example, when the component is loaded, a request is sent, data is assigned to the request response, and the table-body is re-rendered.

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

Related recommendations:

Analysis of the reasons for binding this in React components

Analysis of batch asynchronous updates and nextTick principles in Vue source code

The above is the detailed content of Detailed analysis of the source code of the Element UI table component. For more information, please follow other related articles on the PHP Chinese website!

Related labels:
source:php.cn
Statement of this Website
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
Popular Tutorials
More>
Latest Downloads
More>
Web Effects
Website Source Code
Website Materials
Front End Template