django-vue3-admin-web/node_modules/vxe-table/packages/table/module/edit/hook.ts
2025-10-20 21:21:14 +08:00

1017 lines
39 KiB
TypeScript

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)
}
})