import { reactive, nextTick } from 'vue' import XEUtils from 'xe-utils' import { VxeUI } from '../../../ui' import { isEnableConf } from '../../../ui/src/utils' import { getCellValue, setCellValue, getRowid } from '../../src/util' import { removeClass, addClass } from '../../../ui/src/dom' import { warnLog, errLog } from '../../../ui/src/log' import type { TableEditMethods, TableEditPrivateMethods, VxeTableDefines } from '../../../../types' const { getConfig, renderer, hooks, getI18n } = VxeUI const tableEditMethodKeys: (keyof TableEditMethods)[] = ['insert', 'insertAt', 'insertNextAt', 'insertChild', 'insertChildAt', 'insertChildNextAt', 'remove', 'removeCheckboxRow', 'removeRadioRow', 'removeCurrentRow', 'getRecordset', 'getInsertRecords', 'getRemoveRecords', 'getUpdateRecords', 'getEditRecord', 'getActiveRecord', 'getEditCell', 'getSelectedCell', 'clearEdit', 'clearActived', 'clearSelected', 'isEditByRow', 'isActiveByRow', 'setEditRow', 'setActiveRow', 'setEditCell', 'setActiveCell', 'setSelectCell'] hooks.add('tableEditModule', { setupTable ($xeTable) { const { props, reactData, internalData } = $xeTable const { refElem } = $xeTable.getRefMaps() const { computeMouseOpts, computeEditOpts, computeCheckboxOpts, computeTreeOpts, computeValidOpts } = $xeTable.getComputeMaps() const browseObj = XEUtils.browse() let editMethods = {} as TableEditMethods let editPrivateMethods = {} as TableEditPrivateMethods const getEditColumnModel = (row: any, column: VxeTableDefines.ColumnInfo) => { const { model, editRender } = column if (editRender) { model.value = getCellValue(row, column) model.update = false } } const setEditColumnModel = (row: any, column: VxeTableDefines.ColumnInfo) => { const { model, editRender } = column if (editRender && model.update) { setCellValue(row, column, model.value) model.update = false model.value = null } } const removeCellSelectedClass = () => { const el = refElem.value if (el) { const cell = el.querySelector('.col--selected') if (cell) { removeClass(cell, 'col--selected') } } } const syncActivedCell = () => { const { editStore, tableColumn } = reactData const editOpts = computeEditOpts.value const { actived } = editStore const { row, column } = actived if (row || column) { if (editOpts.mode === 'row') { tableColumn.forEach((column) => setEditColumnModel(row, column)) } else { setEditColumnModel(row, column) } } } const insertTreeRow = (newRecords: any[], isAppend: boolean) => { const { tableFullTreeData, afterFullData, fullDataRowIdData, fullAllDataRowIdData } = internalData const treeOpts = computeTreeOpts.value const { rowField, parentField, mapChildrenField } = treeOpts const childrenField = treeOpts.children || treeOpts.childrenField const funcName = isAppend ? 'push' : 'unshift' newRecords.forEach(item => { const parentRowId = item[parentField] const rowid = getRowid($xeTable, item) const matchObj = parentRowId ? XEUtils.findTree(tableFullTreeData, item => parentRowId === item[rowField], { children: mapChildrenField }) : null if (matchObj) { const { item: parentRow } = matchObj const parentRest = fullAllDataRowIdData[getRowid($xeTable, parentRow)] const parentLevel = parentRest ? parentRest.level : 0 let parentChilds = parentRow[childrenField] let mapChilds = parentRow[mapChildrenField] if (!XEUtils.isArray(parentChilds)) { parentChilds = parentRow[childrenField] = [] } if (!XEUtils.isArray(mapChilds)) { mapChilds = parentRow[childrenField] = [] } parentChilds[funcName](item) mapChilds[funcName](item) const rest = { row: item, rowid, seq: -1, index: -1, _index: -1, $index: -1, treeIndex: -1, _tIndex: -1, items: parentChilds, parent: parentRow, level: parentLevel + 1, height: 0, resizeHeight: 0, oTop: 0, expandHeight: 0 } fullDataRowIdData[rowid] = rest fullAllDataRowIdData[rowid] = rest } else { if (parentRowId) { warnLog('vxe.error.unableInsert') } afterFullData[funcName](item) tableFullTreeData[funcName](item) const rest = { row: item, rowid, seq: -1, index: -1, _index: -1, $index: -1, treeIndex: -1, _tIndex: -1, items: tableFullTreeData, parent: null, level: 0, height: 0, resizeHeight: 0, oTop: 0, expandHeight: 0 } fullDataRowIdData[rowid] = rest fullAllDataRowIdData[rowid] = rest } }) } // const insertGroupRow = (newRecords: any[], isAppend: boolean) => { // } const handleInsertRowAt = (records: any, targetRow: any, isInsertNextRow?: boolean) => { const { treeConfig } = props const { isRowGroupStatus } = reactData const { tableFullTreeData, afterFullData, mergeBodyList, tableFullData, fullDataRowIdData, fullAllDataRowIdData, insertRowMaps, removeRowMaps } = internalData const treeOpts = computeTreeOpts.value const { transform, parentField, rowField, mapChildrenField } = treeOpts const childrenField = treeOpts.children || treeOpts.childrenField if (!XEUtils.isArray(records)) { records = [records] } const newRecords: any[] = reactive($xeTable.defineField(records.map((record: any) => Object.assign(treeConfig && transform ? { [mapChildrenField]: [], [childrenField]: [] } : {}, record)))) let treeRecords: any[] = [] if (treeConfig && transform) { treeRecords = XEUtils.toArrayTree(newRecords, { key: rowField, parentKey: parentField, children: childrenField }) } if (XEUtils.eqNull(targetRow)) { // 如果为虚拟树 if (treeConfig && transform) { insertTreeRow(newRecords, false) } else if (isRowGroupStatus) { // 如果分组 if (treeConfig) { throw new Error(getI18n('vxe.error.noTree', ['insert'])) } warnLog(getI18n('vxe.error.noGroup', ['remove'])) // insertGroupRow(newRecords, false) } else { newRecords.forEach(item => { const rowid = getRowid($xeTable, item) const rest = { row: item, rowid, seq: -1, index: -1, _index: -1, $index: -1, treeIndex: -1, _tIndex: -1, items: afterFullData, parent: null, level: 0, height: 0, resizeHeight: 0, oTop: 0, expandHeight: 0 } fullDataRowIdData[rowid] = rest fullAllDataRowIdData[rowid] = rest afterFullData.unshift(item) tableFullData.unshift(item) }) // 刷新单元格合并 mergeBodyList.forEach((mergeItem: any) => { const { row: mergeRowIndex } = mergeItem if (mergeRowIndex >= 0) { mergeItem.row = mergeRowIndex + newRecords.length } }) } } else { if (targetRow === -1) { // 如果为虚拟树 if (treeConfig && transform) { insertTreeRow(newRecords, true) } else if (isRowGroupStatus) { // 如果分组 if (treeConfig) { throw new Error(getI18n('vxe.error.noTree', ['insert'])) } warnLog(getI18n('vxe.error.noGroup', ['remove'])) // insertGroupRow(newRecords, true) } else { newRecords.forEach(item => { const rowid = getRowid($xeTable, item) const rest = { row: item, rowid, seq: -1, index: -1, _index: -1, treeIndex: -1, _tIndex: -1, $index: -1, items: afterFullData, parent: null, level: 0, height: 0, resizeHeight: 0, oTop: 0, expandHeight: 0 } fullDataRowIdData[rowid] = rest fullAllDataRowIdData[rowid] = rest afterFullData.push(item) tableFullData.push(item) }) } } else { // 如果为虚拟树 if (treeConfig && transform) { const matchMapObj = XEUtils.findTree(tableFullTreeData, item => targetRow[rowField] === item[rowField], { children: mapChildrenField }) if (matchMapObj) { const { parent: parentRow } = matchMapObj const parentMapChilds = parentRow ? parentRow[mapChildrenField] : tableFullTreeData const parentRest = fullAllDataRowIdData[getRowid($xeTable, parentRow)] const parentLevel = parentRest ? parentRest.level : 0 treeRecords.forEach((row, i) => { if (parentRow) { if (row[parentField] !== parentRow[rowField]) { row[parentField] = parentRow[rowField] errLog('vxe.error.errProp', [`${parentField}=${row[parentField]}`, `${parentField}=${parentRow[rowField]}`]) } } else { if (row[parentField] !== null) { row[parentField] = null errLog('vxe.error.errProp', [`${parentField}=${row[parentField]}`, 'null']) } } let targetIndex = matchMapObj.index + i if (isInsertNextRow) { targetIndex = targetIndex + 1 } parentMapChilds.splice(targetIndex, 0, row) }) XEUtils.eachTree(treeRecords, (item) => { const rowid = getRowid($xeTable, item) const rest = { row: item, rowid, seq: -1, index: -1, _index: -1, $index: -1, treeIndex: -1, _tIndex: -1, items: parentMapChilds, parent: parentRow, level: parentLevel + 1, height: 0, resizeHeight: 0, oTop: 0, expandHeight: 0 } if (item[childrenField]) { item[mapChildrenField] = item[childrenField] } fullDataRowIdData[rowid] = rest fullAllDataRowIdData[rowid] = rest }, { children: childrenField }) // 源 if (parentRow) { const matchObj = XEUtils.findTree(tableFullTreeData, item => targetRow[rowField] === item[rowField], { children: childrenField }) if (matchObj) { const parentChilds = matchObj.items let targetIndex = matchObj.index if (isInsertNextRow) { targetIndex = targetIndex + 1 } parentChilds.splice(targetIndex, 0, ...treeRecords) } } } else { warnLog('vxe.error.unableInsert') insertTreeRow(newRecords, true) } } else if (isRowGroupStatus) { // 如果分组 if (treeConfig) { throw new Error(getI18n('vxe.error.noTree', ['insert'])) } warnLog(getI18n('vxe.error.noGroup', ['remove'])) } else { if (treeConfig) { throw new Error(getI18n('vxe.error.noTree', ['insert'])) } let afIndex = -1 // 如果是可视索引 if (XEUtils.isNumber(targetRow)) { if (targetRow < afterFullData.length) { afIndex = targetRow } } else { afIndex = $xeTable.findRowIndexOf(afterFullData, targetRow) } // 如果是插入指定行的下一行 if (isInsertNextRow) { afIndex = Math.min(afterFullData.length, afIndex + 1) } if (afIndex === -1) { throw new Error(getI18n('vxe.error.unableInsert')) } afterFullData.splice(afIndex, 0, ...newRecords) const tfIndex = $xeTable.findRowIndexOf(tableFullData, targetRow) if (tfIndex > -1) { tableFullData.splice(tfIndex + (isInsertNextRow ? 1 : 0), 0, ...newRecords) } else { tableFullData.push(...newRecords) } // 刷新单元格合并 mergeBodyList.forEach((mergeItem: any) => { const { row: mergeRowIndex, rowspan: mergeRowspan } = mergeItem if (mergeRowIndex >= afIndex) { mergeItem.row = mergeRowIndex + newRecords.length } else if (isInsertNextRow ? (mergeRowIndex + mergeRowspan >= afIndex) : (mergeRowIndex + mergeRowspan > afIndex)) { mergeItem.rowspan = mergeRowspan + newRecords.length } }) } } } const handleStatus = (newRow: any) => { const rowid = getRowid($xeTable, newRow) // 如果是被删除的数据,则还原状态 if (removeRowMaps[rowid]) { delete removeRowMaps[rowid] if (insertRowMaps[rowid]) { delete insertRowMaps[rowid] } } else { insertRowMaps[rowid] = newRow } } // 如果为虚拟树 if (treeConfig && transform) { XEUtils.eachTree(treeRecords, handleStatus, { children: mapChildrenField }) } else { newRecords.forEach(handleStatus) } reactData.removeRowFlag++ reactData.insertRowFlag++ $xeTable.cacheRowMap(false) $xeTable.updateScrollYStatus() $xeTable.handleTableData(treeConfig && transform) if (!(treeConfig && transform)) { $xeTable.updateAfterDataIndex() } $xeTable.updateFooter() $xeTable.handleUpdateBodyMerge() $xeTable.checkSelectionStatus() if (reactData.scrollYLoad) { $xeTable.updateScrollYSpace() } return nextTick().then(() => { $xeTable.updateCellAreas() return $xeTable.recalculate(true) }).then(() => { return { row: newRecords.length ? newRecords[newRecords.length - 1] : null, rows: newRecords } }) } const handleInsertChildRowAt = (records: any, parentRow: any, targetRow: any, isInsertNextRow?: boolean) => { const { treeConfig } = props const treeOpts = computeTreeOpts.value const { transform, rowField, parentField } = treeOpts if (treeConfig && transform) { if (!XEUtils.isArray(records)) { records = [records] } return handleInsertRowAt(records.map((item: any) => Object.assign({}, item, { [parentField]: parentRow[rowField] })), targetRow, isInsertNextRow) } else { errLog('vxe.error.errProp', ['tree-config.transform=false', 'tree-config.transform=true']) } return Promise.resolve({ row: null, rows: [] }) } const handleClearEdit = (evnt: Event | null, targetRow?: any) => { const { editStore } = reactData const { actived, focused } = editStore const { row, column } = actived const validOpts = computeValidOpts.value if (row || column) { if (targetRow && getRowid($xeTable, targetRow) !== getRowid($xeTable, row)) { return nextTick() } syncActivedCell() actived.args = null actived.row = null actived.column = null $xeTable.updateFooter() $xeTable.dispatchEvent('edit-closed', { row, rowIndex: $xeTable.getRowIndex(row), $rowIndex: $xeTable.getVMRowIndex(row), column, columnIndex: $xeTable.getColumnIndex(column), $columnIndex: $xeTable.getVMColumnIndex(column) }, evnt || null) } focused.row = null focused.column = null if (validOpts.autoClear) { if (validOpts.msgMode !== 'full' || getConfig().cellVaildMode === 'obsolete') { if ($xeTable.clearValidate) { return $xeTable.clearValidate() } } } return nextTick().then(() => $xeTable.updateCellAreas()) } const handleEditActive = (params: any, evnt: Event | null, isFocus: boolean, isPos: boolean) => { const $xeGrid = $xeTable.xeGrid const $xeGantt = $xeTable.xeGantt const { editConfig, mouseConfig } = props const { editStore, tableColumn } = reactData const editOpts = computeEditOpts.value const { mode } = editOpts const { actived, focused } = editStore const { row, column } = params const { editRender } = column const cell = (params.cell || $xeTable.getCellElement(row, column)) const beforeEditMethod = editOpts.beforeEditMethod || editOpts.activeMethod params.cell = cell if (cell && isEnableConf(editConfig) && isEnableConf(editRender)) { // 激活编辑 if (!$xeTable.isPendingByRow(row) && !$xeTable.isAggregateRecord(row)) { if (actived.row !== row || (mode === 'cell' ? actived.column !== column : false)) { // 判断是否禁用编辑 let type: 'edit-disabled' | 'edit-activated' = 'edit-disabled' if (!beforeEditMethod || beforeEditMethod({ ...params, $table: $xeTable, $grid: $xeGrid, $gantt: $xeGantt })) { if (mouseConfig) { $xeTable.clearSelected() if ($xeTable.clearCellAreas) { $xeTable.clearCellAreas() $xeTable.clearCopyCellArea() } } $xeTable.closeTooltip() if (actived.column) { handleClearEdit(evnt) } type = 'edit-activated' column.renderHeight = cell.offsetHeight actived.args = params actived.row = row actived.column = column if (mode === 'row') { tableColumn.forEach((column: any) => getEditColumnModel(row, column)) } else { getEditColumnModel(row, column) } const afterEditMethod = editOpts.afterEditMethod nextTick(() => { if (isFocus) { $xeTable.handleFocus(params, evnt) } if (afterEditMethod) { afterEditMethod({ ...params, $table: $xeTable, $grid: $xeGrid, $gantt: $xeGantt }) } }) } $xeTable.dispatchEvent(type, { row, rowIndex: $xeTable.getRowIndex(row), $rowIndex: $xeTable.getVMRowIndex(row), column, columnIndex: $xeTable.getColumnIndex(column), $columnIndex: $xeTable.getVMColumnIndex(column) }, evnt) // v4已废弃 if (type === 'edit-activated') { $xeTable.dispatchEvent('edit-actived', { row, rowIndex: $xeTable.getRowIndex(row), $rowIndex: $xeTable.getVMRowIndex(row), column, columnIndex: $xeTable.getColumnIndex(column), $columnIndex: $xeTable.getVMColumnIndex(column) }, evnt) } } else { const { column: oldColumn } = actived if (mouseConfig) { $xeTable.clearSelected() if ($xeTable.clearCellAreas) { $xeTable.clearCellAreas() $xeTable.clearCopyCellArea() } } if (oldColumn !== column) { const { model: oldModel } = oldColumn if (oldModel.update) { setCellValue(row, oldColumn, oldModel.value) } if ($xeTable.clearValidate) { $xeTable.clearValidate(row, column) } } column.renderHeight = cell.offsetHeight actived.args = params actived.column = column if (isPos) { setTimeout(() => { $xeTable.handleFocus(params, evnt) }) } } focused.column = null focused.row = null $xeTable.focus() } } return nextTick() } const handleEditCell = (row: any, fieldOrColumn: string | VxeTableDefines.ColumnInfo, isPos: boolean) => { const { editConfig } = props const column = XEUtils.isString(fieldOrColumn) ? $xeTable.getColumnByField(fieldOrColumn) : fieldOrColumn if (row && column && isEnableConf(editConfig) && isEnableConf(column.editRender) && !$xeTable.isAggregateRecord(row)) { return Promise.resolve(isPos ? $xeTable.scrollToRow(row, column) : null).then(() => { const cell = $xeTable.getCellElement(row, column) if (cell) { handleEditActive({ row, rowIndex: $xeTable.getRowIndex(row), column, columnIndex: $xeTable.getColumnIndex(column), cell, $table: $xeTable }, null, isPos, isPos) internalData._lastCallTime = Date.now() } return nextTick() }) } return nextTick() } editMethods = { /** * 往表格中插入临时数据 * * @param {*} records */ insert (records) { return handleInsertRowAt(records, null) }, /** * 往表格指定行中插入临时数据 * 如果 row 为空则从插入到顶部,如果为树结构,则插入到目标节点顶部 * 如果 row 为 -1 则从插入到底部,如果为树结构,则插入到目标节点底部 * 如果 row 为有效行则插入到该行的位置,如果为树结构,则有插入到效的目标节点该行的位置 * @param {Object/Array} records 新的数据 * @param {Row} targetRow 指定行 */ insertAt (records, targetRow) { return handleInsertRowAt(records, targetRow) }, insertNextAt (records, targetRow) { return handleInsertRowAt(records, targetRow, true) }, insertChild (records, parentRow) { return handleInsertChildRowAt(records, parentRow, null) }, insertChildAt (records, parentRow, targetRow) { return handleInsertChildRowAt(records, parentRow, targetRow) }, insertChildNextAt (records, parentRow, targetRow) { return handleInsertChildRowAt(records, parentRow, targetRow, true) }, /** * 删除指定行数据 * 如果传 row 则删除一行 * 如果传 rows 则删除多行 * 如果为空则删除所有 */ remove (rows: any) { const { treeConfig } = props const { editStore, isRowGroupStatus } = reactData const { tableFullTreeData, selectCheckboxMaps, afterFullData, mergeBodyList, tableFullData, pendingRowMaps, insertRowMaps, removeRowMaps } = internalData const checkboxOpts = computeCheckboxOpts.value const treeOpts = computeTreeOpts.value const { transform, mapChildrenField } = treeOpts const childrenField = treeOpts.children || treeOpts.childrenField const { actived } = editStore const { checkField } = checkboxOpts let delList: any[] = [] if (!rows) { rows = tableFullData } else if (!XEUtils.isArray(rows)) { rows = [rows] } // 如果是新增,则保存记录 rows.forEach((row: any) => { if (!$xeTable.isInsertByRow(row)) { const rowid = getRowid($xeTable, row) removeRowMaps[rowid] = row } }) // 如果绑定了多选属性,则更新状态 if (!checkField) { rows.forEach((row: any) => { const rowid = getRowid($xeTable, row) if (selectCheckboxMaps[rowid]) { delete selectCheckboxMaps[rowid] } }) reactData.updateCheckboxFlag++ } // 从数据源中移除 if (tableFullData === rows) { rows = delList = tableFullData.slice(0) internalData.tableFullData = [] internalData.afterFullData = [] $xeTable.clearMergeCells() } else { // 如果为虚拟树 if (treeConfig && transform) { rows.forEach((row: any) => { const rowid = getRowid($xeTable, row) const matchMapObj = XEUtils.findTree(tableFullTreeData, item => rowid === getRowid($xeTable, item), { children: mapChildrenField }) if (matchMapObj) { const rItems = matchMapObj.items.splice(matchMapObj.index, 1) delList.push(rItems[0]) } const matchObj = XEUtils.findTree(tableFullTreeData, item => rowid === getRowid($xeTable, item), { children: childrenField }) if (matchObj) { matchObj.items.splice(matchObj.index, 1) } const afIndex = $xeTable.findRowIndexOf(afterFullData, row) if (afIndex > -1) { afterFullData.splice(afIndex, 1) } }) } else if (isRowGroupStatus) { // 如果分组 warnLog(getI18n('vxe.error.noGroup', ['remove'])) } else { rows.forEach((row: any) => { const tfIndex = $xeTable.findRowIndexOf(tableFullData, row) if (tfIndex > -1) { const rItems = tableFullData.splice(tfIndex, 1) delList.push(rItems[0]) } const afIndex = $xeTable.findRowIndexOf(afterFullData, row) if (afIndex > -1) { // 刷新单元格合并 mergeBodyList.forEach((mergeItem: any) => { const { row: mergeRowIndex, rowspan: mergeRowspan } = mergeItem if (mergeRowIndex > afIndex) { mergeItem.row = mergeRowIndex - 1 } else if (mergeRowIndex + mergeRowspan > afIndex) { mergeItem.rowspan = mergeRowspan - 1 } }) afterFullData.splice(afIndex, 1) } }) } } // 如果当前行被激活编辑,则清除激活状态 if (actived.row && $xeTable.findRowIndexOf(rows, actived.row) > -1) { editMethods.clearEdit() } // 从新增中移除已删除的数据 rows.forEach((row: any) => { const rowid = getRowid($xeTable, row) if (insertRowMaps[rowid]) { delete insertRowMaps[rowid] } if (pendingRowMaps[rowid]) { delete pendingRowMaps[rowid] } }) reactData.removeRowFlag++ reactData.insertRowFlag++ reactData.pendingRowFlag++ $xeTable.cacheRowMap(false) $xeTable.handleTableData(treeConfig && transform) $xeTable.updateFooter() $xeTable.handleUpdateBodyMerge() if (!(treeConfig && transform)) { $xeTable.updateAfterDataIndex() } $xeTable.checkSelectionStatus() if (reactData.scrollYLoad) { $xeTable.updateScrollYSpace() } return nextTick().then(() => { $xeTable.updateCellAreas() return $xeTable.recalculate(true) }).then(() => { return { row: delList.length ? delList[delList.length - 1] : null, rows: delList } }) }, /** * 删除复选框选中的数据 */ removeCheckboxRow () { return editMethods.remove($xeTable.getCheckboxRecords()).then((params: any) => { $xeTable.clearCheckboxRow() return params }) }, /** * 删除单选框选中的数据 */ removeRadioRow () { const radioRecord = $xeTable.getRadioRecord() return editMethods.remove(radioRecord || []).then((params: any) => { $xeTable.clearRadioRow() return params }) }, /** * 删除当前行选中的数据 */ removeCurrentRow () { const currentRecord = $xeTable.getCurrentRecord() return editMethods.remove(currentRecord || []).then((params: any) => { $xeTable.clearCurrentRow() return params }) }, /** * 获取表格数据集,包含新增、删除、修改、标记 */ getRecordset () { const removeRecords = editMethods.getRemoveRecords() const pendingRecords = $xeTable.getPendingRecords() const delRecords = removeRecords.concat(pendingRecords) // 如果已经被删除,则无需放到更新数组 const updateRecords = editMethods.getUpdateRecords().filter(row => { return !delRecords.some(item => $xeTable.eqRow(item, row)) }) return { insertRecords: editMethods.getInsertRecords(), removeRecords, updateRecords, pendingRecords } }, /** * 获取新增的临时数据 */ getInsertRecords () { const { fullAllDataRowIdData, insertRowMaps } = internalData const insertRecords: any[] = [] XEUtils.each(insertRowMaps, (row, rowid) => { if (fullAllDataRowIdData[rowid]) { insertRecords.push(row) } }) return insertRecords }, /** * 获取已删除的数据 */ getRemoveRecords () { const { removeRowMaps } = internalData const removeRecords: any[] = [] XEUtils.each(removeRowMaps, (row) => { removeRecords.push(row) }) return removeRecords }, /** * 获取更新数据 * 只精准匹配 row 的更改 * 如果是树表格,子节点更改状态不会影响父节点的更新状态 */ getUpdateRecords () { const { keepSource, treeConfig } = props const { tableFullData } = internalData const treeOpts = computeTreeOpts.value if (keepSource) { syncActivedCell() if (treeConfig) { return XEUtils.filterTree(tableFullData, row => $xeTable.isUpdateByRow(row), treeOpts) } return tableFullData.filter((row: any) => $xeTable.isUpdateByRow(row)) } return [] }, getActiveRecord () { warnLog('vxe.error.delFunc', ['getActiveRecord', 'getEditCell']) const { editStore } = reactData const { fullAllDataRowIdData } = internalData const { args, row } = editStore.actived if (args && row && fullAllDataRowIdData[getRowid($xeTable, row)]) { return Object.assign({}, args, { row }) } return null }, getEditRecord () { warnLog('vxe.error.delFunc', ['getEditRecord', 'getEditCell']) const { editStore } = reactData const { fullAllDataRowIdData } = internalData const { args, row } = editStore.actived if (args && row && fullAllDataRowIdData[getRowid($xeTable, row)]) { return Object.assign({}, args, { row }) } return null }, getEditCell () { const { editStore } = reactData const { row, column } = editStore.actived if (column && row) { return { row, rowIndex: $xeTable.getRowIndex(row), column, columnIndex: $xeTable.getColumnIndex(column) } } return null }, /** * 获取选中的单元格 */ getSelectedCell () { const { editStore } = reactData const { row, column } = editStore.selected if (row && column) { return { row, column } } return null }, clearActived (row) { // 即将废弃 warnLog('vxe.error.delFunc', ['clearActived', 'clearEdit']) return $xeTable.clearEdit(row) }, /** * 清除激活的编辑 */ clearEdit (row) { return handleClearEdit(null, row) }, /** * 清除所选中源状态 */ clearSelected () { const { editStore } = reactData const { selected } = editStore selected.row = null selected.column = null removeCellSelectedClass() return nextTick() }, isActiveByRow (row) { warnLog('vxe.error.delFunc', ['isActiveByRow', 'isEditByRow']) // 即将废弃 return $xeTable.isEditByRow(row) }, /** * 判断行是否为激活编辑状态 * @param {Row} row 行对象 */ isEditByRow (row) { const { editStore } = reactData return editStore.actived.row === row }, setActiveRow (row) { warnLog('vxe.error.delFunc', ['setActiveRow', 'setEditRow']) // 即将废弃 return editMethods.setEditRow(row) }, /** * 激活行编辑 */ setEditRow (row, fieldOrColumn) { const { visibleColumn } = internalData let column: any = XEUtils.find(visibleColumn, column => isEnableConf(column.editRender)) let isPos = false if (fieldOrColumn) { isPos = true if (fieldOrColumn !== true) { column = XEUtils.isString(fieldOrColumn) ? $xeTable.getColumnByField(fieldOrColumn) : fieldOrColumn } } return handleEditCell(row, column, isPos) }, setActiveCell (row, fieldOrColumn) { warnLog('vxe.error.delFunc', ['setActiveCell', 'setEditCell']) // 即将废弃 return editMethods.setEditCell(row, fieldOrColumn) }, /** * 激活单元格编辑 */ setEditCell (row, fieldOrColumn) { return handleEditCell(row, fieldOrColumn, true) }, /** * 只对 trigger=dblclick 有效,选中单元格 */ setSelectCell (row, fieldOrColumn) { const { tableData } = reactData const editOpts = computeEditOpts.value const column = XEUtils.isString(fieldOrColumn) ? $xeTable.getColumnByField(fieldOrColumn) : fieldOrColumn if (row && column && editOpts.trigger !== 'manual') { const rowIndex = $xeTable.findRowIndexOf(tableData, row) if (rowIndex > -1 && column) { const cell = $xeTable.getCellElement(row, column) const params = { row, rowIndex, column, columnIndex: $xeTable.getColumnIndex(column), cell } $xeTable.handleSelected(params, {}) } } return nextTick() } } editPrivateMethods = { /** * 处理激活编辑 */ handleEdit (params, evnt) { return handleEditActive(params, evnt, true, true) }, /** * @deprecated */ handleActived (params, evnt) { return editPrivateMethods.handleEdit(params, evnt) }, /** * 处理取消编辑 * @param evnt * @returns */ handleClearEdit, /** * 处理聚焦 */ handleFocus (params) { const { row, column, cell } = params const { editRender } = column const editOpts = computeEditOpts.value if (isEnableConf(editRender)) { const compRender = renderer.get(editRender.name) let autoFocus = editRender.autofocus || editRender.autoFocus let autoSelect = editRender.autoSelect || editRender.autoselect let inputElem // 是否启用聚焦 if (editOpts.autoFocus) { if (!autoFocus && compRender) { autoFocus = compRender.tableAutoFocus || compRender.tableAutofocus || compRender.autofocus } if (!autoSelect && compRender) { autoSelect = compRender.tableAutoSelect || compRender.autoselect } // 如果指定了聚焦 class if (XEUtils.isFunction(autoFocus)) { inputElem = autoFocus(params) } else if (autoFocus) { if (autoFocus === true) { // 自动匹配模式,会自动匹配第一个可输入元素 inputElem = cell.querySelector('input,textarea') } else { inputElem = cell.querySelector(autoFocus) } if (inputElem) { inputElem.focus() } } } if (inputElem) { if (autoSelect) { inputElem.select() } else { // 保持一致行为,光标移到末端 if (browseObj.msie) { const textRange = inputElem.createTextRange() textRange.collapse(false) textRange.select() } } } else { // 是否自动定位 if (editOpts.autoPos) { if (!column.fixed) { // 显示到可视区中 $xeTable.scrollToRow(row, column) } } } } }, /** * 处理选中源 */ handleSelected (params, evnt) { const { mouseConfig } = props const { editStore } = reactData const mouseOpts = computeMouseOpts.value const editOpts = computeEditOpts.value const { actived, selected } = editStore const { row, column } = params const isMouseSelected = mouseConfig && mouseOpts.selected const selectMethod = () => { if (isMouseSelected && (selected.row !== row || selected.column !== column)) { if (actived.row !== row || (editOpts.mode === 'cell' ? actived.column !== column : false)) { handleClearEdit(evnt) $xeTable.clearSelected() if ($xeTable.clearCellAreas) { $xeTable.clearCellAreas() $xeTable.clearCopyCellArea() } selected.args = params selected.row = row selected.column = column if (isMouseSelected) { editPrivateMethods.addCellSelectedClass() } $xeTable.focus() if (evnt) { $xeTable.dispatchEvent('cell-selected', params, evnt) } } } return nextTick() } return selectMethod() }, addCellSelectedClass () { const { editStore } = reactData const { selected } = editStore const { row, column } = selected removeCellSelectedClass() if (row && column) { const cell = $xeTable.getCellElement(row, column) if (cell) { addClass(cell, 'col--selected') } } } } return { ...editMethods, ...editPrivateMethods } }, setupGrid ($xeGrid) { return $xeGrid.extendTableMethods(tableEditMethodKeys) }, setupGantt ($xeGantt) { return $xeGantt.extendTableMethods(tableEditMethodKeys) } })