Element分析(组件篇)——TableHeader

说明

table-header是表头组件,较为复杂,直接看源码解读。

源码解读

import ElCheckbox from 'element-ui/packages/checkbox';
import ElTag from 'element-ui/packages/tag';
import Vue from 'vue';
import FilterPanel from './filter-panel.vue';

/**
 * 获取所有的列,因为会有嵌套所以使用了递归
 * @param columns 原始的所有列
 * @return 所有列
 */
const getAllColumns = (columns) => {
  const result = [];
  columns.forEach((column) => {
    if (column.children) {
      result.push(column);
      result.push.apply(result, getAllColumns(column.children));
    } else {
      result.push(column);
    }
  });
  return result;
};

/**
 * 将所有的列信息转换成表头的行信息
 * @param originColumns 列信息
 * @return 表头信息
 */
const convertToRows = (originColumns) => {
  // 最大层级
  let maxLevel = 1;
  // 遍历列来判断表头每个单元格需要占多少格
  const traverse = (column, parent) => {
    if (parent) {
      column.level = parent.level + 1;
      if (maxLevel < column.level) {
        maxLevel = column.level;
      }
    }
    if (column.children) {
      let colSpan = 0;
      column.children.forEach((subColumn) => {
        traverse(subColumn, column);
        colSpan += subColumn.colSpan;
      });
      column.colSpan = colSpan;
    } else {
      column.colSpan = 1;
    }
  };

  // 获取每一列的层级
  originColumns.forEach((column) => {
    column.level = 1;
    traverse(column);
  });

  const rows = [];
  for (let i = 0; i < maxLevel; i++) {
    rows.push([]);
  }

  const allColumns = getAllColumns(originColumns);

  // 相同的层级作为同一行
  allColumns.forEach((column) => {
    if (!column.children) {
      column.rowSpan = maxLevel - column.level + 1;
    } else {
      column.rowSpan = 1;
    }
    rows[column.level - 1].push(column);
  });

  return rows;
};

export default {
  name: 'ElTableHeader',

  // 渲染函数
  render(h) {
    // 原始列信息
    const originColumns = this.store.states.originColumns;
    // 表头信息
    const columnRows = convertToRows(originColumns, this.columns);

    return (
      // table 上的 style 是为了清除默认的间距
      <table
        class="el-table__header"
        cellspacing="0"
        cellpadding="0"
        border="0">
        {/* colgroup 是用来存储列信息的 */}
        <colgroup>
          {/* 列信息 */}
          {
            this._l(this.columns, column =>
              <col
                name={ column.id }
                width={ column.realWidth || column.width }
              />)
          }
          {/* 如果左侧有固定还需要记录滚动条的宽度 */}
          {
            !this.fixed && this.layout.gutterWidth
              ? <col name="gutter" width={ this.layout.scrollY ? this.layout.gutterWidth : '' }></col>
              : ''
          }
        </colgroup>
        <thead>
          {
            // 按行渲染
            this._l(columnRows, (columns, rowIndex) =>
              <tr>
              {
                // 渲染每个单元格
                this._l(columns, (column, cellIndex) =>
                  <th
                    // 列高
                    colspan={ column.colSpan }
                    // 行宽
                    rowspan={ column.rowSpan }
                    // 鼠标移动事件
                    on-mousemove={ ($event) => this.handleMouseMove($event, column) }
                    // 鼠标移出事件
                    on-mouseout={ this.handleMouseOut }
                    // 鼠标按下事件
                    on-mousedown={ ($event) => this.handleMouseDown($event, column) }
                    // 鼠标单击事件
                    on-click={ ($event) => this.handleHeaderClick($event, column) }
                    class={
                      [
                        column.id,
                        column.order,
                        column.headerAlign,
                        column.className || '',
                        // 判断是否隐藏,为了处理 fixed
                        rowIndex === 0 && this.isCellHidden(cellIndex, columns) ? 'is-hidden' : '',
                        // 判断是不是最后一级
                        !column.children ? 'is-leaf' : ''
                      ]
                    }>
                    <div
                      class={
                        [
                          'cell',
                          // 如果这列有选择筛选条件就高亮
                          column.filteredValue && column.filteredValue.length > 0 ? 'highlight' : ''
                        ]
                      }>
                    {
                      // 渲染单元格内部的内容
                      column.renderHeader
                        ? column.renderHeader.call(
                          this._renderProxy,
                          h,
                          {
                            column,
                            $index: cellIndex,
                            store: this.store,
                            _self: this.$parent.$vnode.context
                          })
                        : column.label
                    }
                    {
                      // 渲染排序的标志
                      column.sortable
                        ? <span class="caret-wrapper" on-click={ ($event) => this.handleSortClick($event, column) }>
                            <i class="sort-caret ascending"></i>
                            <i class="sort-caret descending"></i>
                          </span>
                        : ''
                    }
                    {
                      // 渲染筛选器的箭头
                      column.filterable
                         ? <span
                            class="el-table__column-filter-trigger"
                            on-click={ ($event) => this.handleFilterClick($event, column) }>
                            <i
                              class={
                                [
                                  'el-icon-arrow-down',
                                  column.filterOpened ? 'el-icon-arrow-up' : ''
                                ]
                              }>
                            </i>
                          </span>
                        : ''
                    }
                    </div>
                  </th>
                )
              }
              {
                // 弥补滚动条的宽度
                !this.fixed && this.layout.gutterWidth
                  ? <th class="gutter" style={{ width: this.layout.scrollY ? this.layout.gutterWidth + 'px' : '0' }}></th>
                  : ''
              }
              </tr>
            )
          }
        </thead>
      </table>
    );
  },

  props: {
    fixed: String,
    store: {
      required: true
    },
    layout: {
      required: true
    },
    border: Boolean,
    defaultSort: {
      type: Object,
      default() {
        return {
          prop: '',
          order: ''
        };
      }
    }
  },

  components: {
    ElCheckbox,
    ElTag
  },

  computed: {
    // 判断是不是全选了
    isAllSelected() {
      return this.store.states.isAllSelected;
    },

    // 判断总的列数
    columnsCount() {
      return this.store.states.columns.length;
    },

    // 左侧固定列数
    leftFixedCount() {
      return this.store.states.fixedColumns.length;
    },

    // 右侧固定列数
    rightFixedCount() {
      return this.store.states.rightFixedColumns.length;
    },

    // 所有的列
    columns() {
      return this.store.states.columns;
    }
  },

  created() {
    this.filterPanels = {};
  },

  mounted() {
    if (this.defaultSort.prop) {
      const states = this.store.states;
      // 排序的属性
      states.sortProp = this.defaultSort.prop;
      // 升序或降序
      states.sortOrder = this.defaultSort.order || 'ascending';
      this.$nextTick(_ => {
        for (let i = 0, length = this.columns.length; i < length; i++) {
          let column = this.columns[i];
          // 如果是要排序的属性
          if (column.property === states.sortProp) {
            column.order = states.sortOrder;
            states.sortingColumn = column;
            break;
          }
        }

        if (states.sortingColumn) {
          this.store.commit('changeSortCondition');
        }
      });
    }
  },

  beforeDestroy() {
    const panels = this.filterPanels;
    for (let prop in panels) {
      // 销毁全部的筛选面板
      if (panels.hasOwnProperty(prop) && panels[prop]) {
        panels[prop].$destroy(true);
      }
    }
  },

  methods: {
    // 判断单元格是否应当隐藏
    isCellHidden(index, columns) {
      // 左侧固定的 wrapper 中,那么除了固定的这些列都应该隐藏掉
      if (this.fixed === true || this.fixed === 'left') {
        return index >= this.leftFixedCount;
      } else if (this.fixed === 'right') {  // 右侧固定的 wrapper 中,规定列之前的也都应当隐藏
        let before = 0;
        for (let i = 0; i < index; i++) {
          before += columns[i].colSpan;
        }
        return before < this.columnsCount - this.rightFixedCount;
      } else {  // 剩下的就是隐藏固定的列
        return (index < this.leftFixedCount) || (index >= this.columnsCount - this.rightFixedCount);
      }
    },

    // 切换全选
    toggleAllSelection() {
      this.store.commit('toggleAllSelection');
    },

    // 处理点击筛选器,应当显示对应的筛选面板
    handleFilterClick(event, column) {
      event.stopPropagation();
      const target = event.target;
      const cell = target.parentNode;
      const table = this.$parent;

      // 查找对应的筛选面板
      let filterPanel = this.filterPanels[column.id];

      // 如果存在,并且打开了,就关闭它
      if (filterPanel && column.filterOpened) {
        filterPanel.showPopper = false;
        return;
      }

      // 如果不存在,就创建它
      if (!filterPanel) {
        filterPanel = new Vue(FilterPanel);
        this.filterPanels[column.id] = filterPanel;

        filterPanel.table = table;
        filterPanel.cell = cell;
        filterPanel.column = column;
        !this.$isServer && filterPanel.$mount(document.createElement('div'));
      }

      // 创建后打开
      setTimeout(() => {
        filterPanel.showPopper = true;
      }, 16);
    },

    // 处理表头点击事件
    handleHeaderClick(event, column) {
      if (!column.filters && column.sortable) {  // 排序
        this.handleSortClick(event, column);
      } else if (column.filters && !column.sortable) {  // 筛选
        this.handleFilterClick(event, column);
      }

      this.$parent.$emit('header-click', column, event);
    },

    // 鼠标按下
    handleMouseDown(event, column) {
      if (this.$isServer) return;
      // 如果这一列还有孩子直接返回,应该操作孩子的宽度
      if (column.children && column.children.length > 0) return;
      /* istanbul ignore if */
      // 如果有拖拽的列,并且有边框
      if (this.draggingColumn && this.border) {
        // 表示正在拖动
        this.dragging = true;

        // 显示 resize
        this.$parent.resizeProxyVisible = true;

        const tableEl = this.$parent.$el;
        const tableLeft = tableEl.getBoundingClientRect().left;
        // 拖动列
        const columnEl = this.$el.querySelector(`th.${column.id}`);
        const columnRect = columnEl.getBoundingClientRect();
        const minLeft = columnRect.left - tableLeft + 30;

        columnEl.classList.add('noclick');

        this.dragState = {
          startMouseLeft: event.clientX,  // 鼠标开始位置
          startLeft: columnRect.right - tableLeft,  // 开始位置距离表格最左边的距离
          startColumnLeft: columnRect.left - tableLeft,  // 开始时列左边离表格最左边的距离
          tableLeft
        };

        // 显示拖拽位置的线
        const resizeProxy = this.$parent.$refs.resizeProxy;
        resizeProxy.style.left = this.dragState.startLeft + 'px';

        document.onselectstart = function() { return false; };
        document.ondragstart = function() { return false; };

        // 鼠标移动的时候,计算移动距离并且移动辅助线
        const handleMouseMove = (event) => {
          const deltaLeft = event.clientX - this.dragState.startMouseLeft;
          const proxyLeft = this.dragState.startLeft + deltaLeft;

          resizeProxy.style.left = Math.max(minLeft, proxyLeft) + 'px';
        };

        // 鼠标抬起
        const handleMouseUp = () => {
          if (this.dragging) {
            const finalLeft = parseInt(resizeProxy.style.left, 10);  // 最终停止的位置
            const columnWidth = finalLeft - this.dragState.startColumnLeft;  // 应该变成的列宽
            column.width = column.realWidth = columnWidth;  // 应用列宽改变

            this.store.scheduleLayout();  // 重新更新布局

            document.body.style.cursor = '';
            this.dragging = false;
            this.draggingColumn = null;
            this.dragState = {};

            this.$parent.resizeProxyVisible = false;
          }

          // 移除相应的监听器
          document.removeEventListener('mousemove', handleMouseMove);
          document.removeEventListener('mouseup', handleMouseUp);
          document.onselectstart = null;
          document.ondragstart = null;

          setTimeout(function() {
            columnEl.classList.remove('noclick');
          }, 0);
        };

        document.addEventListener('mousemove', handleMouseMove);
        document.addEventListener('mouseup', handleMouseUp);
      }
    },

    // 鼠标移动事件
    handleMouseMove(event, column) {
      if (column.children && column.children.length > 0) return;
      let target = event.target;
      while (target && target.tagName !== 'TH') {  // 寻找 th 标签
        target = target.parentNode;
      }

      // 如果没有列,或者不能改变大小
      if (!column || !column.resizable) return;

      // 如果正在拖动并且有边框
      if (!this.dragging && this.border) {
        let rect = target.getBoundingClientRect();

        const bodyStyle = document.body.style;
        if (rect.width > 12 && rect.right - event.pageX < 8) {
          bodyStyle.cursor = 'col-resize';
          this.draggingColumn = column;
        } else if (!this.dragging) {
          bodyStyle.cursor = '';
          this.draggingColumn = null;
        }
      }
    },

    // 鼠标移除后
    handleMouseOut() {
      if (this.$isServer) return;
      document.body.style.cursor = '';
    },

    // 切换排序顺序
    toggleOrder(order) {
      return !order ? 'ascending' : order === 'ascending' ? 'descending' : null;
    },

    // 点击排序
    handleSortClick(event, column) {
      event.stopPropagation();
      // 切换排序顺序
      let order = this.toggleOrder(column.order);

      // 寻找 TH
      let target = event.target;
      while (target && target.tagName !== 'TH') {
        target = target.parentNode;
      }

      // 如果这时候有 `noclick` 类,就移除
      if (target && target.tagName === 'TH') {
        if (target.classList.contains('noclick')) {
          target.classList.remove('noclick');
          return;
        }
      }

      // 如果不能排序就直接返回
      if (!column.sortable) return;

      const states = this.store.states;
      let sortProp = states.sortProp;
      let sortOrder;
      const sortingColumn = states.sortingColumn;

      // 如果排序列不是当前列,就切换成当前列
      if (sortingColumn !== column) {
        if (sortingColumn) {
          sortingColumn.order = null;
        }
        states.sortingColumn = column;
        sortProp = column.property;
      }

      // 如果没有顺序
      if (!order) {
        sortOrder = column.order = null;
        states.sortingColumn = null;
        sortProp = null;
      } else {
        sortOrder = column.order = order;
      }

      states.sortProp = sortProp;
      states.sortOrder = sortOrder;

      this.store.commit('changeSortCondition');
    }
  },

  data() {
    return {
      draggingColumn: null,
      dragging: false,
      dragState: {}
    };
  }
};

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,095评论 25 707
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 11,949评论 4 60
  • 前段时间组建了一个家长群,时常为大家做分享。可是后来我发现,除了我在群里分享,所有的人都很少说话。我说了很多次让大...
    艾秋阅读 153评论 0 1
  • 石匠雕石圆, 磨房驴转圈。 农耕大社会, 先祖不简单。
    草原骑手阅读 251评论 0 0
  • 我现在算是一个重度患者吧,也是这两天觉出自己已经是一个重度患者,会因此感到恐惧,但还是对自己那个渴望着什么的灵魂抱...
    小红心安东尼阅读 407评论 0 0