1020 lines
48 KiB
JavaScript
1020 lines
48 KiB
JavaScript
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';
|
|
const { getConfig, renderer, hooks, getI18n } = VxeUI;
|
|
const tableEditMethodKeys = ['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 = {};
|
|
let editPrivateMethods = {};
|
|
const getEditColumnModel = (row, column) => {
|
|
const { model, editRender } = column;
|
|
if (editRender) {
|
|
model.value = getCellValue(row, column);
|
|
model.update = false;
|
|
}
|
|
};
|
|
const setEditColumnModel = (row, column) => {
|
|
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, isAppend) => {
|
|
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, targetRow, isInsertNextRow) => {
|
|
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 = reactive($xeTable.defineField(records.map((record) => Object.assign(treeConfig && transform ? { [mapChildrenField]: [], [childrenField]: [] } : {}, record))));
|
|
let treeRecords = [];
|
|
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) => {
|
|
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) => {
|
|
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) => {
|
|
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, parentRow, targetRow, isInsertNextRow) => {
|
|
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) => 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, targetRow) => {
|
|
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, evnt, isFocus, isPos) => {
|
|
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';
|
|
if (!beforeEditMethod || beforeEditMethod(Object.assign(Object.assign({}, 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) => getEditColumnModel(row, column));
|
|
}
|
|
else {
|
|
getEditColumnModel(row, column);
|
|
}
|
|
const afterEditMethod = editOpts.afterEditMethod;
|
|
nextTick(() => {
|
|
if (isFocus) {
|
|
$xeTable.handleFocus(params, evnt);
|
|
}
|
|
if (afterEditMethod) {
|
|
afterEditMethod(Object.assign(Object.assign({}, 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, fieldOrColumn, isPos) => {
|
|
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) {
|
|
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 = [];
|
|
if (!rows) {
|
|
rows = tableFullData;
|
|
}
|
|
else if (!XEUtils.isArray(rows)) {
|
|
rows = [rows];
|
|
}
|
|
// 如果是新增,则保存记录
|
|
rows.forEach((row) => {
|
|
if (!$xeTable.isInsertByRow(row)) {
|
|
const rowid = getRowid($xeTable, row);
|
|
removeRowMaps[rowid] = row;
|
|
}
|
|
});
|
|
// 如果绑定了多选属性,则更新状态
|
|
if (!checkField) {
|
|
rows.forEach((row) => {
|
|
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) => {
|
|
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) => {
|
|
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) => {
|
|
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) => {
|
|
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) => {
|
|
$xeTable.clearCheckboxRow();
|
|
return params;
|
|
});
|
|
},
|
|
/**
|
|
* 删除单选框选中的数据
|
|
*/
|
|
removeRadioRow() {
|
|
const radioRecord = $xeTable.getRadioRecord();
|
|
return editMethods.remove(radioRecord || []).then((params) => {
|
|
$xeTable.clearRadioRow();
|
|
return params;
|
|
});
|
|
},
|
|
/**
|
|
* 删除当前行选中的数据
|
|
*/
|
|
removeCurrentRow() {
|
|
const currentRecord = $xeTable.getCurrentRecord();
|
|
return editMethods.remove(currentRecord || []).then((params) => {
|
|
$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 = [];
|
|
XEUtils.each(insertRowMaps, (row, rowid) => {
|
|
if (fullAllDataRowIdData[rowid]) {
|
|
insertRecords.push(row);
|
|
}
|
|
});
|
|
return insertRecords;
|
|
},
|
|
/**
|
|
* 获取已删除的数据
|
|
*/
|
|
getRemoveRecords() {
|
|
const { removeRowMaps } = internalData;
|
|
const removeRecords = [];
|
|
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) => $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 = 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 Object.assign(Object.assign({}, editMethods), editPrivateMethods);
|
|
},
|
|
setupGrid($xeGrid) {
|
|
return $xeGrid.extendTableMethods(tableEditMethodKeys);
|
|
},
|
|
setupGantt($xeGantt) {
|
|
return $xeGantt.extendTableMethods(tableEditMethodKeys);
|
|
}
|
|
});
|