el-table:列表中相同名称的数据实现行合并

1. 效果图

效果图

2. 页面

使用el-table的span-method属性控制合并行, 和row-class-name属性控制行样式

<el-table
  :data="tableData"
  :height="500"
  :span-method="objectSpanMethod"
  :row-class-name="getRowClass"
>
...省略...
</el-table>

3. 方法

handleData: 处理表格数据,将同一名称的数据进行组合
getSpanArr:获取单元格的合并行数
objectSpanMethod:合并单元格
getRowClass:设置表格行的样式类

/**
 * 处理表格数据,将同一名称的数据进行组合
 */
const handleData = () => {
  // 排序方法1:将相同名称的数据移动在一起,但会导致所有数据的顺序都被改变
  // state.tableData.sort((a, b) => {
  //   if (a.name != b.name) {
  //     return a.name.localeCompare(b.name) // stringObject.localeCompare(target): 用本地特定的顺序来比较两个字符串
  //   }
  // })

  // 排序方法2:只移动有相同名称的数据,保持其它数据的相对顺序不改变
  let keys = [] // 唯一值的数组
  state.tableData.forEach((item, index) => {
    if (!keys.includes(item.moduleName)) {
      keys.push(item.moduleName)
    }
  })

  let temp = []
  keys.forEach((item) => {  // 将同一名称的数据放在相邻位置
    state.tableData.forEach((cell) => {
      if (item === cell.name) {
        temp.push(cell)
      }
    })
  })

  state.tableData = temp
  getSpanArr(state.tableData)
}

let spanArr = [] // 每一条数据合并的行数,避免表格错乱!
/**
 *  获取单元格的合并行数
 * */
const getSpanArr = (data) => {
  let position  // 当前合并的行位置(连续相同名称的第一条数据位置)
  spanArr = []
  data.forEach((item, index) => {
    if (index === 0) {  // 第一行, 不进行处理
      spanArr.push(1)
      position = 0
    } else {
      if (data[index].name === data[index - 1].name) {
        // 当条数据跟上一条数据名称相同,要合并
        spanArr[position] += 1  // 首条相同名称行合并行数增加
        spanArr.push(0) // 当前行消除
      } else { // 不需要处理的数据
        spanArr.push(1)
        position = index
      }
    }
  })
}

/** 
 * 合并单元格,此处只合并前三列的属性值 
 * 
*/
const objectSpanMethod = ({ row, column, rowIndex, columnIndex }) => {
  if (columnIndex < 3) {  // 对前三列进行操作
    const _row = spanArr[rowIndex]  // 合并行数
    const _col = _row > 0 ? 1 : 0 // 合并列数,1:不改变,0:被消除
    return {
      rowspan: _row,
      colspan: _col
    }
  }
}

/**
 * 设置表格行的样式类(去除合并行内多余的线条)
 */
const getRowClass = ({ row, rowIndex }) => {
  if (spanArr[rowIndex] > 1) { // 相同名称排列的首行数据
    return 'show-span-row'
  } else if (spanArr[rowIndex] === 0 && spanArr[rowIndex + 1] === 0) {  // 相同名称不处于末尾的被合并数据
    return 'center-span'
  }
}

4. 样式

getRowClass设置的css样式

.el-table {
  :deep(tr) {
    &.show-span-row {
      td:nth-of-type(n + 3) {
        border-bottom: none;
      }
    }

    &.center-span {
      td {
        border-bottom: none;
      }
    }
  }
}

5. 整体代码(Vue3)

<template>
  <el-table
    :data="tableData"
    :height="500"
    :span-method="objectSpanMethod"
    :row-class-name="getRowClass"
  >
  ...省略...
  </el-table>
</template>

<script>
import { reactive, ref, toRefs } from '@vue/reactivity'
import { onMounted, watch } from '@vue/runtime-core'
export default {
  setup(props, context) {

    const state = reactive({
      tableData: []
    })

    onMounted(() => {
      handleData()
    })

    /**
     * 处理表格数据,将同一名称的数据进行组合
     */
    const handleData = () => {
      // 排序方法1:将相同名称的数据移动在一起,但会导致所有数据的顺序都被改变
      // state.tableData.sort((a, b) => {
      //   if (a.name != b.name) {
      //     return a.name.localeCompare(b.name) // stringObject.localeCompare(target): 用本地特定的顺序来比较两个字符串
      //   }
      // })

      // 排序方法2:只移动有相同名称的数据,保持其它数据的相对顺序不改变
      let keys = [] // 唯一值的数组
      state.tableData.forEach((item, index) => {
        if (!keys.includes(item.moduleName)) {
          keys.push(item.moduleName)
        }
      })

      let temp = []
      keys.forEach((item) => {  // 将同一名称的数据放在相邻位置
        state.tableData.forEach((cell) => {
          if (item === cell.name) {
            temp.push(cell)
          }
        })
      })

      state.tableData = temp
      getSpanArr(state.tableData)
    }

    let spanArr = [] // 每一条数据合并的行数,避免表格错乱!
    /**
     *  获取单元格的合并行数
     * */
    const getSpanArr = (data) => {
      let position  // 当前合并的行位置(连续相同名称的第一条数据位置)
      spanArr = []
      data.forEach((item, index) => {
        if (index === 0) {  // 第一行, 不进行处理
          spanArr.push(1)
          position = 0
        } else {
          if (data[index].name === data[index - 1].name) {
            // 当条数据跟上一条数据名称相同,要合并
            spanArr[position] += 1  // 首条相同名称行合并行数增加
            spanArr.push(0) // 当前行消除
          } else { // 不需要处理的数据
            spanArr.push(1)
            position = index
          }
        }
      })
    }

    /** 
     * 合并单元格,此处只合并前三列的属性值 
     * 
    */
    const objectSpanMethod = ({ row, column, rowIndex, columnIndex }) => {
      if (columnIndex < 3) {  // 对前三列进行操作
        const _row = spanArr[rowIndex]  // 合并行数
        const _col = _row > 0 ? 1 : 0 // 合并列数,1:不改变,0:被消除
        return {
          rowspan: _row,
          colspan: _col
        }
      }
    }

    /**
     * 设置表格行的样式类(去除合并行内多余的线条)
     */
    const getRowClass = ({ row, rowIndex }) => {
      if (spanArr[rowIndex] > 1) { // 相同名称排列的首行数据
        return 'show-span-row'
      } else if (spanArr[rowIndex] === 0 && spanArr[rowIndex + 1] === 0) {  // 相同名称不处于末尾的被合并数据
        return 'center-span'
      }
    }

    return {
      ...toRefs(state),
      objectSpanMethod
    }
  }
}
</script>

<style lang="scss" scoped>
.el-table {
  :deep(tr) {
    &.show-span-row {
      td:nth-of-type(n + 3) {
        border-bottom: none;
      }
    }

    &.center-span {
      td {
        border-bottom: none;
      }
    }
  }
}
</style>

5. 其它

objectSpanMethod原理:对每一个单元格返回一个[rowSpan, colSpan]数组, rowSpan表示当前单元格会展示的行数,colSpan表示当前单元格会展示的列数,设置为0时当前单元格被消除。

参考文章:
element-ui table :span-method(行合并)
js将数组中的相同项放在毗邻位置

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

推荐阅读更多精彩内容