主頁 > 企業開發 > element el-table表格的vue組件二次封裝(附表格高度自適應)

element el-table表格的vue組件二次封裝(附表格高度自適應)

2021-01-19 08:40:10 企業開發

基于vue的el-table表格二次封裝組件方法

前言

在公司實習使用vue+element-ui框架進行前端開發,使用表格el-table較為多,有些業務邏輯比較相似,有些地方使用的重復性高,如果多個頁面使用相同的功能,就要多次重復寫邏輯上差不多的代碼,所以打算對表格這個組件進行封裝,將相同的代碼和邏輯封裝在一起,把不同的業務邏輯抽離出來,話不多說,下面就來實作一下吧,

一、原生el-tbale代碼——簡單の封裝

這里直接參考官方的基礎使用模板,直接抄過來(?ω?),下面代碼中主要是抽離html部分,可以看出每個el-table-column中都含有prop、label、width屬性,只不過這些屬性值不太一樣罷了,其余的部分都差不多一樣,所以表頭(表格每列el-table-column的定義)這里可以封裝一下,把不同的地方封裝成一個陣列物件結構,然后通過for回圈來完成html中的部分,

  • 封裝前
  <template>
    <el-table
      :data="https://www.cnblogs.com/zero-dg/archive/2021/01/18/tableData"
      style="width: 100%">
      <el-table-column
        prop="date"
        label="日期"
        width="180">
      </el-table-column>
      <el-table-column
        prop="name"
        label="姓名"
        width="180">
      </el-table-column>
      <el-table-column
        prop="address"
        label="地址">
      </el-table-column>
    </el-table>
  </template>

  <script>
    export default {
      data() {
        return {
          tableData: [{
            date: '2016-05-02',
            name: '王小虎',
            address: '上海市普陀區金沙江路 1518 弄'
          }, {
            date: '2016-05-04',
            name: '王小虎',
            address: '上海市普陀區金沙江路 1517 弄'
          }, {
            date: '2016-05-01',
            name: '王小虎',
            address: '上海市普陀區金沙江路 1519 弄'
          }, {
            date: '2016-05-03',
            name: '王小虎',
            address: '上海市普陀區金沙江路 1516 弄'
          }]
        }
      }
    }
  </script>
  • 表格の樣子
    表格展示

  • 封裝后

<template>
  <el-table :data="https://www.cnblogs.com/zero-dg/archive/2021/01/18/tableData" style="width: 100%">
    <template v-for="(item, key) in header">
      <el-table-column
        :key="key"
        :prop="itm.prop ? itm.prop : null"
        :label="itm.label ? itm.label : null"
        :width="itm.width ? itm.width : null"
      >
      </el-table-column>
    </template>
  </el-table>
</template>

<script>
export default {
  data() {
    return {
      header: [
        { prop: "date", label: "日期", width: "180" },
        { prop: "name", label: "姓名", width: "180" },
        { prop: "address", label: "地址" }
      ],
      tableData: [
        {
          date: "2016-05-02",
          name: "王小虎",
          address: "上海市普陀區金沙江路 1518 弄"
        },
        {
          date: "2016-05-04",
          name: "王小虎",
          address: "上海市普陀區金沙江路 1517 弄"
        },
        {
          date: "2016-05-01",
          name: "王小虎",
          address: "上海市普陀區金沙江路 1519 弄"
        },
        {
          date: "2016-05-03",
          name: "王小虎",
          address: "上海市普陀區金沙江路 1516 弄"
        }
      ]
    };
  }
};
</script>

現在資料還比較少,可能看不出封裝組件封裝的優勢,但是相對于之前代碼,這里邏輯上看起來更加清晰,而且修改列的時候直接改動data中的header資料即可,不用再去html代碼中去“開刀”( ̄▽ ̄)/,上面是最最最簡單的封裝了,嚴格來說只是簡單的抽離了一下代碼中的資料結構,在正常的業務中肯定不止這么簡單的封裝,接下來才是重點─━ _ ─━?

二、el-tbale代碼——復雜の封裝

在真正的開發程序中,表格不僅僅要展示資料,還要完成一些額外的任務,比如CRUD(增刪改查操作)和資料格式轉化,表格內每一條資料都有可能被單獨修改或者執行一些功能性的互動,這時候就要在單元格內內嵌一些按鈕、輸入框、標簽等等的代碼,element官方給出的方法是使用插槽slot,獲取對應行的資料使用slot-scope,在對應的列中設定相應的代碼,但是這里給我們二次封裝就會帶來不小的問題,如果只是單純的修改資料的格式使用官方提供的formatter屬性還可以實作,但是要內嵌代碼就會比較麻煩,內嵌代碼必然就會帶來封裝上的困難,這也是我在封裝代碼的時候遇到的最大的阻礙,如果要想封裝好這個表格,就必須將這部分代碼抽離出組件外,在查詢閱讀了大量博客之后(其實是我菜了,學藝不精(T▽T)),我終于找到了將內嵌代碼剝離出組件的方法ヾ(????)?",那就是render函式,關于render可以參考一下這篇博客,使用render函式就可以輕而易舉的將這部分邏輯代碼抽離出來了,

el-table真正の二次封裝

  • 二次封裝源代碼
<template>
  <el-table
    empty-text="暫無資料"
    ref="table"
    :data="https://www.cnblogs.com/zero-dg/archive/2021/01/18/tableList"
    border
    stripe
    fit
    highlight-current-row
    :height="inTableHeight"
    @selection-change="selectionChange"
    @row-click="rowClick"
  >
    <!-- 選擇框 -->
    <el-table-column
      v-if="select"
      type="selection"
      fixed="left"
      width="55"
      align="center"
    />
    <template v-for="(itm, idx) in header">
      <!-- 特殊處理列 -->
      <el-table-column
        v-if="itm.render"
        :key="idx"
        :prop="itm.prop ? itm.prop : null"
        :label="itm.label ? itm.label : null"
        :width="itm.width ? itm.width : null"
        :sortable="itm.sortable ? itm.sortable : false"
        :align="itm.align ? itm.align : 'center'"
        :fixed="itm.fixed ? itm.fixed : null"
        :show-overflow-tooltip="itm.tooltip"
        min-width="50"
      >
        <template slot-scope="scope">
          <ex-slot
            :render="itm.render"
            :row="scope.row"
            :index="scope.$index"
            :column="itm"
          />
        </template>
      </el-table-column>
      <!-- 正常列 -->
      <el-table-column
        v-else
        :key="idx"
        :prop="itm.prop ? itm.prop : null"
        :label="itm.label ? itm.label : null"
        :width="itm.width ? itm.width : null"
        :sortable="itm.sortable ? itm.sortable : false"
        :align="itm.align ? itm.align : 'center'"
        :fixed="itm.fixed ? itm.fixed : null"
        :formatter="itm.formatter"
        :show-overflow-tooltip="itm.tooltip"
        min-width="50"
      />
    </template>
  </el-table>
</template>

<script>
// 自定義內容的組件
var exSlot = {
  functional: true,
  props: {
    row: Object,
    render: Function,
    index: Number,
    column: {
      type: Object,
      default: null
    }
  },
  render: (h, context) => {
    const params = {
      row: context.props.row,
      index: context.props.index
    };
    if (context.props.column) params.column = context.props.column;
    return context.props.render(h, params);
  }
};

export default {
  components: { exSlot },
  props: {
    tableList: {
      type: Array,
      default: () => []
    },
    header: {
      type: Array,
      default: () => []
    },
    select: {
      type: Boolean,
      default: () => false
    },
    height: {
      type: [Number, String, Function],
      default: () => null
    }
  },
  data() {
    return {
      inTableHeight: null
    };
  },
  created() {
    //該階段可以接收父組件的傳遞引數
    this.inTableHeight = this.height;
  },
  mounted() {
    this.$nextTick(() => {
      //表格高度自適應瀏覽器大小
      this.changeTableHight();
      if (!this.height) {
        window.onresize = () => {
          this.changeTableHight();
        };
      }
    });
  },
  destroyed() {
    //高度自適應事件注銷
    window.onresize = null;
  },
  watch: {
    /**
     * 資料變化后 高度自適應
     */
    tableList() {
      this.$nextTick(() => {
        this.changeTableHight();
      });
    }
  },
  methods: {
    /**
     * 選擇框選擇后更改,事件分發
     */
    selectionChange(selection) {
      this.$emit("selection-change", selection);
    },
    /**
     * 點擊事件
     */
    rowClick(row, column, event) {
      this.$emit("row-click", row, column, event);
    },
    /**
     * 高度自適應
     * 當表格展示空間小于460按460px展示,大于的時候高度填充
     */
    changeTableHight() {
      if (this.height) {
        //如果有傳進來高度就取消自適應
        this.inTableHeight = this.height;
        this.$refs.table.doLayout();
        return;
      }
      let tableHeight = window.innerHeight || document.body.clientHeight;
      //高度設定
      let disTop = this.$refs.table.$el;
      //如果表格上方有元素則減去這些高度適應視窗,66是底下留白部分
      tableHeight -= disTop.offsetTop + 66;
      if (disTop.offsetParent) tableHeight -= disTop.offsetParent.offsetTop;
      this.inTableHeight = tableHeight < 460 ? 460 : tableHeight;
      //重繪表格
      this.$refs.table.doLayout();
    }
  }
};
</script>
<style></style>
  • 封裝代碼的相關解釋

以上就是我封裝的代碼,部分屬性或者方法由于沒有使用到所以我就沒有將對應的方法和屬性封裝進去,如果你們開發中有用到對應的地方其實可以照貓畫虎的填上去即可,我封裝表格的時候在屬性這里使用了三目運算子,用于做一些兼容,如果不傳對應的屬性就給個默認值,比如align屬性,我設定的是默認居中,還有就是方法,在表格的方法參考方面,其實就是把官方的方法用$emit事件將對應的引數和方法名用同樣的方法分發給父組件,這樣父組件使用完全可以參照element官方檔案使用這些方法,在組件內我只是進行了一次轉發而已,我自己寫的時候并沒有用到太多的方法,所以只封裝了一兩個,如果有需要可以自行添加,除了上述兩個封裝,有一個特別的地方就是勾選框,不能放在回圈內,不然會出現錯誤,可能是索引的問題吧,所以我單獨使用一個引數來控制是否顯示選擇框,另外就是,在公司產品要求表格能夠自適應頁面的高度,這個功能我也是修改了好久,460是最小的高度,關于高度自適應的全部在changeTableHight()方法中,如果不需要這個功能,將函式和所有參考該函式的地方洗掉即可,

height:如果不傳入這個屬性,那么表格高度就如上面所說的是自適應高度,可以通過這個屬性來指定表格的高度,

formatter:這個屬性在列中如果使用插槽就會失效,所以我設定了兩個列,如果有render方法說明單元格要內嵌代碼,就是用特殊列,反之就是正常列,所以formatterrender不能同時使用,

render:終于到了最關鍵的地方了( ̄▽ ̄)/,這個可是我封裝表格的最大難點了,render對我個人理解而言就是虛擬結點,在DOM和CSSOM樹合并為render樹的階段,對代碼進行修改,

以上就是我封裝表格的詳細解釋了,可能有遺漏的部分,畢竟封裝這個表格也讓我學了不少東西,所以之前有些地方可能解釋不清楚或者不到位,還望各位大佬指正,

三、父組件參考封裝的組件

封裝這么久的組件,當然要使用起來才知道的到底好不好用,關于參考方面,首先要在參考的地方進行組件注冊,如果全域注冊過了,可以忽略在區域注冊,關于組件的注冊這里就不做詳解了(o?▽?)o ,我使用了全域注冊,所以這里是直接引入我自己封裝好的組件,

<template>
  <div >
    <xd-table :table-list="tableData" :header="header" height="300"></xd-table>
  </div>
</template>

<script>
export default {
  name: "HelloWorld",
  data() {
    return {
      header: [
        { prop: "w", label: "w" },
        { prop: "x", label: "x",
          formatter: (row) => {
            return row.x.toFixed(3);
          },
        },
        { prop: "d", label: "d",
          formatter: (row) => {
            return row.d.toFixed(2);
          },
        },
        {
          label: "操作",
          render: (h, data) => {
            return (
              <el-button
                type="primary"
                onClick={() => {
                  this.handleClick(data.row);
                }}
              >
                點我獲取行資料
              </el-button>
            );
          },
        },
      ],
      tableData: [
        { w: 1, x: 99.25123, d: 0.23892 },
        { w: 1, x: 255.6666, d: 0.99134 },
      ],
    };
  },
  methods: {
    handleClick(row) {
      console.log(row);
    },
  },
};
</script>

參考組件之前一定要記得先注冊,這里我只使用了幾個屬性,其他屬性沒有使用,因為是demo,主要還是展示render內嵌代碼的方法,還有一個就是官方formatter方法的使用,

有一個需要注意點就是render內我使用了JSX模板語法這里需要在VUE專案中單獨去配置一下JSX語法,如果不想使用JSX,直接寫也可以,因為不使用JSX語法寫出來的內嵌模板代碼比較難讀所以我就不展示了,個人建議還是使用JSX語法,雖然和原生vue有些地方使用方法不太一樣,

  • 效果截圖

組件使用截圖

結語

這次封裝vue組件,花了我將近半個月的時間,從剛開始的接觸到發現bug然后去修改,來來回回終于精簡封裝出現在這個二次封裝的組件,可能對于熟悉vue和element的人來說這個封裝其實很簡單,但是對于我來說這個算是我這段實習期封裝的比較好的一個組件了吧,當然除了封裝這個組件還有別的事在做,不可能放著公司的活不干(哈哈哈哈︿( ̄︶ ̄)︿),總之在封裝組件程序中學習到了不少東西,也算是一個大大的進步,于是來寫一篇博客來記錄一下,

轉載請註明出處,本文鏈接:https://www.uj5u.com/qiye/250557.html

標籤:其他

上一篇:(2) 使用phpstudy 實作局域網內遠程訪問本地ThreeJS示例

下一篇:進口鑄鐵氣動隔膜泵原理特點及品牌(圖)

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • IEEE1588PTP在數字化變電站時鐘同步方面的應用

    IEEE1588ptp在數字化變電站時鐘同步方面的應用 京準電子科技官微——ahjzsz 一、電力系統時間同步基本概況 隨著對IEC 61850標準研究的不斷深入,國內外學者提出基于IEC61850通信標準體系建設數字化變電站的發展思路。數字化變電站與常規變電站的顯著區別在于程序層傳統的電流/電壓互 ......

    uj5u.com 2020-09-10 03:51:52 more
  • HTTP request smuggling CL.TE

    CL.TE 簡介 前端通過Content-Length處理請求,通過反向代理或者負載均衡將請求轉發到后端,后端Transfer-Encoding優先級較高,以TE處理請求造成安全問題。 檢測 發送如下資料包 POST / HTTP/1.1 Host: ac391f7e1e9af821806e890 ......

    uj5u.com 2020-09-10 03:52:11 more
  • 網路滲透資料大全單——漏洞庫篇

    網路滲透資料大全單——漏洞庫篇漏洞庫 NVD ——美國國家漏洞庫 →http://nvd.nist.gov/。 CERT ——美國國家應急回應中心 →https://www.us-cert.gov/ OSVDB ——開源漏洞庫 →http://osvdb.org Bugtraq ——賽門鐵克 →ht ......

    uj5u.com 2020-09-10 03:52:15 more
  • 京準講述NTP時鐘服務器應用及原理

    京準講述NTP時鐘服務器應用及原理京準講述NTP時鐘服務器應用及原理 安徽京準電子科技官微——ahjzsz 北斗授時原理 授時是指接識訓通過某種方式獲得本地時間與北斗標準時間的鐘差,然后調整本地時鐘使時差控制在一定的精度范圍內。 衛星導航系統通常由三部分組成:導航授時衛星、地面檢測校正維護系統和用戶 ......

    uj5u.com 2020-09-10 03:52:25 more
  • 利用北斗衛星系統設計NTP網路時間服務器

    利用北斗衛星系統設計NTP網路時間服務器 利用北斗衛星系統設計NTP網路時間服務器 安徽京準電子科技官微——ahjzsz 概述 NTP網路時間服務器是一款支持NTP和SNTP網路時間同步協議,高精度、大容量、高品質的高科技時鐘產品。 NTP網路時間服務器設備采用冗余架構設計,高精度時鐘直接來源于北斗 ......

    uj5u.com 2020-09-10 03:52:35 more
  • 詳細解讀電力系統各種對時方式

    詳細解讀電力系統各種對時方式 詳細解讀電力系統各種對時方式 安徽京準電子科技官微——ahjzsz,更多資料請添加VX 衛星同步時鐘是我京準公司開發研制的應用衛星授時時技術的標準時間顯示和發送的裝置,該裝置以M國全球定位系統(GLOBAL POSITIONING SYSTEM,縮寫為GPS)或者我國北 ......

    uj5u.com 2020-09-10 03:52:45 more
  • 如何保證外包團隊接入企業內網安全

    不管企業規模的大小,只要企業想省錢,那么企業的某些服務就一定會采用外包的形式,然而看似美好又經濟的策略,其實也有不好的一面。下面我通過安全的角度來聊聊使用外包團的安全隱患問題。 先看看什么服務會使用外包的,最常見的就是話務/客服這種需要大量重復性、無技術性的服務,或者是一些銷售外包、特殊的職能外包等 ......

    uj5u.com 2020-09-10 03:52:57 more
  • PHP漏洞之【整型數字型SQL注入】

    0x01 什么是SQL注入 SQL是一種注入攻擊,通過前端帶入后端資料庫進行惡意的SQL陳述句查詢。 0x02 SQL整型注入原理 SQL注入一般發生在動態網站URL地址里,當然也會發生在其它地發,如登錄框等等也會存在注入,只要是和資料庫打交道的地方都有可能存在。 如這里http://192.168. ......

    uj5u.com 2020-09-10 03:55:40 more
  • [GXYCTF2019]禁止套娃

    git泄露獲取原始碼 使用GET傳參,引數為exp 經過三層過濾執行 第一層過濾偽協議,第二層過濾帶引數的函式,第三層過濾一些函式 preg_replace('/[a-z,_]+\((?R)?\)/', NULL, $_GET['exp'] (?R)參考當前正則運算式,相當于匹配函式里的引數 因此傳遞 ......

    uj5u.com 2020-09-10 03:56:07 more
  • 等保2.0實施流程

    流程 結論 ......

    uj5u.com 2020-09-10 03:56:16 more
最新发布
  • 使用Django Rest framework搭建Blog

    在前面的Blog例子中我們使用的是GraphQL, 雖然GraphQL的使用處于上升趨勢,但是Rest API還是使用的更廣泛一些. 所以還是決定回到傳統的rest api framework上來, Django rest framework的官網上給了一個很好用的QuickStart, 我參考Qu ......

    uj5u.com 2023-04-20 08:17:54 more
  • 記錄-new Date() 我忍你很久了!

    這里給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 大家平時在開發的時候有沒被new Date()折磨過?就是它的諸多怪異的設定讓你每每用的時候,都可能不小心踩坑。造成程式意外出錯,卻一下子找不到問題出處,那叫一個煩透了…… 下面,我就列舉它的“四宗罪”及應用思考 可惡的四宗罪 1. Sa ......

    uj5u.com 2023-04-20 08:17:47 more
  • 使用Vue.js實作文字跑馬燈效果

    實作文字跑馬燈效果,首先用到 substring()截取 和 setInterval計時器 clearInterval()清除計時器 效果如下: 實作代碼如下: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta ......

    uj5u.com 2023-04-20 08:12:31 more
  • JavaScript 運算子

    JavaScript 運算子/運算子 在 JavaScript 中,有一些運算子可以使代碼更簡潔、易讀和高效。以下是一些常見的運算子: 1、可選鏈運算子(optional chaining operator) ?.是可選鏈運算子(optional chaining operator)。?. 可選鏈操 ......

    uj5u.com 2023-04-20 08:02:25 more
  • CSS—相對單位rem

    一、概述 rem是一個相對長度單位,它的單位長度取決于根標簽html的字體尺寸。rem即root em的意思,中文翻譯為根em。瀏覽器的文本尺寸一般默認為16px,即默認情況下: 1rem = 16px rem布局原理:根據CSS媒體查詢功能,更改根標簽的字體尺寸,實作rem單位隨螢屏尺寸的變化,如 ......

    uj5u.com 2023-04-20 08:02:21 more
  • 我的第一個NPM包:panghu-planebattle-esm(胖虎飛機大戰)使用說明

    好家伙,我的包終于開發完啦 歡迎使用胖虎的飛機大戰包!! 為你的主頁添加色彩 這是一個有趣的網頁小游戲包,使用canvas和js開發 使用ES6模塊化開發 效果圖如下: (覺得圖片太sb的可以自己改) 代碼已開源!! Git: https://gitee.com/tang-and-han-dynas ......

    uj5u.com 2023-04-20 08:01:50 more
  • 如何在 vue3 中使用 jsx/tsx?

    我們都知道,通常情況下我們使用 vue 大多都是用的 SFC(Signle File Component)單檔案組件模式,即一個組件就是一個檔案,但其實 Vue 也是支持使用 JSX 來撰寫組件的。這里不討論 SFC 和 JSX 的好壞,這個仁者見仁智者見智。本篇文章旨在帶領大家快速了解和使用 Vu ......

    uj5u.com 2023-04-20 08:01:37 more
  • 【Vue2.x原始碼系列06】計算屬性computed原理

    本章目標:計算屬性是如何實作的?計算屬性快取原理以及洋蔥模型的應用?在初始化Vue實體時,我們會給每個計算屬性都創建一個對應watcher,我們稱之為計算屬性watcher ......

    uj5u.com 2023-04-20 08:01:31 more
  • http1.1與http2.0

    一、http是什么 通俗來講,http就是計算機通過網路進行通信的規則,是一個基于請求與回應,無狀態的,應用層協議。常用于TCP/IP協議傳輸資料。目前任何終端之間任何一種通信方式都必須按Http協議進行,否則無法連接。tcp(三次握手,四次揮手)。 請求與回應:客戶端請求、服務端回應資料。 無狀態 ......

    uj5u.com 2023-04-20 08:01:10 more
  • http1.1與http2.0

    一、http是什么 通俗來講,http就是計算機通過網路進行通信的規則,是一個基于請求與回應,無狀態的,應用層協議。常用于TCP/IP協議傳輸資料。目前任何終端之間任何一種通信方式都必須按Http協議進行,否則無法連接。tcp(三次握手,四次揮手)。 請求與回應:客戶端請求、服務端回應資料。 無狀態 ......

    uj5u.com 2023-04-20 08:00:32 more