el-table树形表格下拉新增功能和子列表的折叠功能

公司新来了一个需求:表格最左侧为新增,然后分类可以展开,子列表超过5条进行折叠收缩

视频操作顺序依次为:展开分类=> 展开更多=> 删除 => 新增 => 编辑=> 设置生效与失效


录制的gif

重要难点:新增和删除时,会对折叠功能造成影响

其他难点:

  1. 新增列必须使用多选框改造, 否则会被el-table认成展开图标列
  2. 每次点击都需要通过id找到当前数据,因为树表格会把坐标打乱
  3. 将子级数量超过5条后的全部折叠起来
  4. 需要复制一个表格数据出来,需要每次新增或删除需要重排下索引值和折叠功能
  5. 点击新增使当前的分类展开后,需要将:expand-row-keys数组置空,否则新增其他分类时,会将上一个分类也给展开

因为代码量不算多,我直接简化拿了过来,并加上注释

    <el-table 
      v-loading="loading"
      element-loading-background="rgba(255, 255, 255, 0.7)"
      :data="tableData" 
      class="tableBox" 
      border 
      style="width: 100%" 
      :row-class-name="tableRowClassName" 
      :row-key="getRowKeys"
      :span-method="arraySpanMethod"
      :expand-row-keys="expands"
      :tree-props="{children: 'children'}"
      @selection-change="handleSelectionChange">
      <!-- <el-table-column align="center" type="selection" class-name="DisabledSelectionAll" label-class-name="DisabledSelection" width="55"></el-table-column> -->
      <el-table-column align="center" type="selection" label-class-name="DisabledSelection" width="50"> 
        <template slot-scope="scope">
          <span @click="addRow(scope.row)" v-if="scope.row.type == 'father'" class="hand">+</span>
        </template>
      </el-table-column>
      <el-table-column align="left"  prop="post" label="适用岗位" width="130"> </el-table-column>
      <el-table-column align="center"  prop="index" label="序号" width="50"> </el-table-column>
      <el-table-column align="center"  prop="address" label="工作内容" width="180">
        <template slot-scope="scope">
          <div v-if="!scope.row.type">
            <span v-if="scope.row.modify">{{scope.row.name}}</span>
            <el-input v-else v-model="scope.row.name" type="text" />
          </div>
        </template>  
      </el-table-column>
      <el-table-column align="center" label="创建日期"> 
        <template slot-scope="scope" >
          <span v-if="!scope.row.type">{{scope.row.date}}</span>
          <el-button v-if="scope.row.type === 'shrink'" type="text" @click="handleShrink(scope.row)">{{scope.row.date}}</el-button>
        </template>
      </el-table-column>
      <el-table-column align="center"  prop="requency" label="频率">
        <template slot-scope="scope" v-if="!scope.row.type">
          <span v-if="scope.row.modify">{{optionsFrequency[Number(scope.row.requency)-1].label}}</span>
          <el-select v-else v-model="scope.row.requency" placeholder="请选择">
            <el-option
              v-for="item in optionsFrequency"
              :key="item.value"
              :label="item.label"
              :value="item.value">
            </el-option>
          </el-select>
        </template>
      </el-table-column>
      <el-table-column align="center" label="操作">
        <template slot-scope="scope" v-if="!scope.row.type">
          <el-button v-if="scope.row.modify" type="text" @click="handleEdit(scope.row, true)">修改</el-button>
          <el-button v-else type="text" @click="handleSave(scope.row)">保存</el-button>
          <el-button v-if="scope.row.takeEffect"@click="handleTakeEffect(scope.row)" type="text">生效</el-button>
          <el-button v-else type="text"@click="handleTakeEffect(scope.row)">失效</el-button>
          <el-button @click="handleDelete(scope.row)" type="text">删除</el-button>
        </template>
      </el-table-column>
    </el-table>
js

moment插件和lodash可以不用,自己用其他方法

<script>
import moment from 'moment';
import cloneDeep from 'lodash/cloneDeep';

export default {
  data() {
    return {
      loading: false,
      num: 10000,
      tableData: [],
      tableDataCopy: [],
      optionsFrequency: [
        {
          value: '1',
          label: '工作日'
        }, 
        {
          value: '2',
          label: '每周一'
        }, 
        {
          value: '3',
          label: '每周二'
        }, 
        {
          value: '4',
          label: '每周三'
        }, 
        {
          value: '5',
          label: '每周四'
        }, 
        {
          value: '6',
          label: '每周五'
        }, 
        {
          value: '7',
          label: '每月初'
        }, 
        {
          value: '8',
          label: '每周五'
        }, 
        {
          value: '9',
          label: '每月末'
        }, 
        {
          value: '10',
          label: '不定期'
        }, 
      ],
      expands: [],
    };
  },
  created(){
    this.start()
  },
  mounted() {
    // 默认展开第0个
    this.expands = [this.tableData[0].id.toString()]
    this.$nextTick(() => {
      this.expands = []
    })
  },
  methods: {
    start(){
      this.tableData = [
        {
          id: 1,
          post: 'OPM运营岗',
          type:'father',
          takeEffect: false,
          children: [
            {
              id: 2,
              date: '2016-05-01',
              name: '王小虎',
              requency: '2',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '3',
            }, {
              id: 3,
              date: '2016-05-01',
              name: '王小虎',
              requency: '10',
              takeEffect: false,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '1',
            },
            {
              id: 12,
              date: '2016-05-01',
              name: '王小虎',
              requency: '2',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '3',
            }, {
              id: 13,
              date: '2016-05-01',
              name: '王小虎',
              requency: '10',
              takeEffect: false,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '1',
            },
            {
              id: 14,
              date: '2016-05-01',
              name: '王小虎',
              requency: '2',
              takeEffect: false,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '3',
            }, {
              id: 15,
              date: '2016-05-01',
              name: '王小虎',
              requency: '10',
              takeEffect: false,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '1',
            }
          ]
        }, 
        {
          id: 4,
          post: '参数录入岗',
          type:'father',
          children: [
            {
              id: 5,
              date: '2016-05-01',
              name: '王小虎',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '3',
            }, {
              id: 6,
              date: '2016-05-01',
              name: '王小虎',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '1',
            }
          ]
        }, 
        {
          id: 7,
          post: '基金会计港',
          type:'father',
          children: [
            {
              id: 31,
              date: '2016-05-01',
              name: '王小虎',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '2',
            }, {
              id: 32,
              date: '2016-05-01',
              name: '王小虎',
              requency: '4',
              takeEffect: false,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '1',
            },
            {
              id: 33,
              date: '2016-05-01',
              name: '王小虎',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '2',
            }, {
              id: 34,
              date: '2016-05-01',
              name: '王小虎',
              requency: '4',
              takeEffect: false,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '1',
            },
            {
              id: 35,
              date: '2016-05-01',
              name: '王小虎',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '2',
            }, {
              id: 36,
              date: '2016-05-01',
              name: '王小虎',
              requency: '4',
              takeEffect: false,
              address: '上海市普陀区金沙江路 1519 弄',
              state: '1',
            },
          ]
        }, 
        {
          id: 10,
          name: '王小虎',
          state: '3',
          post: '头寸管理岗',
          requency: '6',
          type:'father',
          takeEffect: true,
          address: '上海市普陀区金沙江路 1516 弄',
          children:[
            {
              id: 41,
              date: '2016-05-01',
              name: '王小虎1',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄1',
              state: '3',
            }, {
              id: 42,
              date: '2016-05-01',
              name: '王小虎2',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄2',
              state: '1',
            },
            {
              id: 43,
              date: '2016-05-01',
              name: '王小虎3',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄3',
              state: '3',
            }, {
              id: 44,
              date: '2016-05-01',
              name: '王小虎4',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄4',
              state: '1',
            },
            {
              id: 45,
              date: '2016-05-01',
              name: '王小虎5',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄5',
              state: '3',
            }, {
              id: 46,
              date: '2016-05-01',
              name: '王小虎6',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄6',
              state: '1',
            },
            {
              id: 47,
              date: '2016-05-01',
              name: '王小虎7',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄7',
              state: '3',
            }, {
              id: 48,
              date: '2016-05-01',
              name: '王小虎8',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄8',
              state: '1',
            },
            {
              id: 49,
              date: '2016-05-01',
              name: '王小虎9',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄9',
              state: '3',
            }, {
              id: 410,
              date: '2016-05-01',
              name: '王小虎10',
              requency: '3',
              takeEffect: true,
              address: '上海市普陀区金沙江路 1519 弄10',
              state: '1',
            }
          ]
        },       
      ]
      this.tableDataCopy = cloneDeep(this.tableData)
      this.serialnumber()
      this.tableShrink()
    },
    // 设置序号
    serialnumber(){
      this.tableData.forEach((item,index) => {
      if(item.children && item.children.length){
        item.children.forEach((item2, index2) => {
         // 设置序号
          if(item.type !== 'shrink'){ // 收缩键
            this.$set(item2, 'index', index2 + 1)
          }else{
            this.$set(item2, 'index', '')
          }
         // 当前为保存状态(无可编辑需求可以忽略)
          this.$set(item2, 'modify', true)
        })
      }
      })
    },

    // 展开/收缩
    tableShrink(){
      this.tableData.forEach(item => {
        if(item.children.length > 5){
          const newChildren = item.children.splice(5, item.children.length)
          this.$set(item, 'newChildren', newChildren)
          this.num ++
          item.children.push({
            id: this.num,
            type:'shrink',
            date:'展开》'
          })
        }
      })
    },

    // 获取row的key值
    getRowKeys(row) {
      return row.id;
    },

    handleShrink(row, indexArr=null){
      const arr = indexArr? indexArr : this.ObtainIndex(row.id)
      if(row.date == '展开》'){
        // 取出被折叠的数据
        const oldChildren = this.tableData[arr[0]].newChildren.splice(0, this.tableData[arr[0]].newChildren.length)
        // 取出最后的折叠按钮
        const Shrink = this.tableData[arr[0]].children.splice(this.tableData[arr[0]].children.length-1, 1)
        this.$nextTick(() => {
          this.tableData[arr[0]].children.push(...oldChildren)
          Shrink[0].date = '收起》'
          this.tableData[arr[0]].children.push(...Shrink)
        })
      }else{
        // 需要折叠的数据
        const newChildren = this.tableData[arr[0]].children.splice(5, this.tableData[arr[0]].children.length -6)
        this.tableData[arr[0]].newChildren = newChildren
        // 最后的折叠按钮
        this.tableData[arr[0]].children[this.tableData[arr[0]].children.length-1].date = '展开》'
      }

    },

    // 合并
    arraySpanMethod({ row, column, rowIndex, columnIndex }){
      // 合并展开/收缩
      // if(row.type === 'shrink'){
      //   if (columnIndex === 4) {
      //     return { 
      //       rowspan: 1,
      //       colspan: 6
      //     }
      //   }
      // }

      // 合并岗位
      if(row.type === 'father'){
        if (columnIndex === 1) {
          return { 
            rowspan: 1,
            colspan: 6
          }
        }
      }
    },

    // 新增
    addRow(row){
      const index = this.tableData.findIndex(item => {
        return item.id === row.id
      })
      console.log(row.id)
      this.tableDataCopy[index].children.unshift({
        id: Date.now(),
        index: 1,
        post:'',
        name:'',
        date:moment(new Date()).format("YYYY-MM-DD"),
        requency:'1',
        modify: false,
        takeEffect: false,
      })
      this.tableData = cloneDeep(this.tableDataCopy)

      this.serialnumber()
      this.tableShrink([index])

      this.expands = [this.tableData[index].id.toString()]
      this.$nextTick(() => {
        this.expands = []
      })
    },

    // 获取坐标
    ObtainIndex(id){
      const indexArr = []
      for(let i = 0;i < this.tableData.length;i ++){
        if(!(this.tableData[i].children && this.tableData[i].children.length)) continue
        const itemIndex = this.tableData[i].children.findIndex(item => {
          return item.id == id
        })
        if(itemIndex > -1){
          indexArr.push(i)
          indexArr.push(itemIndex)
          break
        }
      }
      return indexArr
    },

    // 编辑
    handleEdit(row){
      const arr = this.ObtainIndex(row.id)
      this.tableData[arr[0]].children[arr[1]].modify = !this.tableData[arr[0]].children[arr[1]].modify
    },

    // 保存
    handleSave(row){
      this.loading = true
      const that = this
      // ajax
      setTimeout(() => {
        this.handleEdit(row)
        that.loading = false
      },300)
    },

    // 生效/失效
    handleTakeEffect(row){
      const arr = this.ObtainIndex(row.id)
      this.loading = true
      
      // ajax
      setTimeout(() => {
        this.loading = false
        this.tableData[arr[0]].children[arr[1]].takeEffect = !this.tableData[arr[0]].children[arr[1]].takeEffect
      },300)
    },

    // 删除
    handleDelete(row){
      this.loading = true

      setTimeout(() => {
      this.loading = false
      const arr = this.ObtainIndex(row.id)
      this.tableData = []
      this.tableDataCopy[arr[0]].children.splice(arr[1], 1)
      this.tableData = cloneDeep(this.tableDataCopy)
      this.serialnumber()
      this.tableShrink(arr)
      }, 300)
    },
    
    // 设置失效行的class
    tableRowClassName({row, rowIndex}) {
      if (!row.type && row.takeEffect === false) {
        return 'warning-row';
      }
      return '';
    },
  }
};
</script>
css
<style lang='scss' scoped>
.hand{
  cursor: pointer;
}

.el-table /deep/ .DisabledSelection .cell .el-checkbox__inner {
  display: none;
  position: relative;
}
/*表格全选框改为:新增*/
.el-table /deep/ .DisabledSelection .cell:before {
  content: "新增";
  position: absolute;
  left: 14px;
}

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

推荐阅读更多精彩内容