import { nextTick } from 'vue' import { VxeUI } from '../../../ui' import XEUtils from 'xe-utils' import { getColumnList } from '../../src/util' import type { TableCustomMethods, TableCustomPrivateMethods, VxeColumnPropTypes, VxeTableDefines } from '../../../../types' const tableCustomMethodKeys: (keyof TableCustomMethods)[] = ['openCustom', 'closeCustom', 'toggleCustom', 'saveCustom', 'cancelCustom', 'resetCustom', 'toggleCustomAllCheckbox', 'setCustomAllCheckbox'] VxeUI.hooks.add('tableCustomModule', { setupTable ($xeTable) { const { reactData, internalData } = $xeTable const { computeCustomOpts, computeRowGroupFields } = $xeTable.getComputeMaps() const { refElem } = $xeTable.getRefMaps() const $xeGrid = $xeTable.xeGrid const $xeGantt = $xeTable.xeGantt const calcMaxHeight = () => { const { customStore } = reactData let wrapperEl = refElem.value // 判断面板不能大于表格高度 let tableHeight = 0 if ($xeGantt) { const { refGanttContainerElem } = $xeGantt.getRefMaps() const ganttContainerElem = refGanttContainerElem.value if (ganttContainerElem) { wrapperEl = ganttContainerElem } } if (wrapperEl) { tableHeight = wrapperEl.clientHeight - 28 } customStore.maxHeight = Math.max(88, tableHeight) } const openCustom = () => { const { initStore, customStore } = reactData customStore.visible = true initStore.custom = true handleUpdateCustomColumn() checkCustomStatus() calcMaxHeight() return nextTick().then(() => calcMaxHeight()) } const handleUpdateCustomColumn = () => { const { customStore } = reactData const { collectColumn } = internalData if (customStore.visible) { const sortMaps: Record = {} const fixedMaps: Record = {} const visibleMaps: Record = {} XEUtils.eachTree(collectColumn, column => { const colid = column.getKey() column.renderFixed = column.fixed column.renderVisible = column.visible column.renderResizeWidth = column.renderWidth sortMaps[colid] = column.renderSortNumber fixedMaps[colid] = column.fixed visibleMaps[colid] = column.visible }) customStore.oldSortMaps = sortMaps customStore.oldFixedMaps = fixedMaps customStore.oldVisibleMaps = visibleMaps reactData.customColumnList = collectColumn.slice(0) } } const closeCustom = () => { const { customStore } = reactData const customOpts = computeCustomOpts.value if (customStore.visible) { customStore.visible = false if (!customOpts.immediate) { $xeTable.handleCustom() } } return nextTick() } const toggleCustom = () => { const { customStore } = reactData if (customStore.visible) { return closeCustom() } return openCustom() } const saveCustom = () => { const { customColumnList, aggHandleFields, rowGroupList } = reactData const customOpts = computeCustomOpts.value const { allowVisible, allowSort, allowFixed, allowResizable, allowGroup, allowValues } = customOpts XEUtils.eachTree(customColumnList, (column, index, items, path, parentColumn) => { if (parentColumn) { // 更新子列信息 column.fixed = parentColumn.fixed } else { if (allowSort) { const sortIndex = index + 1 column.renderSortNumber = sortIndex } if (allowFixed) { column.fixed = column.renderFixed } } if (allowResizable) { if (column.renderVisible && (!column.children || column.children.length)) { if (column.renderResizeWidth !== column.renderWidth) { column.resizeWidth = column.renderResizeWidth column.renderWidth = column.renderResizeWidth } } } if (allowVisible) { column.visible = column.renderVisible } if (allowGroup && allowValues) { column.aggFunc = column.renderAggFn } }) reactData.isCustomStatus = true if (allowGroup && !!$xeTable.handlePivotTableAggregateData) { if (rowGroupList.length !== aggHandleFields.length || rowGroupList.some((conf, i) => conf.field !== aggHandleFields[i])) { // 更新数据分组 if (aggHandleFields.length) { $xeTable.setRowGroups(aggHandleFields) } else { $xeTable.clearRowGroups() } } else if (allowValues) { // 更新聚合函数 $xeTable.handleUpdateAggData() } } if (allowSort) { internalData.collectColumn = customColumnList } return $xeTable.saveCustomStore('confirm') } const cancelCustom = () => { const { customColumnList, customStore } = reactData const { oldSortMaps, oldFixedMaps, oldVisibleMaps } = customStore const customOpts = computeCustomOpts.value const { allowVisible, allowSort, allowFixed, allowResizable } = customOpts XEUtils.eachTree(customColumnList, column => { const colid = column.getKey() const visible = !!oldVisibleMaps[colid] const fixed = oldFixedMaps[colid] || '' if (allowVisible) { column.renderVisible = visible column.visible = visible } if (allowFixed) { column.renderFixed = fixed column.fixed = fixed } if (allowSort) { column.renderSortNumber = oldSortMaps[colid] || 0 } if (allowResizable) { column.renderResizeWidth = column.renderWidth } }, { children: 'children' }) return nextTick() } const setCustomAllCheckbox = (checked: boolean) => { const { customStore } = reactData const { customColumnList } = reactData const customOpts = computeCustomOpts.value const { checkMethod, visibleMethod } = customOpts const isAll = !!checked if (customOpts.immediate) { XEUtils.eachTree(customColumnList, (column) => { if (visibleMethod && !visibleMethod({ $table: $xeTable, column })) { return } if (checkMethod && !checkMethod({ $table: $xeTable, column })) { return } column.visible = isAll column.renderVisible = isAll column.halfVisible = false }) customStore.isAll = isAll reactData.isCustomStatus = true $xeTable.handleCustom() $xeTable.saveCustomStore('update:visible') } else { XEUtils.eachTree(customColumnList, (column) => { if (visibleMethod && !visibleMethod({ $table: $xeTable, column })) { return } if (checkMethod && !checkMethod({ $table: $xeTable, column })) { return } column.renderVisible = isAll column.halfVisible = false }) customStore.isAll = isAll } $xeTable.checkCustomStatus() return nextTick() } const customMethods: TableCustomMethods = { getCustomVisible () { const { customStore } = reactData return customStore.visible }, openCustom, closeCustom, toggleCustom, saveCustom, cancelCustom, resetCustom (options) { const { rowGroupList } = reactData const { collectColumn } = internalData const customOpts = computeCustomOpts.value const { checkMethod } = customOpts const opts: VxeTableDefines.VxeTableCustomStorageObj = Object.assign({ visible: true, resizable: options === true, fixed: options === true, sort: options === true, aggFunc: options === true }, options) const allCols: VxeTableDefines.ColumnInfo[] = [] XEUtils.eachTree(collectColumn, (column) => { if (opts.resizable) { column.resizeWidth = 0 } if (opts.fixed) { column.fixed = column.defaultFixed } if (opts.sort) { column.renderSortNumber = column.sortNumber column.parentId = column.defaultParentId } if (!checkMethod || checkMethod({ $table: $xeTable, column })) { column.visible = column.defaultVisible } if (opts.aggFunc) { column.aggFunc = column.defaultAggFunc column.renderAggFn = column.defaultAggFunc } column.renderResizeWidth = column.renderWidth allCols.push(column) }) if (opts.sort) { const newCollectCols = XEUtils.toArrayTree(XEUtils.orderBy(allCols, 'renderSortNumber'), { key: 'id', parentKey: 'parentId', children: 'children' }) internalData.collectColumn = newCollectCols internalData.tableFullColumn = getColumnList(newCollectCols) } reactData.isCustomStatus = false return $xeTable.handleCustom().then(() => { if (opts.aggFunc && ($xeTable as any).handlePivotTableAggregateData) { const rowGroupFields = computeRowGroupFields.value if (rowGroupFields ? rowGroupFields.length : rowGroupList.length) { if (rowGroupFields && rowGroupFields.length) { $xeTable.setRowGroups(rowGroupFields) } else { $xeTable.clearRowGroups() } } else { $xeTable.handleUpdateAggData() } } $xeTable.saveCustomStore('reset') }) }, toggleCustomAllCheckbox () { const { customStore } = reactData const isAll = !customStore.isAll return setCustomAllCheckbox(isAll) }, setCustomAllCheckbox } const checkCustomStatus = () => { const { customStore } = reactData const { collectColumn } = internalData const customOpts = computeCustomOpts.value const { checkMethod } = customOpts customStore.isAll = collectColumn.every((column) => (checkMethod ? !checkMethod({ $table: $xeTable, column }) : false) || column.renderVisible) customStore.isIndeterminate = !customStore.isAll && collectColumn.some((column) => (!checkMethod || checkMethod({ $table: $xeTable, column })) && (column.renderVisible || column.halfVisible)) } const emitCustomEvent = (type: VxeTableDefines.CustomType, evnt: Event) => { const comp = $xeGrid || $xeGantt || $xeTable comp.dispatchEvent('custom', { type }, evnt) } const customPrivateMethods: TableCustomPrivateMethods = { checkCustomStatus, emitCustomEvent, triggerCustomEvent (evnt) { const reactData = $xeTable.reactData const { customStore } = reactData if (customStore.visible) { closeCustom() emitCustomEvent('close', evnt) } else { customStore.btnEl = evnt.target as HTMLDivElement openCustom() emitCustomEvent('open', evnt) } }, customOpenEvent (evnt) { const reactData = $xeTable.reactData const { customStore } = reactData if (!customStore.visible) { customStore.activeBtn = true customStore.btnEl = evnt.target as HTMLDivElement $xeTable.openCustom() $xeTable.emitCustomEvent('open', evnt) } }, customCloseEvent (evnt) { const reactData = $xeTable.reactData const { customStore } = reactData if (customStore.visible) { customStore.activeBtn = false $xeTable.closeCustom() $xeTable.emitCustomEvent('close', evnt) } }, handleUpdateCustomColumn } return { ...customMethods, ...customPrivateMethods } }, setupGrid ($xeGrid) { return $xeGrid.extendTableMethods(tableCustomMethodKeys) }, setupGantt ($xeGantt) { return $xeGantt.extendTableMethods(tableCustomMethodKeys) } })