Bootstrap

基于element UI 显示编辑表格

基于element ui 实现编辑表格,支持文本、下拉、日期、链接等形式,实现了回车到下一个输入框,每一个回车都可以触发业务回车事件,例如计算小计;最后一格回车可以触发自动增行等效果。

实现效果如下

组件入口 index.vue

<template>
  <my-form :model="tableDataTmp" ref="tableForm" class="edit-table-dialog" @submit.native.prevent :class="{'edit-table-disabled': disabled}">
    <el-row justify="end" type="flex">
      <div class="title-tag" v-if="title">{{ title }}</div>
      <slot name='btn-more-before'></slot>
      <el-button type="primary" plain size="small" icon="el-icon-plus"     @click="addRow" v-if="!disabled && (options.showAddBtn==undefined || options.showAddBtn==null || options.showAddBtn)">新增行</el-button>
      <el-button type="primary" plain size="small" icon="el-icon-delete"     @click="deleteRow" v-if="!disabled && (options.showDelBtn==undefined || options.showDelBtn==null || options.showDelBtn)">删除行</el-button>
      <slot name="btn-more"></slot>
    </el-row>
    <el-table ref="editTableRef" border :data="tableDataTmp.tableData" row-key="__rowKey" @selection-change="selectionChange" @select="handleSelect" class="edit-table" :row-class-name="tableRowClassName" :height="height" empty-text=" ">
      <template v-for="item in columnData">
        <column ref="columnRef" :key="item.label" :columnInfo="item" :options="options" :tableRules="tableRules" :rowKey="rowKey" :defaultShowEdit="defaultShowEdit"
          :selectList="selectList" :isDiff="isDiff" :disabled="disabled" :editIndex="editIndex" @editFocus="editFocus" @keyupEnter="keyupEnter" @onfocus="onfocus" @link="clickLink"></column>
      </template>
    </el-table>
    <table-dialog ref="tableDialogRef" :open.sync="open" :tabName="tabName" :tabParams="tabParams" :hideColumns="hideColumns" @oncheck="oncheck"></table-dialog>
  </my-form>
</template>

<script>
import MyForm from './form/form.vue'
import Column from './column.vue'
import TableDialog from '@/components/TableDialog/index.vue'
export default {
  name: "EditTable",
  components: { TableDialog, MyForm, Column },
  props: {
    height: {
      type: [String, Number]
    },
    columnData: {
      type: Array,
      default: () => {
        return []
      }
    },
    emptyRow: {
      type: Object,
      default: () => {
        return {}
      }
    },
    tableData: {
      type: Array,
      default: () => {
        return []
      }
    },
    tableRules: {
      type: Object,
      default: () => {
        return {}
      }
    },
    options: {
      type: Object,
      default: () => {
        return {
          showAddBtn: true,
          showDelBtn: true
        }
      }
    },
    rowKey: {
      type: [String, Function]
    },
    defaultShowEdit: {
      type: Boolean,
      default: false
    },
    // 是否显示比对
    isDiff: {
      type: Boolean,
      default: false
    },
    disabled: {
      type: Boolean,
      default: false
    },
    title: {
      type: String,
      default: ''
    },
  },
  data() {
    return {
      tableDataTmp: {
        tableData: []
      },
      // 末级列数据
      columnDataLast: [],
      tabName: null,
      tabParams: {},
      hideColumns: null,
      editIndex: '', // 当前编辑单元格
      open: false,
      chooseData: {},
      selections: [],
      rowids: [],
      selectList: [],
      isEnter: false // 是否回车
    }
  },
  created() {},
  watch: {
    'tableData': {
      handler() {
        let rowKeyTmp = new Date().getTime()
        this.tableData.forEach((tb, index) => {
          tb.rowEdit = (tb.rowEdit === undefined || tb.rowEdit === null) ? true : tb.rowEdit
          if (!tb.__rowKey) {
            tb.__rowKey = tb[this.rowKey] || (rowKeyTmp+'_'+index)
          }
        });
        this.tableDataTmp = { tableData: this.tableData }
      },
      immediate: true
    },
    'columnData': {
      handler() {
        let columnDataLast = []
        let func = (list, level) => {
          list.forEach((item, index)=>{
            if (item.children && item.children.length>0) {
              func(item.children, level+'_'+index)
            } else {
              columnDataLast.push({ ...item, level__: level+'_'+index })
              // 如果没有设置renderHeader,且是必填字段,添加必填星号
              if (this.tableRules[item.prop] && !item.renderHeader) {
                let valIndex = this.tableRules[item.prop].findIndex((validate)=>{
                  return validate.required
                })
                if (valIndex > -1) {
                  item.renderHeader = (h, data)=>{
                    return [
                      h('span', { style: 'color: #F56C6C;margin-right: 5px;' }, '*'),
                      h('span', item.label)
                    ]
                  }
                }
              }
              // 如果是dataSelect类型,预处理数据
              if (item.editType === 'dataSelect') {
                // let _options = { ...item.selectData, headers: { hideMessage: true } }
                // request(_options).then((rsp)=>{
                //   this.$set(this.cacheSelectData, item.prop, rsp.data)
                // })
              }
            }
          })
        }
        func(this.columnData, 'a')
        this.columnDataLast = columnDataLast
      },
      deep: true,
      immediate: true
    }
  },
  methods: {
    // 表格行样式
    tableRowClassName({row, rowIndex}) {
      if (!this.isDiff) return ''
      if (row.DELROW) {
        return 'warning-row';
      } else if (row.ADDROW) {
        return 'success-row';
      }
      return '';
    },
    addRow() {
      let index = this.tableDataTmp.tableData.length
      let row = { __rowKey: new Date().getTime() }
      if(this.emptyRow) row = { ...row, ...this.emptyRow }
      this.tableDataTmp.tableData.splice(index, 0, row)
      // this.$emit('update:tableData', this.tableDataTmp.tableData)
    },
    deleteRow() {
      if (!this.rowids || this.rowids.length==0) {
        this.$modal.msgError("请选择要删除的行");
        return
      }

      this.rowids.forEach((id)=>{
        let index = this.tableDataTmp.tableData.findIndex(item=>(item.__rowKey==id))
        this.tableDataTmp.tableData.splice(index, 1)
      })
      // this.$emit('update:tableData', this.tableDataTmp.tableData)
    },
    selectionChange (selections) {
      this.selections = selections
      this.rowids = this.selections.map(item=>item.__rowKey)
      this.$emit('selection-change', selections)
    },
    handleSelect(selection, row) {
      this.$emit('select', selection, row)
    },
    getSelection () {
      return this.selections
    },
    selectSame (sameList, checked) {
      this.$nextTick(()=>{
        sameList.forEach(item=>{
          this.$refs.editTableRef.toggleRowSelection(item, checked)
        })
      })
    },
    onfocus(index_) {
      setTimeout(async () => {
        if (this.disabled) return false
        this.editIndex = index_
        const { rowIndex, colIndex, row, columnConfig, columnValue } = this.getCommonData()
        if (columnConfig.editType === 'select') {
          row[columnConfig.valprop || columnConfig.prop] = null
          this.selectList = await columnConfig.selectData.getData(row)
          row[columnConfig.valprop || columnConfig.prop] = columnValue
        }
        this.$nextTick(() => {
          const editEl = this.getElementRef(rowIndex, colIndex)
          if (columnConfig.editType === 'select') {
            editEl.toggleMenu()
          } else {
            editEl && editEl.focus()
          }
        })
      }, 200) // 日期回车下一个是下拉时,触发了下拉的回车事件,触发了两次回车
    },
    getElementRef (rowIndex, colIndex) {
      let columnConfig = this.columnDataLast[colIndex]
      let levels = columnConfig.level__.split('_')
      let editEl = this
      for(let n = 1;n<levels.length;n++) {
        editEl = editEl.$refs['columnRef'][parseInt(levels[n])]
      }
      editEl = editEl.$refs['flaga-' + rowIndex + '-' + columnConfig.prop].$children[1]
      // if (columnConfig.editType === 'dataSelect') {
      //   editEl = editEl.$children[0]
      // }
      return editEl
    },
    getCommonData () {
      let indexs = this.editIndex.split('-')
      let rowIndex = parseInt(indexs[0])
      let colProp = indexs[1]
      let colIndex = this.columnDataLast.findIndex((item)=>item.prop===colProp)
      let columnConfig = this.columnDataLast[colIndex]
      let row = this.tableDataTmp.tableData[rowIndex]
      let columnValue = columnConfig.editType === 'select' ? row[columnConfig.valprop || columnConfig.prop] : row[columnConfig.prop]
      return {rowIndex, colProp, colIndex, columnConfig, row, columnValue}
    },
    // 进入下一个可编辑单元格
    nextStep() {
      if (!this.editIndex) {
        return false
      }
      let { rowIndex, colIndex, columnConfig } = this.getCommonData()
      let nextColIndex = -1
      // 查找当前行下一个可编辑项
      if (colIndex < this.columnDataLast.length - 1) {
        for (let ind = colIndex + 1; ind < this.columnDataLast.length; ind++) {
          if (this.columnDataLast[ind].edit && this.columnDataLast[ind].editType !== 'link') {
            nextColIndex = ind
            columnConfig = this.columnDataLast[ind]
            break
          }
        }
      }
      let editEl = this.getElementRef(rowIndex, colIndex)
      // 找到下一个编辑列
      if (nextColIndex !== -1) {
        editEl.blur()
        this.onfocus(rowIndex + '-' + columnConfig.prop)
      }
      // 未找到判断是否有下一行,有下一行进入下一行第一个
      else if (rowIndex < this.tableDataTmp.tableData.length - 1) {
        let nextRowIndex = this.tableDataTmp.tableData.findIndex((row, index)=>{
          return index > rowIndex && row.rowEdit
        })

        if (nextRowIndex > -1) {
          for (let ind = 0; ind < this.columnDataLast.length; ind++) {
            if (this.columnDataLast[ind].edit && this.columnDataLast[ind].editType !== 'link') {
              nextColIndex = ind
              columnConfig = this.columnDataLast[ind]
              break
            }
          }
          // const editEl = this.getElementRef(rowIndex, colIndex)
          editEl.blur()
          this.onfocus(nextRowIndex + '-' + columnConfig.prop)
        }
      }
    },
    // 聚焦
    editFocus(index_) {
      this.editIndex = index_
      // let {rowIndex, colIndex, columnConfig, row, columnValue} = this.getCommonData()
      // let editEl = this.getElementRef(rowIndex, colIndex)
      // if (columnConfig.editType === 'select') {
      //   editEl.toggleMenu()
      // } else {
      //   editEl.focus()
      // }
    },
    next () {
      let {rowIndex, colIndex, columnConfig, row, columnValue} = this.getCommonData()
      if (columnConfig.editType === 'date') {
        this.nextStep()
        return
      } else if (columnConfig.editType === 'dialog') {
        this.nextStep()
      } else {
        this.nextStep()
      }
    },
    // 回车事件
    keyupEnter(isNext=true, e) {
      // console.log('====', isNext, this.isEnter, e)
      if (isNext) {
        this.isEnter = true
      } else if (!isNext) {
        let curField = this.getCommonData()
        // 展示下拉项时,不要执行操作 | dialog失焦时不执行操作
        if (curField.columnConfig.editType === 'select' && e == true) {
          return false
        } else if (curField.columnConfig.editType === 'dialog') {
          return false
        }
        // 如果当前已标记为回车,不执行接下来的操作
        if (this.isEnter) {
          this.isEnter = false
          return
        }
        // 下面这段逻辑暂时好像不需要
        //  else {
        //   let curField = this.getCommonData()
        //   if (curField.columnConfig.editType === 'select' && ) {
        //     // e===false && (this.editIndex=null)
        //   } else {
        //     this.editIndex = null
        //   }
        // }
      }
      // console.log('====keyupEnter', this.editIndex)
      let {rowIndex, colIndex, columnConfig, row, columnValue} = this.getCommonData()
      // 如果是弹框,直接弹框
      if (columnConfig.editType === 'dialog') {
        // 单元格内容
        this.tabName = columnConfig.dialogData.tabName
        let tabParams = { [columnConfig.prop]: columnValue }
        // 解析传入参数
        let tabParamsIn = {}
        if (columnConfig.dialogData.tabParams && typeof columnConfig.dialogData.tabParams == 'function') {
          tabParamsIn = columnConfig.dialogData.tabParams(row)
        } else if (columnConfig.dialogData.tabParams) {
          tabParamsIn = columnConfig.dialogData.tabParams
        }
        tabParams = {...tabParamsIn, ...tabParams}
        this.tabParams = tabParams
        this.hideColumns = columnConfig.dialogData.hideColumns
        this.open = true
        this.$nextTick(()=>{
          this.$refs.tableDialogRef.focusDialog()
        })
        return
      } else if (columnConfig.editType === 'select') {
        let current = this.selectList.find(item=>item[columnConfig.selectData.value] == columnValue)
        if (!current) current = {}
        this.chooseData = current
        let updateFields = (columnConfig.selectData ? columnConfig.selectData.updateFields : null)
        if (!updateFields && columnConfig.selectData) {
          let defaultUpdateFields = {}
          if (columnConfig.valprop) {
            defaultUpdateFields = { [columnConfig.valprop]: columnConfig.selectData.value, [columnConfig.prop]: columnConfig.selectData.label }
          } else {
            defaultUpdateFields = { [columnConfig.prop]: columnConfig.selectData.value }
          }
          updateFields = defaultUpdateFields
        }
        this.updateChooseData(rowIndex, updateFields, true)
      }
      this.validateNext(isNext)
    },
    // 验证与下一步
    validateNext (isNext = true) {
      let {rowIndex, colIndex, columnConfig, row, columnValue} = this.getCommonData()
      if (columnConfig.editType !== 'dialog') {
        // this.$emit('update:tableData', this.tableDataTmp.tableData)
      }
      // 弹框传递额外参数
      let otherParams = (columnConfig.editType === 'dialog') ? this.updateChooseData(rowIndex, (columnConfig.dialogData ? columnConfig.dialogData.updateFields : null)) : null
      // let otherParams = JSON.parse(JSON.stringify(this.tableDataTmp.tableData[rowIndex]))
      // 验证单元格数据,验证不通过不继续执行
      this.$refs.tableForm.validateField('tableData.'+rowIndex+'.'+columnConfig.prop, (result) => {
        if (result) {
          if (columnConfig.editType === 'dialog') this.$modal.msgError(result)
          return
        }
        if (columnConfig.editType === 'dialog') {
          this.updateChooseData(rowIndex, (columnConfig.dialogData ? columnConfig.dialogData.updateFields : null), true)
          // this.$emit('update:tableData', this.tableDataTmp.tableData)
          this.open = false
        }

        // 判断是否跳转到下一步
        let next = isNext ? this.next : ()=>{}
        // 判断是否有业务回调 判断是否跳转到下一步
        if (columnConfig.callback) {
          columnConfig.callback({index: rowIndex, row: row, isEnter: isNext}, next )
          return
        }
        next()
      }, otherParams)
    },
    // 弹框选择带回
    oncheck (chooseData) {
      if (!chooseData) return;
      this.chooseData = chooseData
      // let {rowIndex, columnConfig} = this.getCommonData()
      // this.updateChooseData(rowIndex, (columnConfig.dialogData ? columnConfig.dialogData.updateFields : null), true)
      // this.open = false
      this.validateNext()
    },
    /**
     * 更新弹框数据。isTemp传true-更新数据,传false-返回临时数据
     */
    updateChooseData (rowIndex, updateFields, isTemp = false) {
      let tmpData = JSON.parse(JSON.stringify(this.tableDataTmp.tableData[rowIndex]))
      if (updateFields) {
        if (updateFields instanceof Array) {
          updateFields.forEach((key) => {
            tmpData[key] = this.chooseData[key]
            isTemp && this.$set(this.tableDataTmp.tableData[rowIndex], key, this.chooseData[key])
          })
        } else {
          var objkeys = Object.keys(updateFields)
          objkeys.forEach((key) => {
            tmpData[key] = this.chooseData[updateFields[key]]
            isTemp && this.$set(this.tableDataTmp.tableData[rowIndex], key, this.chooseData[updateFields[key]])
          })
        }
      }
      return tmpData
    },
    validate (callback) {
      return new Promise((resolve, reject) => {
        this.$refs.tableForm.validate(callback).then((result)=>{
          resolve(result)
        }).catch(e=>{
          let errorArr = Object.values(e)
          this.$modal.msgError(errorArr[0][0].message);
          reject(e)
        })
      })
    },
    validateRow (index, callback) {
      return this.$refs.tableForm.validateRow(index, callback)
    },
    resetTableform() {
      this.resetForm('tableForm')
    },
    clickLink(prop, scope) {
      this.$emit('link', prop, scope)
    }
  },
};
</script>
<style lang="scss" scoped>
  // .edit-table-dialog ::v-deep .table-form-item.is-error:hover .el-form-item__error{
  //   display: block!important;
  // }
  .edit-table-dialog{
    // 标题按钮行
    .title-tag {
      margin: 4px 0;
      flex-grow: 1;
    }
    .edit-table {
      margin-top: 10px;
      ::v-deep .el-table__body-wrapper {
        tr:hover > td.el-table__cell{
          background-color: #fff;
        }
        // overflow: unset; // 超出的字段滚动
        .el-table__cell {
          padding: 0!important;
          &.el-table-column--selection .el-checkbox {
            margin-left: 16px;
          }
          .cell {
            // overflow: unset;
            padding: 0!important;
            .column-text {
              width: 100%;
              height: 48px;
              line-height: 48px;
              overflow: hidden;
              text-overflow: ellipsis;
              white-space: nowrap;
              display: flex;
              justify-content: center;
              align-items: center;
              .edit-border {
                cursor: pointer;
              }
              .date {
                position: relative;
                width: calc(100% - 12px);
                height: 36px;
                line-height: 36px;
                border: 1px solid #F0F2F7;
                border-radius: 4px;
                text-align: left;
                &::before {
                  content: "";
                  font-family: "element-icons";
                  color: #C0C4CC;
                  width: 30px;
                  position: relative;
                  top: -1px;
                  line-height: 34px;
                  text-align: center;
                  display: inline-block;
                }
              }
              .select {
                position: relative;
                width: calc(100% - 12px);
                height: 36px;
                line-height: 36px;
                border: 1px solid #F0F2F7;
                border-radius: 4px;
                text-align: left;
                padding: 0 30px 0 10px;
                &::after {
                  content: "";
                  color: #C0C4CC;
                  font-size: 14px;
                  font-family: "element-icons";
                  width: 25px;
                  text-align: center;
                  display: inline-block;
                  position: absolute;
                  right: 5px;
                  top: 0;
                  transform: rotate(180deg);
                }
              }
              .input {
                position: relative;
                width: calc(100% - 12px);
                height: 36px;
                line-height: 36px;
                border: 1px solid #F0F2F7;
                border-radius: 4px;
                text-align: left;
                padding: 0 10px;
              }
              .dialog {
                position: relative;
                width: calc(100% - 12px);
                height: 36px;
                line-height: 36px;
                border: 1px solid #F0F2F7;
                border-radius: 4px;
                text-align: left;
                padding: 0 10px;
                &::after {
                  content: "";
                  color: #C0C4CC;
                  font-size: 14px;
                  font-family: "element-icons";
                  width: 25px;
                  text-align: center;
                  display: inline-block;
                  position: absolute;
                  right: 4px;
                  top: 0;
                }
              }
            }
            .table-form-item {
              margin: 0;
              .diff-tag {
                display: none;
              }
              /**.edit-tag {
                position: absolute;
                top: 0;
                left: 0;
                width: 8px!important;
                height: 0;
                border-left: 8px solid #1890ff;
                border-bottom: 5px solid #fff;
                z-index: 1;
              }
              &.is-error .edit-tag{
                border-left-color: #ff4949;
              } */

              /** 错误 */
              &.is-error .edit-border {
                background: #FFF7F5;
                border: 1px solid #F2B1A7;
              }
              &.diff{
                color: #ff4949;
                .diff-tag{
                  display: block;
                  position: absolute;
                  top: 30px;
                  left: 0;
                }

              }
              .el-form-item__content{
                & >div >div{
                  width: calc(100% - 14px);
                  &.has-unit {
                    width: 80%;
                    float: left;
                  }
                  // .el-date-editor,.el-input {
                  //   width: calc(100% - 12px);
                  // }
                  /*&.el-select .el-input input {
                    height: 45px;
                    line-height: 45px;
                    border-radius: 0;
                    border-color: #fff;
                    padding: 0 30px 0 10px;
                    &:focus {
                      outline: none;
                      border-color: #1890ff;
                    }
                  }
                  &.el-input input {
                    height: 45px;
                    line-height: 45px;
                    border-radius: 0;
                    border-color: #fff;
                    padding: 0 30px 0 10px;
                    &:focus {
                      outline: none;
                      border-color: #1890ff;
                    }
                  }
                  &.el-date-editor input {
                    height: 45px;
                    line-height: 45px;
                    border-radius: 0;
                    padding: 0 30px;
                  }*/
                }
                .el-form-item__error{
                  display: none;
                  position: absolute;
                  top: unset;
                  bottom: 42px;
                  left: 0;
                  background-color: antiquewhite;
                  z-index: 999999;
                  width: 100%;
                  padding: 10px;
                  border-radius: 5px;
                  line-height: 20px;
                  &::after{
                    content: "";
                    height: 0;
                    width: 0;
                    border: 4px solid transparent;
                    border-top-color: antiquewhite;
                    position: absolute;
                    top: 100%;
                    right: 20px;
                  }
                }
              }
            }
          }
          /** 复选框 */
          &.el-table-column--selection .cell {
            padding: 0 10px;
          }
        }

        .el-table__empty-block {
          background-image: none;
          min-height: 60px!important;;
          margin-bottom: 0;
          span {
            bottom: 0;
          }
        }
      }
      ::v-deep .warning-row {
        background: oldlace;
      }

      ::v-deep .success-row {
        background: #f0f9eb;
      }
    }
    &.edit-table-disabled .edit-table {
      ::v-deep .el-table__body-wrapper {
        .el-table__cell .column-text .edit-border{
          cursor: not-allowed;
          background-color: #f9f9f9!important;
          border-color: #f0f2f7;
          color: #BBBBBB;
        }
      }
    }
  }
  .no-btns {
    .edit-table {
      margin-top: 0;
    }
  }
</style>

column.vue

import Vue from 'vue';
// import { hasClass, addClass, removeClass } from 'element-ui/src/utils/dom';
// import ElCheckbox from 'element-ui/packages/checkbox';
// import FilterPanel from './filter-panel.vue';
// import LayoutObserver from './layout-observer';
// import { mapStates } from './store/helper';

export default {
  name: 'EditColumnRender',

  props: {
    column: {
      type: Object
    },
    scope: {
      type: Object
    },
    store: {
      type: Object
      // required: true
    },
  },

  render(h) {
    return (
      <div>
        {
          this.column.renderTemp
            ? this.column.renderTemp.call(this._renderProxy, h, { column: this.column, scope: this.scope, store: this.store, _self: this.$parent.$vnode.context })
            : ''
        }
      </div>
    );
  },

  components: {
    
  },

  created() {
    
  },

  mounted() {
  },

  beforeDestroy() {
  },

  methods: {
  },

  data() {
    return {
      draggingColumn: null,
      dragging: false,
      dragState: {}
    };
  }
};

支持column-render.js ,支持render自定义列内容

import Vue from 'vue';
// import { hasClass, addClass, removeClass } from 'element-ui/src/utils/dom';
// import ElCheckbox from 'element-ui/packages/checkbox';
// import FilterPanel from './filter-panel.vue';
// import LayoutObserver from './layout-observer';
// import { mapStates } from './store/helper';

export default {
  name: 'EditColumnRender',

  props: {
    column: {
      type: Object
    },
    scope: {
      type: Object
    },
    store: {
      type: Object
      // required: true
    },
  },

  render(h) {
    return (
      <div>
        {
          this.column.renderTemp
            ? this.column.renderTemp.call(this._renderProxy, h, { column: this.column, scope: this.scope, store: this.store, _self: this.$parent.$vnode.context })
            : ''
        }
      </div>
    );
  },

  components: {
    
  },

  created() {
    
  },

  mounted() {
  },

  beforeDestroy() {
  },

  methods: {
  },

  data() {
    return {
      draggingColumn: null,
      dragging: false,
      dragState: {}
    };
  }
};

使用方法如下

我这里全局引用了,你也可以局部引用进来

<edit-table
          ref="editTable"
          :empty-row="emptyRow"
          :column-data="columnData"
          :table-data.sync="tableData"
          :table-rules="tableRules"
          row-key="id"
        ></edit-table>
data() {
    return {
      symbolList: [
        {label: '相似', value: "%%%"},
        {label: '左相似', value: "%%"},
        {label: '右相似', value: "%"},
        {label: '大于', value: ">"},
        {label: '大于等于', value: ">="},
        {label: '等于', value: "="},
        {label: '小于', value: "<"},
        {label: '小于等于', value: "<="}
      ],
      relationList: [
        {label: '并且', value: '&&'},
        {label: '或者', value: '||'}
      ],
      columnData: [
        { prop: 'name', type: 'selection' },
        {
          label: "列名",
          prop: "queryColumn",
          edit: true,
          editType: "select",
          selectData: {
            label: "desc",
            value: "value",
            getData: () => {
              return this.fields
            }
          },
          format: (scope) => {
            let current = this.fields.find(
              (item) => item.value === scope.row.queryColumn
            );
            if (current) return current.desc;
          }
        },
        {
          label: "计算符",
          prop: "querySymbol",
          edit: true,
          editType: "select",
          selectData: {
            label: "label",
            value: "value",
            getData: () => {
              return this.symbolList
            }
          },
          format: (scope) => {
            let current = this.symbolList.find(
              (item) => item.value === scope.row.querySymbol
            );
            if (current) return current.label;
          }
        },
        { label: "值", prop: "queryValue", edit: true, editType: 'template', renderTemp: (h, {scope})=>{
          return h(ColumnEdit, {
              props: {
                column: 'queryValue',
                scope: scope
              }
            }
          )
        }},
        {
          label: "列关系",
          prop: "queryRelation",
          edit: true,
          editType: "select",
          selectData: {
            label: "label",
            value: "value",
            getData: () => {
              return this.relationList
            }
          },
          format: (scope) => {
            let current = this.relationList.find(
              (item) => item.value === scope.row.queryRelation
            );
            if (current) return current.label;
          }
        },
      ],
      emptyRow: {
        queryColumn: null,
        querySymbol: null,
        queryValue: null,
        queryRelation: '&&'
      },
      tableData: [],
      tableRules: {
        queryColumn: [
          { required: true, message: "列名不能为空", trigger: "change" },
        ],
        querySymbol: [
          { required: true, message: "计算符不能为空", trigger: "change" },
        ],
        queryValue: [
          { required: true, message: "值不能为空", trigger: "change" },
        ],
        queryRelation: [
          { required: true, message: "列关系不能为空", trigger: "change" },
        ],
      },
    };

;