element-ui中的el-table合并单元格

描述:

在写项目的时候有时候会经常遇到把行和列合并起来的情况,因为有些数据是重复渲染的,不合并行列会使表格看起来非常的混乱,如下:

 而我们想要的数据是下面这种情况,将重复的行进行合并,使表格看起来简单明了,如下:

 


解决方案:

一:合并行

1:html部分

所谓的合并行就是将多行相同的数据变成一行来显示,页面的布局比较简单

<template><div class="table"><el-table :data="tableData" :span-method="objectSpanMethod" border style="width: 100%"><el-table-column prop="time" label="时间"></el-table-column><el-table-column prop="grade" label="年级"></el-table-column><el-table-column prop="name" label="姓名"></el-table-column><el-table-column prop="subjects" label="科目"></el-table-column><el-table-column prop="score" label="成绩"></el-table-column></el-table></div>
</template>

2:模拟data数据

span-methodel-table上属性,其值是一个函数,objectSpanMethod方法是用来处理合并行的返回值,tableData数据如下

tableData: [{time:'2020-08-10',grade:'三年二班',name: '小明',subjects:'语文',score: 80 },{time:'2020-08-10',grade:'三年二班',name: '小明',subjects:'数学',score: 80 },{time:'2020-08-10',grade:'三年一班',name: '小雷',subjects:'语文',score: 70 },{time:'2020-08-10',grade:'三年一班',name: '小雷',subjects:'数学',score: 80 },{time:'2020-08-11',grade:'三年三班',name: '小花',subjects:'语文',score: 60 },{time:'2020-08-11',grade:'三年三班',name: '小花',subjects:'数学',score: 60 },],mergeObj: {}, // 用来记录需要合并行的下标tableProps: ['time', 'grade', 'name', 'subjects', 'score'] // 表格中的列名

3:梳理数据以及方法调用

首先需要对数据就行处理,就是比较当前行与上一行的值是否相等(如果是第一行数据,直接将值赋值为1)

watch中监听表格中的数据,当不为空的的时候,调用数据初始化数据的方法,如下:

watch:{"tableData":function (newVal,oldVal){console.log("nnnnnnnnnnnn",newVal)console.log("oooooooooooo",oldVal)if(newVal.length>0){this.getSpanArr(this.tableData);}}},
// getSpanArr方法
getSpanArr(data) {this.tableProps.forEach((key, index1) => {let count = 0; // 用来记录需要合并行的起始位置this.mergeObj[key] = []; // 记录每一列的合并信息data.forEach((item, index) => {// index == 0表示数据为第一行,直接 push 一个 1if(index === 0) {this.mergeObj[key].push(1); } else {/*判断当前行是否与上一行其值相等 如果相等 在 count 记录的位置其值 +1表示当前行需要合并 并push 一个 0 作为占位*/  if(item[key] === data[index - 1][key]) { this.mergeObj[key][count] += 1;this.mergeObj[key].push(0);} else {// 如果当前行和上一行其值不相等 count = index; // 记录当前位置 this.mergeObj[key].push(1); // 重新push 一个 1}}})})
}

数据处理好之后就可以调用objectSpanMethod方法了,如下:

// objectSpanMethod方法/*默认接受四个值----row==当前行的数据----column==当前列的数据----rowIndex==行的下标----columnIndex==列的下标*/
// 默认接受四个值 { 当前行的值, 当前列的值, 行的下标, 列的下标 }
objectSpanMethod({ row, column, rowIndex, columnIndex }) {// 判断列的属性if(this.tableProps.indexOf(column.property) !== -1) { // 判断其值是不是为0 if(this.mergeObj[column.property][rowIndex]) { return [this.mergeObj[column.property][rowIndex], 1]} else {// 如果为0则为需要合并的行return [0, 0]; }// 只有 第一列 第二列 第三列 合并行// if(columnIndex===1||columnIndex===2||columnIndex===3){//   // 判断列的属性//   if(this.tableProps.indexOf(column.property) !== -1) {//     // 判断其值是不是为0 //     if(this.mergeObj[column.property][rowIndex]) { //       return {//         rowspan: this.mergeObj[column.property][rowIndex],//         colspan: 1//       };//     } else {//       // 如果为0则为需要合并的行//       return {//         rowspan: 0,//         colspan: 0//       }; //     }//   }// }}
}

4:效果图

合并后的结果就是我们想要的形式:

 5:合并行的完整代码

<template><div class="table"><el-table :data="tableData" :span-method="objectSpanMethod" border style="width: 100%"><el-table-column prop="time" label="时间"></el-table-column><el-table-column prop="grade" label="年级"></el-table-column><el-table-column prop="name" label="姓名"></el-table-column><el-table-column prop="subjects" label="科目"></el-table-column><el-table-column prop="score" label="成绩"></el-table-column></el-table></div>
</template><script>
export default {name: 'Table',data() {return {tableData: [{ time: '2020-08-10', grade: '三年二班', name: '小明', subjects: '语文', score: 80 },{ time: '2020-08-10', grade: '三年二班',name: '小明', subjects: '数学', score: 80 },{ time: '2020-08-10', grade: '三年一班',name: '小雷', subjects: '语文', score: 70 },{ time: '2020-08-10', grade: '三年一班', name: '小雷', subjects: '数学', score: 80 },{ time: '2020-08-11', grade: '三年三班', name: '小花', subjects: '语文', score: 60 }, { time: '2020-08-11', grade: '三年三班', name: '小花', subjects: '数学', score: 60 }, ],mergeObj: {},mergeArr: ['time', 'grade', 'name', 'subjects', 'score'],};},watch:{"tableData":function (newVal,oldVal){console.log("nnnnnnnnnnnn",newVal)console.log("oooooooooooo",oldVal)if(newVal.length>0){this.getSpanArr(this.tableData);}}},methods: {getSpanArr(data) {this.mergeArr.forEach((key, index1) => {let count = 0; // 用来记录需要合并行的起始位置this.mergeObj[key] = []; // 记录每一列的合并信息data.forEach((item, index) => {// index == 0表示数据为第一行,直接 push 一个 1if(index === 0) {this.mergeObj[key].push(1); } else {/*判断当前行是否与上一行其值相等如果相等 在 count 记录的位置其值 +1表示当前行需要合并 并push 一个 0 作为占位 */   if(item[key] === data[index - 1][key]) { this.mergeObj[key][count] += 1;this.mergeObj[key].push(0);} else {// 如果当前行和上一行其值不相等 count = index; // 记录当前位置 this.mergeObj[key].push(1); // 重新push 一个 1}}})})},// 默认接受四个值 { 当前行的值, 当前列的值, 行的下标, 列的下标 }objectSpanMethod({ row, column, rowIndex, columnIndex }) {// 判断列的属性if(this.mergeArr.indexOf(column.property) !== -1) { // 判断其值是不是为0 if(this.mergeObj[column.property][rowIndex]) { return [this.mergeObj[column.property][rowIndex], 1]} else {// 如果为0则为需要合并的行return [0, 0]; }}}},};
</script><style lang="stylus" scoped>
.table height 100vhwidth 100%padding 40pxbox-sizing border-box/deep/ .el-table__body tr:hover > tdbackground-color: #fff;
</style>

二:合并行列

1:模拟data数据

span-method是el-table上属性,其值是一个函数,objectSpanMethod方法是用来处理合并行的返回值,tableData数据如下

tableData: [{ time: '2020-08-10', grade: '三年二班', name: '小明', subjects: '语文', score: 80 },{ time: '2020-08-10', grade: '三年二班', name: '小明', subjects: '数学', score: 80 }, { time: '2020-08-10', grade: '总成绩', name: '总成绩', subjects: '总成绩', score: 160 },{ time: '2020-08-10', grade: '三年一班', name: '小雷', subjects: '语文', score: 70 },{ time: '2020-08-10', grade: '三年一班', name: '小雷', subjects: '数学', score: 80 },{ time: '2020-08-10', grade: '总成绩', name: '总成绩', subjects: '总成绩', score: 150 }, { time: '2020-08-11', grade: '三年三班', name: '小花', subjects: '语文', score: 60 }, { time: '2020-08-11', grade: '三年三班', name: '小花', subjects: '数学', score: 60 }, { time: '2020-08-11', grade: '总成绩', name: '总成绩', subjects: '总成绩', score: 120 }
],

2:对比当一次的图片

可以看到上面的数据多了一行总成绩,现在的数据在页面显示效果如下:

3:html调整

可以看到总成绩的三个列并没有合并,并不是我们想要的效果,所以需要换一种思路来处理数据

页面的布局也有所调整,如下:

<template><div class="table"><el-table :data="tableData" :span-method="objectSpanMethods" border style="width: 100%"><template v-for="cols in colConfigs"><!-- 无需合并的列 --><el-table-columnv-if="cols.type === 'label' && !cols.children":key="cols.prop":prop="cols.prop":label="cols.label"></el-table-column><!-- 需要合并的列 --><template v-else-if="cols.type === 'label' && cols.children"><el-table-columnv-for="children in cols.children":key="children.prop":prop="children.prop":label="children.label"/></template></template></el-table></div>
</template>

4:在data中声明的变量

// 表格的信息 需要合并的需要放在 children 中
colConfigs: [{type: 'label',children: [{ prop: 'time', label: '时间' },{ prop: 'grade', label: '年级' },{ prop: 'name', label: '姓名' },{ prop: 'subjects', label: '科目' },{ prop: 'score', label: '成绩' }]}
],
// 需要合并的行列信息
mergeCols: [{ index: 0, name: 'time' },{ index: 1, name: 'grade' },{ index: 2, name: 'name' },{ index: 3, name: 'subjects' },{ index: 4, name: 'score' },
],
// 用来记录每一个单元格的下标
tableMergeIndex: [],

5:梳理数据以及方法调用

watch:{"tableData":function (newVal,oldVal){console.log("nnnnnnnnnnnn",newVal)console.log("oooooooooooo",oldVal)if(this.mergeCols.length > 0) {this.newTableMergeData();}}},
// newTableMergeData方法
newTableMergeData() {for (let i = 0; i < this.tableData.length; i++) {for (let j = 0; j < this.mergeCols.length; j++) {// 初始化行、列坐标信息let rowIndex = 1;let columnIndex = 1;// 比较横坐标左方的第一个元素if (j > 0 && this.tableData[i][this.mergeCols[j]['name']] === this.tableData[i][this.mergeCols[j - 1]['name']]) {columnIndex = 0;}// 比较纵坐标上方的第一个元素if (i > 0 && this.tableData[i][this.mergeCols[j]['name']] === this.tableData[i - 1][this.mergeCols[j]['name']]) {rowIndex = 0;}// 比较横坐标右方元素if (columnIndex > 0) {columnIndex = this.onColIndex(this.tableData[i], j, j + 1, 1, this.mergeCols.length);}// 比较纵坐标下方元素if (rowIndex > 0) {rowIndex = this.onRowIndex(this.tableData, i, i + 1, 1, this.mergeCols[j]['name']);}let key = this.mergeCols[j]['index'] + '_' + i;this.tableMergeIndex[key] = [rowIndex, columnIndex];}}
},
/*** 计算列坐标信息* data 单元格所在行数据* index 当前下标* nextIndex 下一个元素坐标* count 相同内容的数量* maxLength 当前行的列总数*/
onColIndex(data, index, nextIndex, count, maxLength) {// 比较当前单元格中的数据与同一行之后的单元格是否相同if (nextIndex < maxLength && data[this.mergeCols[index]['name']] === data[this.mergeCols[nextIndex]['name']]) {return this.onColIndex(data, index, ++nextIndex, ++count, maxLength);}return count;
},
/*** 计算行坐标信息* data 表格总数据* index 当前下标* nextIndex 下一个元素坐标* count 相同内容的数量* name 数据的key*/
onRowIndex(data, index, nextIndex, count, name) {// 比较当前单元格中的数据与同一列之后的单元格是否相同if (nextIndex < data.length && data[index][name] === data[nextIndex][name]) {return this.onRowIndex(data, index, ++nextIndex, ++count, name);}return count;
}

数据处理好之后就可以调用objectSpanMethods方法了,如下:

objectSpanMethods({ row, column, rowIndex, columnIndex }) {let key = columnIndex + '_' + rowIndex;if (this.tableMergeIndex[key]) {return this.tableMergeIndex[key];}
}

6:效果图

 7:合并行列的完整代

<template><div class="table"><el-table :data="tableData" :span-method="objectSpanMethods" border style="width: 100%"><template v-for="cols in colConfigs"><!-- 无需合并的列 --><el-table-columnv-if="cols.type === 'label' && !cols.children":key="cols.prop":prop="cols.prop":label="cols.label"></el-table-column><!-- 需要合并的列 --><template v-else-if="cols.type === 'label' && cols.children"><el-table-columnv-for="children in cols.children":key="children.prop":prop="children.prop":label="children.label"/></template></template></el-table></div>
</template><script>
export default {name: 'Table',data() {return {tableData: [{ time: '2020-08-10', grade: '三年二班', name: '小明', subjects: '语文', score: 80 },{ time: '2020-08-10', grade: '三年二班', name: '小明', subjects: '数学', score: 80 }, { time: '2020-08-10', grade: '总成绩', name: '总成绩', subjects: '总成绩', score: 160 },{ time: '2020-08-10', grade: '三年一班', name: '小雷', subjects: '语文', score: 70 },{ time: '2020-08-10', grade: '三年一班', name: '小雷', subjects: '数学', score: 80 },{ time: '2020-08-10', grade: '总成绩', name: '总成绩', subjects: '总成绩', score: 150 }, { time: '2020-08-11', grade: '三年三班', name: '小花', subjects: '语文', score: 60 }, { time: '2020-08-11', grade: '三年三班', name: '小花', subjects: '数学', score: 60 }, { time: '2020-08-11', grade: '总成绩', name: '总成绩', subjects: '总成绩', score: 120 }],// 表格的信息 需要合并的需要放在 children 中colConfigs: [{type: 'label',children: [{ prop: 'time', label: '时间' },{ prop: 'grade', label: '年级' },{ prop: 'name', label: '姓名' },{ prop: 'subjects', label: '科目' },{ prop: 'score', label: '成绩' }]},// { type: 'label', prop: 'age', label: '年龄' }],// 需要合并的行列信息 index必须是table表格对应的下标 不能随意修改mergeCols: [{ index: 0, name: 'time' },{ index: 1, name: 'grade' },{ index: 2, name: 'name' },{ index: 3, name: 'subjects' },{ index: 4, name: 'score' },// { index: 5, name: 'age' }],// 用来记录每一个单元格的下标tableMergeIndex: [],};},methods: {objectSpanMethods({ row, column, rowIndex, columnIndex }) {let key = columnIndex + '_' + rowIndex;if (this.tableMergeIndex[key]) {return this.tableMergeIndex[key];}},newTableMergeData() {for (let i = 0; i < this.tableData.length; i++) {for (let j = 0; j < this.mergeCols.length; j++) {// 初始化行、列坐标信息let rowIndex = 1;let columnIndex = 1;// 比较横坐标左方的第一个元素if (j > 0 && this.tableData[i][this.mergeCols[j]['name']] === this.tableData[i][this.mergeCols[j - 1]['name']]) {columnIndex = 0;}// 比较纵坐标上方的第一个元素if (i > 0 && this.tableData[i][this.mergeCols[j]['name']] === this.tableData[i - 1][this.mergeCols[j]['name']]) {rowIndex = 0;}// 比较横坐标右方元素if (columnIndex > 0) {columnIndex = this.onColIndex(this.tableData[i],j,j+1,1,this.mergeCols.length);}// 比较纵坐标下方元素if (rowIndex > 0) {rowIndex = this.onRowIndex(this.tableData,i,i+1,1,this.mergeCols[j]['name']);}let key = this.mergeCols[j]['index'] + '_' + i;this.tableMergeIndex[key] = [rowIndex, columnIndex];}}},/*** 计算列坐标信息* data 单元格所在行数据* index 当前下标* nextIndex 下一个元素坐标* count 相同内容的数量* maxLength 当前行的列总数*/onColIndex(data, index, nextIndex, count, maxLength) {// 比较当前单元格中的数据与同一行之后的单元格是否相同if (nextIndex < maxLength && data[this.mergeCols[index]['name']] === data[this.mergeCols[nextIndex]['name']]) {return this.onColIndex(data, index, ++nextIndex, ++count, maxLength);}return count;},/*** 计算行坐标信息* data 表格总数据* index 当前下标* nextIndex 下一个元素坐标* count 相同内容的数量* name 数据的key*/onRowIndex(data, index, nextIndex, count, name) {// 比较当前单元格中的数据与同一列之后的单元格是否相同if (nextIndex < data.length && data[index][name] === data[nextIndex][name]) {return this.onRowIndex(data,index,++nextIndex,++count,name);}return count;}},mounted() {if(this.mergeCols.length > 0) {this.newTableMergeData();}}
};
</script><style lang="stylus" scoped>
.table height 100vhwidth 100%padding 40pxbox-sizing border-box/deep/ .el-table__body tr:hover > tdbackground-color: #fff;
</style>

三:兼容

如果不想合并的行需要在colConfigs中调整,如下:

前言:需要做的调整

// 增加一个年龄属性 但是不进行合并
colConfigs: [{type: 'label',children: [{ prop: 'time', label: '时间' },{ prop: 'grade', label: '年级' },{ prop: 'name', label: '姓名' },{ prop: 'subjects', label: '科目' },{ prop: 'score', label: '成绩' }]},{ type: 'label', prop: 'age', label: '年龄' }
]

1:效果图

 2: 如果想要合并,需要在mergeCols中添加数据:

mergeCols: [{ index: 0, name: 'time' },{ index: 1, name: 'grade' },{ index: 2, name: 'name' },{ index: 3, name: 'subjects' },{ index: 4, name: 'score' },{ index: 5, name: 'age' } // 添加需要合并的age列信息 注意index的值
],

3:新添加的属性合并后效果图

 

另:单纯的行合并

 data{return {spanArr: [],position: 0,}
}   rowspan(data) {this.spanArr=[];data.forEach((item,index) => {if( index === 0){this.spanArr.push(1);this.position = 0;}else{if(data[index].FId === data[index-1].FId ){this.spanArr[this.position] += 1;this.spanArr.push(0);}else{this.spanArr.push(1);this.position = index;}}})},objectSpanMethod({ row, column, rowIndex, columnIndex }) {if (columnIndex === 0||columnIndex === 1) {const _row = this.spanArr[rowIndex];const _col = _row>0 ? 1 : 0;return {rowspan: _row,colspan: _col}}},

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/81587.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

stm32串口通信(PC--stm32;中断接收方式;附proteus电路图;开发方式:cubeMX)

单片机型号STM32F103R6: 最后实现的效果是&#xff0c;开机后PC内要求输入1或0&#xff0c;输入1则打开灯泡&#xff0c;输入0则关闭灯泡&#xff0c;输入其他内容则显示错误&#xff0c;值得注意的是这个模拟的东西只能输入英文 之所以用2个LED灯是因为LED电阻粗略一算就是1…

【考研数学】矩阵、向量与线性方程组解的关系梳理与讨论

文章目录 引言一、回顾二、梳理齐次线性方程组非齐次线性方程组 写在最后 引言 两个原因让我想写这篇文章&#xff0c;一是做矩阵题目的时候就发现这三货经常绑在一起&#xff0c;让人想去探寻其中奥秘&#xff1b;另一就是今天学了向量组的秩&#xff0c;让我想起来了之前遗留…

Linux后门大全-xinetd后门(二)

环境 靶机&#xff1a;centos7.6 攻击机&#xff1a;Linux 使用docker搭建靶机环境&#xff0c;当然也可以不使用docker&#xff0c;直接跳过创建容器的步骤即可 创建容器 #创建名为backdoorT4的特权容器&#xff0c;并使用/usr/sbin/init&#xff0c;因为容器默认不开启sy…

java:Servlet

背景 我们访问浏览器访问一个地址&#xff0c;最终是访问到了这个 java 类&#xff0c;而 java 是运行在 Tomcat 上的&#xff0c;所以 Tomcat 作为一个服务器会把这个访问地址指向这个类中&#xff0c;这个类就是 Servlet&#xff0c;Servlet 就是一个具有一定规范的类&#x…

牡丹宣言|对国潮化妆品品牌的理解

化妆品的国潮概念&#xff1f; ■ 是中国的时代潮流。 ■ 是传统元素与现代元素的碰撞。 ■ 是一股年轻的力量。 ■ 是大国崛起的象征。 ■ 是中国文化自信的体现。 如何正确认知化妆品&#xff1f; ■ 化妆品不是药品 ■ 化妆品是一种观念 ■ 化妆品是一种习惯 ■ 化…

【科研】-- 如何将Endnote中参考文献格式插入到Word?

文章目录 如何将Endnote中参考文献格式插入到Word&#xff1f; 如何将Endnote中参考文献格式插入到Word&#xff1f; 1、首先确保Endnote和Word安装正确&#xff0c;正常可以从学校官网中下载到正版软件&#xff0c;下载后在word栏目中会出现EndNote的标签&#xff1b; 2、可…

Python使用pyqt5写windows桌面应用实战教程

本篇文章主要讲解,Python使用pyqt5写windows桌面应用的详细实战教程文章,主要涵盖单个界面的布局构建说明,表单构建说明,数据交互构建说明以及可直接开发的简易多界面框架实例构建说明,能够让你很快的了解pyqt在python中的使用和构建方式,快速实现一个简单的windows桌面图…

特殊的矩阵与特殊的矩阵关系———实对称、正定、对角、零矩阵

一、特殊的矩阵 1、实对称矩阵 定义&#xff1a;都是实数&#xff0c;且 性质&#xff1a; &#xff08;1&#xff09;可以用特征值来求A的大小 &#xff08;2&#xff09;可以得到A的秩 &#xff08;3&#xff09;必定可以相似对角化 运用&#xff1a; 与实对称矩阵A合同的矩…

【云原生】Docker私有仓库 RegistryHabor

目录 1.Docker私有仓库&#xff08;Registry&#xff09; 1.1 Registry的介绍 1.2 Registry的部署 步骤一&#xff1a;拉取相关的镜像 步骤二&#xff1a;进行 Registry的相关yml文件配置&#xff08;docker-compose&#xff09; 步骤三&#xff1a;镜像的推送 2. Regist…

centos7安装hadoop 单机版

1.解压 &#xff08;1&#xff09;将hadoop压缩包复制到/opt/software路径下 &#xff08;2&#xff09;解压hadoop到/opt/module目录下 [rootkb135 software]# tar -zxvf hadoop-3.1.3.tar.gz -C /opt/module/ &#xff08;3&#xff09;修改hadoop属主和属组 [rootkb135 m…

如何五分钟设计制作自己的蛋糕店小程序

在现如今的互联网时代&#xff0c;小程序已成为企业推广和销售的重要利器。对于蛋糕店来说&#xff0c;搭建一个小程序可以为其带来更多的品牌曝光和销售渠道。下面&#xff0c;我们将以乔拓云平台为例&#xff0c;来教你如何从零开始搭建自己的蛋糕店小程序。 首先&#xff0c…

【方案】安防监控EasyCVR智慧工地视频监管风险预警平台的应用

智慧工地方案是一种结合现代化技术与工地管理实践的创新型解决方案。它通过实时监控、数据分析、人工智能等技术手段&#xff0c;使工地管理更加高效、智能化。在建设智慧工地的过程中&#xff0c;除了上述提到的利用物联网技术实现设备互联、数据采集及分析以外&#xff0c;还…