Vue.js中el-table表格自定义列控制与拖拽

 更新时间:2023年05月23日 09:20:25   作者:Kier  
本文主要介绍了Vue.js中el-table表格自定义列控制与拖拽,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

背景

昨天客户:提了一个需求,你们这个表格的数据太乱不是我们重点关注的,然后列出了一些属性值A,B,C,D。。。这些必须展示。

y0rth-404du.gif

前端仔:简单,30s给你安排上。

今天客户:你们这个表格的列字段不能拖动呀,我们觉得这个几个字段才是我们重点关注的,我要放到前面。。

前端仔:无语,行给你安排。

这些无聊的需求不改又不行,但是自己又不想被他们拖着天天伺候,直接给他们自己可以配置,想咋弄就咋弄。

需求与实现分析(重点)

面对客户的这些需求,总结起来也就是2个功能点,一是表格的列可以自由拖拽,二表格可以展示的列数据可以进行自定义控制。考虑到组件可能会在很多地方使用,所以使用父子传值的模式来进行数据传输。

功能实现

自由拖拽:偷个懒直接应用别人封装好的sortable.js。地址:www.sortablejs.com/

引入

npm install sortablejs

使用

import Sortable from "sortablejs";

**列控制:**这里考虑使用leement的多选框组el-check-group,我们先自定义一部分默认显示的数据,还有一部分由客户自己勾选设置,这样既可以避免表格内容过多,导致出现横向滚动条的丑陋,也为客户提供了选择性。

上组件代码

<template>
  <div class="myWrap">
    <div class="colum-set">
      <el-popover width="150" placement="bottom-end" trigger="click">
        <div>
          <el-checkbox-group v-model="childCheckedItems" @change="handleCheckedColumChange">
            <el-checkbox v-for="item in tableHeader" :label="item.label" :key="item.label"></el-checkbox>
          </el-checkbox-group>
        </div>
        <el-button slot="reference" size="mini" icon="el-icon-setting">{{ $t('common.column_set') }}</el-button>
      </el-popover>
    </div>
    <el-table :data="tableBody" @selection-change="handleSelectionChange" border row-key="id" :header-cell-style="{
      height: '48px',
      background: '#DFEBF9',
      color: '#606266',
      fontWeight: 'bold',
      fontSize: '14px',
    }">
      <!-- 勾选框列 -->
      <el-table-column type="selection" width="48" fixed v-if="needCheckBox"></el-table-column>
      <!-- 表头列 -->
      <el-table-column v-for="(item, index) in checkedHeader" :key="index" :prop="item.prop" :label="item.label"
        width="auto" show-overflow-tooltip>
      </el-table-column>
      <!-- 表格操作 -->
      <el-table-column :width="tableOperate.width" :label="tableOperate.label" fixed="right" v-if="tableOperate.show">
        <template slot-scope="scope">
          <slot name="tablerow" :row="scope.row"></slot>
        </template>
      </el-table-column>
    </el-table>
  </div>
</template>
<script>
// 引入sortablejs插件,拖拽主要靠的是这个插件
import Sortable from "sortablejs";
export default {
  props: {
    tableBody: {
      required: true,
      type: Array
    },
    tableHeader: {
      type: Array,
      required: true
    },
    needCheckBox: {
      required: true,
      type: Boolean
    },
    tableOperate:{
      required:true,
      type:Object
    }
  },
  data() {
    return {
      checkedHeader: [],
      childCheckedItems: []
    };
  },
  watch: {
    '$store.state.common.lang': function (val) {
      window.location.reload()
    },
    tableHeader: {
      handler(newVal, oldVal) {
        console.log(newVal)
        this.checkedHeader = newVal.filter(item => item.show == true)
        this.childCheckedItems = this.checkedHeader.map(item => {
          return item.label
        })
      },
      immediate: true,
      deep: true
    }
  },
  mounted() {
    // 列的拖拽初始化
    this.columnDropInit();
  },
  methods: {
    //列拖拽
    columnDropInit() {
      // 第一步,获取列容器
      const wrapperColumn = document.querySelector(
        ".el-table__header-wrapper tr"
      );
      // 第二步,给列容器指定对应拖拽规则
      this.sortable = Sortable.create(wrapperColumn, {
        animation: 200,
        delay: 0,
        onEnd: (event) => {
          console.log(
            "根据旧索引和新索引去更新,其实就是交换位置",
            event.oldIndex,
            event.newIndex
          );
          // 接下来做索引的交换
          let tempHableHeader = [...this.checkedHeader]; // 先存一份临时变量表头数据
          let temp; // 临时变量用于交换
          // 这里预留一个变量,因为表头前加了表格多选,且固定。那么得-1,如果还有2列,则-2,以此类推
          temp = tempHableHeader[event.oldIndex - this.control_index]; 
          tempHableHeader[event.oldIndex - this.control_index] =
            tempHableHeader[event.newIndex - this.control_index]; 
          tempHableHeader[event.newIndex - this.control_index] = temp;
          // 这里一定要先把表头数据清空,然后再下一轮中去赋值,bug处理
          this.checkedHeader = []
          this.$nextTick(() => {
            this.checkedHeader = tempHableHeader
            this.childCheckedItems = this.checkedHeader.map(item => {
              return item.label
            })
          })
        },
      });
    },
    handleCheckedColumChange(value) {
      this.$emit('changeCheckedColum', value)
    },
    handleSelectionChange(value){
      console.log(value)
    }
  },
};
</script>
<style lang='stylus' scoped>
.myWrap {
  width: 100%;
  height: 100%;
  box-sizing: border-box;
  .colum-set{
    float: right;
  }
  /deep/ .el-table {
    .el-table__header-wrapper {
      tr {
        th {
          cursor: move;
        }
      }
    }
  }
}
.check-colum-div{
  width: 150px;
  display: flex;
  flex-direction: column;
}
</style>

父组件核心代码

<template>
    <MyTable :tableBody="matlist" :tableHeader="tableHeader" :needCheckBox="true" :tableOperate="tableOperate"
    @changeCheckedColum="changeCheckedColum" >
    <template v-slot:tablerow="scope">
    <el-button type="primary" size="small" @click="editRow(scope.row)">
    编辑
    </el-button>
    </template>
    </MyTable>
</template>
<script>
tableHeader: [
        { label: this.$t('storage.shelf_id'), prop: 'shelf_id', show: true },
        { label: this.$t('storage.save_id'), prop: 'save_id', show: false },
        { label: this.$t('storage.label_code'), prop: 'label_code', show: true },
        { label: this.$t('storage.part_num'), prop: 'part_num', show: true },  
        { label: this.$t('storage.lot_code'), prop: 'lot_code', show: true },
        { label: this.$t('storage.quantity'), prop: 'quantity', show: true },
        { label: this.$t('storage.description'), prop: 'description', show: true },
        { label: this.$t('storage.supplier_name'), prop: 'supplier_name', show: true },
        { label: this.$t('storage.mfg_date'), prop: 'mfg_date', show: true },
        { label: this.$t('storage.in_time'), prop: 'in_time', show: true },
        { label: this.$t('storage.position'), prop: 'position_info', show: true },
        { label: this.$t('storage.category_name'), prop: 'category_name', show: false },
        { label: this.$t('storage.warehouse_name'), prop: 'warehouse_name',show:false },
      ],
      tableOperate:{
        show:true,
        width:120,
        label:this.$t('common.operation')
      },
</script>

核心代码分析(重点)

columnDropInit(),该方法由sortable提供,实现了表格的拖拽功能,我这里只写了列拖拽,行拖拽可以自己在官网查看。

watch中的tableHeader,为什么要这么做?因为表头所展示的数据,和多选框中被选中项是响应关系,勾选与取消的同时需要相应到表格列的变化。其次表头绑定的是对象数组,额checkbox绑定值只能是字符串或者数字,原本想直接绑定个对象数组的计划泡汤,这是elementui设计决定的。(从elementui源码上解决参考:https://www.jb51.net/javascript/2852009x5.htm)

tableHeader: {
      handler(newVal, oldVal) {
        console.log(newVal)
        this.checkedHeader = newVal.filter(item => item.show == true)
        this.childCheckedItems = this.checkedHeader.map(item => {
          return item.label
        })
      },
      immediate: true,
      deep: true
    }

handleCheckedColumChange,子组件通过$emit触发父组件的数据变更

handleCheckedColumChange(value) {
      this.$emit('changeCheckedColum', value)
},

父组件数据变更,子组件的watch,再次对数据进行监听改变

 changeCheckedColum(value) {
      this.tableHeader.forEach(item => {
        if (value.includes(item.label)) {
          item.show = true
        } else {
          item.show = false
        }
      })
  },

额外预留功能点

表格多选

微信截图_20230512154609.png

<!-- 勾选框列 -->
<el-table-column type="selection" width="48" fixed v-if="needCheckBox"></el-table-column>

上面的勾选框主要用来对表格的的数据进行多选操作,方便以后预留使用,通过needCheckBox来控制是否展示,同时子组件监听该属性,来计算表格拖拽时的偏移量,在columnDropInit方法中有详细注释,计算属性方法如下:

computed: {
    control_index: function () {
      return this.needCheckBox ? 1 : 0
    },
},

表格操作,增删查改

微信截图_20230512160101.png

<!-- 表格操作 子组件-->
<el-table-column :width="tableOperate.width" :label="tableOperate.label" fixed="right" v-if="tableOperate.show">
    <template slot-scope="scope">
    <slot name="tablerow" :row="scope.row"></slot>
    </template>
</el-table-column>
<!--父组件,插槽取值-->
 <template v-slot:tablerow="scope">
       <el-button type="primary" size="small" @click="editRow(scope.row)">
        {{ $t('common.edit') }}
       </el-button>
 </template>

这个插槽预留,为了方便表格对数据的增删查改,通过插槽传值,获取当前行的row。

最后

本文主要的是通过,对需求的拆分与理解,从表格的拖拽,以及列控制,衍生出开发者个人的开发习惯,对组建的预留以及扩展,方便以后使用。再技术上也有多方面的考量,对大都数开发者也有一定的学习和参考价值。好了,本文到此结束,希望对你有帮助。

到此这篇关于Vue.js中el-table表格自定义列控制与拖拽的文章就介绍到这了,更多相关Vue.js el-table自定义列控制与拖拽内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • vue3+echarts实现渐变色环形图过程

    vue3+echarts实现渐变色环形图过程

    这篇文章主要介绍了vue3+echarts实现渐变色环形图过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-08-08
  • vue项目history模式下部署子路由跳转失败的解决

    vue项目history模式下部署子路由跳转失败的解决

    这篇文章主要介绍了vue项目history模式下部署子路由跳转失败的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-03-03
  • vue2自定义组件通过rollup配置发布到npm的详细步骤

    vue2自定义组件通过rollup配置发布到npm的详细步骤

    这篇文章主要介绍了vue2自定义组件通过rollup配置发布到npm,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-03-03
  • vue3中ace-editor的简单使用方法实例

    vue3中ace-editor的简单使用方法实例

    这篇文章主要给大家介绍了关于vue3中ace-editor简单使用的相关资料,ace-editor是一种代码编辑器,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2023-10-10
  • Vue如何使用混合Mixins和插件开发详解

    Vue如何使用混合Mixins和插件开发详解

    这篇文章主要介绍了Vue如何使用混合Mixins和插件开发详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-02-02
  • 关于vuejs中v-if和v-show的区别及v-show不起作用问题

    关于vuejs中v-if和v-show的区别及v-show不起作用问题

    v-if 有更高的切换开销,而 v-show 有更高的出事渲染开销.因此,如果需要非常频繁的切换,那么使用v-show好一点;如果在运行时条件不太可能改变,则使用v-if 好点
    2018-03-03
  • vue实现竖屏滚动公告效果

    vue实现竖屏滚动公告效果

    这篇文章主要为大家详细介绍了vue实现竖屏滚动公告效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-04-04
  • Bootstrap+Vue滑动监听Scrollspy实现餐厅餐品展示

    Bootstrap+Vue滑动监听Scrollspy实现餐厅餐品展示

    本文主要介绍了Bootstrap+Vue滑动监听Scrollspy实现餐厅餐品展示,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-03-03
  • vue.js给动态绑定的radio列表做批量编辑的方法

    vue.js给动态绑定的radio列表做批量编辑的方法

    下面小编就为大家分享一篇vue.js给动态绑定的radio列表做批量编辑的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-02-02
  • 基于Vue3实现列表虚拟滚动效果

    基于Vue3实现列表虚拟滚动效果

    这篇文章主要为大家介绍了如何利用Vue3实现列表虚拟滚动效果,文中的示例代码讲解详细,对我们学习或工作有一定价值,需要的可以参考一下
    2022-04-04

最新评论