|
|
@ -2,7 +2,7 @@ import type { Ref } from 'vue' |
|
|
|
import { computed } from 'vue' |
|
|
|
import { computed } from 'vue' |
|
|
|
import dayjs from 'dayjs' |
|
|
|
import dayjs from 'dayjs' |
|
|
|
import type { MaybeRef } from '@vueuse/core' |
|
|
|
import type { MaybeRef } from '@vueuse/core' |
|
|
|
import type { ColumnType, LinkToAnotherRecordType, TableType, UserFieldRecordType } from 'nocodb-sdk' |
|
|
|
import type { ColumnType, LinkToAnotherRecordType, TableType, UserFieldRecordType, ViewType } from 'nocodb-sdk' |
|
|
|
import { RelationTypes, UITypes, dateFormats, isDateMonthFormat, isSystemColumn, isVirtualCol, timeFormats } from 'nocodb-sdk' |
|
|
|
import { RelationTypes, UITypes, dateFormats, isDateMonthFormat, isSystemColumn, isVirtualCol, timeFormats } from 'nocodb-sdk' |
|
|
|
import { parse } from 'papaparse' |
|
|
|
import { parse } from 'papaparse' |
|
|
|
import type { Cell } from './cellRange' |
|
|
|
import type { Cell } from './cellRange' |
|
|
@ -27,6 +27,7 @@ import { |
|
|
|
useGlobal, |
|
|
|
useGlobal, |
|
|
|
useI18n, |
|
|
|
useI18n, |
|
|
|
useMetas, |
|
|
|
useMetas, |
|
|
|
|
|
|
|
useUndoRedo, |
|
|
|
} from '#imports' |
|
|
|
} from '#imports' |
|
|
|
|
|
|
|
|
|
|
|
const MAIN_MOUSE_PRESSED = 0 |
|
|
|
const MAIN_MOUSE_PRESSED = 0 |
|
|
@ -49,6 +50,7 @@ export function useMultiSelect( |
|
|
|
syncCellData?: Function, |
|
|
|
syncCellData?: Function, |
|
|
|
bulkUpdateRows?: Function, |
|
|
|
bulkUpdateRows?: Function, |
|
|
|
fillHandle?: MaybeRef<HTMLElement | undefined>, |
|
|
|
fillHandle?: MaybeRef<HTMLElement | undefined>, |
|
|
|
|
|
|
|
view?: MaybeRef<ViewType | undefined>, |
|
|
|
) { |
|
|
|
) { |
|
|
|
const meta = ref(_meta) |
|
|
|
const meta = ref(_meta) |
|
|
|
|
|
|
|
|
|
|
@ -66,12 +68,16 @@ export function useMultiSelect( |
|
|
|
|
|
|
|
|
|
|
|
const { api } = useApi() |
|
|
|
const { api } = useApi() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const { addUndo, clone, defineViewScope } = useUndoRedo() |
|
|
|
|
|
|
|
|
|
|
|
const editEnabled = ref(_editEnabled) |
|
|
|
const editEnabled = ref(_editEnabled) |
|
|
|
|
|
|
|
|
|
|
|
const isMouseDown = ref(false) |
|
|
|
const isMouseDown = ref(false) |
|
|
|
|
|
|
|
|
|
|
|
const isFillMode = ref(false) |
|
|
|
const isFillMode = ref(false) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const activeView = ref(view) |
|
|
|
|
|
|
|
|
|
|
|
const selectedRange = reactive(new CellRange()) |
|
|
|
const selectedRange = reactive(new CellRange()) |
|
|
|
|
|
|
|
|
|
|
|
const fillRange = reactive(new CellRange()) |
|
|
|
const fillRange = reactive(new CellRange()) |
|
|
@ -226,7 +232,6 @@ export function useMultiSelect( |
|
|
|
) { |
|
|
|
) { |
|
|
|
return JSON.stringify({ |
|
|
|
return JSON.stringify({ |
|
|
|
rowId: extractPkFromRow(rowObj.row, meta.value?.columns as ColumnType[]), |
|
|
|
rowId: extractPkFromRow(rowObj.row, meta.value?.columns as ColumnType[]), |
|
|
|
columnId: columnObj.id, |
|
|
|
|
|
|
|
fk_related_model_id: (columnObj.colOptions as LinkToAnotherRecordType).fk_related_model_id, |
|
|
|
fk_related_model_id: (columnObj.colOptions as LinkToAnotherRecordType).fk_related_model_id, |
|
|
|
value: !isNaN(+textToCopy) ? +textToCopy : 0, |
|
|
|
value: !isNaN(+textToCopy) ? +textToCopy : 0, |
|
|
|
}) |
|
|
|
}) |
|
|
@ -866,7 +871,7 @@ export function useMultiSelect( |
|
|
|
(columnObj.colOptions as LinkToAnotherRecordType)?.type === RelationTypes.BELONGS_TO |
|
|
|
(columnObj.colOptions as LinkToAnotherRecordType)?.type === RelationTypes.BELONGS_TO |
|
|
|
) { |
|
|
|
) { |
|
|
|
const clipboardContext = JSON.parse(clipboardData!) |
|
|
|
const clipboardContext = JSON.parse(clipboardData!) |
|
|
|
let pasteVal = convertCellData( |
|
|
|
const pasteVal = convertCellData( |
|
|
|
{ |
|
|
|
{ |
|
|
|
value: clipboardContext, |
|
|
|
value: clipboardContext, |
|
|
|
to: columnObj.uidt as UITypes, |
|
|
|
to: columnObj.uidt as UITypes, |
|
|
@ -897,15 +902,9 @@ export function useMultiSelect( |
|
|
|
columnObj.uidt === UITypes.Links && |
|
|
|
columnObj.uidt === UITypes.Links && |
|
|
|
(columnObj.colOptions as LinkToAnotherRecordType)?.type === RelationTypes.MANY_TO_MANY |
|
|
|
(columnObj.colOptions as LinkToAnotherRecordType)?.type === RelationTypes.MANY_TO_MANY |
|
|
|
) { |
|
|
|
) { |
|
|
|
const clipboardContext = JSON.parse(clipboardData) |
|
|
|
const pasteVal = convertCellData( |
|
|
|
if (clipboardContext?.fk_related_model_id !== (columnObj.colOptions as LinkToAnotherRecordType).fk_related_model_id) { |
|
|
|
|
|
|
|
throw new Error('Invalid paste data for MM LTAR cell') |
|
|
|
|
|
|
|
return |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
let pasteVal = convertCellData( |
|
|
|
|
|
|
|
{ |
|
|
|
{ |
|
|
|
value: clipboardContext, |
|
|
|
value: clipboardData, |
|
|
|
to: columnObj.uidt as UITypes, |
|
|
|
to: columnObj.uidt as UITypes, |
|
|
|
column: columnObj, |
|
|
|
column: columnObj, |
|
|
|
appInfo: unref(appInfo), |
|
|
|
appInfo: unref(appInfo), |
|
|
@ -913,75 +912,95 @@ export function useMultiSelect( |
|
|
|
isMysql(meta.value?.source_id), |
|
|
|
isMysql(meta.value?.source_id), |
|
|
|
) |
|
|
|
) |
|
|
|
|
|
|
|
|
|
|
|
console.log('paste data', clipboardContext, pasteVal) |
|
|
|
if (pasteVal === undefined) return |
|
|
|
const relatedTableMeta = await getMeta((columnObj.colOptions as LinkToAnotherRecordType).fk_related_model_id!) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const extractedPk = extractPkFromRow(rowObj.row, meta.value?.columns as ColumnType[]) |
|
|
|
const pasteRowPk = extractPkFromRow(rowObj.row, meta.value?.columns as ColumnType[]) |
|
|
|
if (!extractedPk) return |
|
|
|
if (!pasteRowPk) return |
|
|
|
|
|
|
|
|
|
|
|
const [copiedCellchildrenList, pasteCellchildrenList] = await Promise.all([ |
|
|
|
const oldCellValue = rowObj.row[columnObj.title!] |
|
|
|
api.dbDataTableRow.nestedList( |
|
|
|
|
|
|
|
meta.value?.id as string, |
|
|
|
rowObj.row[columnObj.title!] = pasteVal.value |
|
|
|
columnObj.id as string, |
|
|
|
|
|
|
|
encodeURIComponent((clipboardContext.rowId as string) || ''), |
|
|
|
let result |
|
|
|
), |
|
|
|
|
|
|
|
api.dbDataTableRow.nestedList( |
|
|
|
try { |
|
|
|
meta.value?.id as string, |
|
|
|
result = await api.dbDataTableRow.nestedLinkUnlink(meta.value?.id as string, columnObj.id as string, [ |
|
|
|
columnObj.id as string, |
|
|
|
{ |
|
|
|
encodeURIComponent(extractPkFromRow(rowObj.row, meta.value?.columns as ColumnType[]) || ''), |
|
|
|
operation: 'copy', |
|
|
|
), |
|
|
|
rowId: pasteVal.rowId, |
|
|
|
]) |
|
|
|
fk_related_model_id: pasteVal.fk_related_model_id, |
|
|
|
|
|
|
|
}, |
|
|
|
const relatedTablePrimaryKeys = (extractPk(relatedTableMeta?.columns as ColumnType[]) || '').split('__') |
|
|
|
{ |
|
|
|
|
|
|
|
operation: 'paste', |
|
|
|
function filterAndMapRows( |
|
|
|
rowId: pasteRowPk, |
|
|
|
sourceList: Record<string, any>[], |
|
|
|
fk_related_model_id: |
|
|
|
targetList: Record<string, any>[], |
|
|
|
(columnObj.colOptions as LinkToAnotherRecordType).fk_related_model_id || pasteVal.fk_related_model_id, |
|
|
|
primaryKeys: string[] = relatedTablePrimaryKeys, |
|
|
|
}, |
|
|
|
): Record<string, any>[] { |
|
|
|
]) |
|
|
|
return sourceList |
|
|
|
} catch { |
|
|
|
.filter( |
|
|
|
rowObj.row[columnObj.title!] = oldCellValue |
|
|
|
(sourceRow: Record<string, any>) => |
|
|
|
} |
|
|
|
!targetList.some((targetRow: Record<string, any>) => |
|
|
|
|
|
|
|
primaryKeys.every((key) => sourceRow[key] === targetRow[key]), |
|
|
|
if (result && result?.link && result?.unlink && Array.isArray(result.link) && Array.isArray(result.unlink)) { |
|
|
|
), |
|
|
|
addUndo({ |
|
|
|
) |
|
|
|
redo: { |
|
|
|
.map((item: Record<string, any>) => |
|
|
|
fn: async ( |
|
|
|
primaryKeys.reduce((acc, key) => { |
|
|
|
tableId: string, |
|
|
|
acc[key] = item[key] |
|
|
|
columnId: string, |
|
|
|
return acc |
|
|
|
pasteRowPk: string, |
|
|
|
}, {} as Record<string, any>), |
|
|
|
result: { link: any[]; unlink: any[] }, |
|
|
|
) |
|
|
|
value: number, |
|
|
|
|
|
|
|
activeCell: Nullable<Cell>, |
|
|
|
|
|
|
|
) => { |
|
|
|
|
|
|
|
await Promise.all([ |
|
|
|
|
|
|
|
result.link.length && |
|
|
|
|
|
|
|
api.dbDataTableRow.nestedLink(tableId, columnId, encodeURIComponent(pasteRowPk), result.link), |
|
|
|
|
|
|
|
result.unlink.length && |
|
|
|
|
|
|
|
api.dbDataTableRow.nestedUnlink( |
|
|
|
|
|
|
|
meta.value?.id as string, |
|
|
|
|
|
|
|
columnObj.id as string, |
|
|
|
|
|
|
|
encodeURIComponent(pasteRowPk), |
|
|
|
|
|
|
|
result.unlink, |
|
|
|
|
|
|
|
), |
|
|
|
|
|
|
|
]) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rowObj.row[columnObj.title!] = value |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
await syncCellData?.(activeCell) |
|
|
|
|
|
|
|
}, |
|
|
|
|
|
|
|
args: [meta.value?.id as string, columnObj.id as string, pasteRowPk, result, pasteVal.value, clone(activeCell)], |
|
|
|
|
|
|
|
}, |
|
|
|
|
|
|
|
undo: { |
|
|
|
|
|
|
|
fn: async ( |
|
|
|
|
|
|
|
tableId: string, |
|
|
|
|
|
|
|
columnId: string, |
|
|
|
|
|
|
|
pasteRowPk: string, |
|
|
|
|
|
|
|
result: { link: any[]; unlink: any[] }, |
|
|
|
|
|
|
|
value: number, |
|
|
|
|
|
|
|
activeCell: Nullable<Cell>, |
|
|
|
|
|
|
|
) => { |
|
|
|
|
|
|
|
await Promise.all([ |
|
|
|
|
|
|
|
result.unlink.length && |
|
|
|
|
|
|
|
api.dbDataTableRow.nestedLink(tableId, columnId, encodeURIComponent(pasteRowPk), result.unlink), |
|
|
|
|
|
|
|
result.link.length && |
|
|
|
|
|
|
|
api.dbDataTableRow.nestedUnlink( |
|
|
|
|
|
|
|
meta.value?.id as string, |
|
|
|
|
|
|
|
columnObj.id as string, |
|
|
|
|
|
|
|
encodeURIComponent(pasteRowPk), |
|
|
|
|
|
|
|
result.link, |
|
|
|
|
|
|
|
), |
|
|
|
|
|
|
|
]) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rowObj.row[columnObj.title!] = value |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
await syncCellData?.(activeCell) |
|
|
|
|
|
|
|
}, |
|
|
|
|
|
|
|
args: [meta.value?.id as string, columnObj.id as string, pasteRowPk, result, oldCellValue, clone(activeCell)], |
|
|
|
|
|
|
|
}, |
|
|
|
|
|
|
|
scope: defineViewScope({ view: activeView?.value }), |
|
|
|
|
|
|
|
}) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
const filteredRowsToLink = filterAndMapRows(copiedCellchildrenList.list, pasteCellchildrenList.list) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const filteredRowsToUnlink = filterAndMapRows(pasteCellchildrenList.list, copiedCellchildrenList.list) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
console.log('copied cell child list', copiedCellchildrenList) |
|
|
|
|
|
|
|
console.log('paste cell child list', pasteCellchildrenList) |
|
|
|
|
|
|
|
console.log('filtered', filteredRowsToLink, filteredRowsToUnlink) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rowObj.row[columnObj.title!] = clipboardContext.value |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const result = await Promise.all([ |
|
|
|
|
|
|
|
filteredRowsToLink.length && |
|
|
|
|
|
|
|
api.dbDataTableRow.nestedLink( |
|
|
|
|
|
|
|
meta.value?.id as string, |
|
|
|
|
|
|
|
columnObj.id as string, |
|
|
|
|
|
|
|
encodeURIComponent(extractPkFromRow(rowObj.row, meta.value?.columns as ColumnType[]) || ''), |
|
|
|
|
|
|
|
filteredRowsToLink, |
|
|
|
|
|
|
|
), |
|
|
|
|
|
|
|
filteredRowsToUnlink.length && |
|
|
|
|
|
|
|
api.dbDataTableRow.nestedUnlink( |
|
|
|
|
|
|
|
meta.value?.id as string, |
|
|
|
|
|
|
|
columnObj.id as string, |
|
|
|
|
|
|
|
encodeURIComponent(extractPkFromRow(rowObj.row, meta.value?.columns as ColumnType[]) || ''), |
|
|
|
|
|
|
|
filteredRowsToUnlink, |
|
|
|
|
|
|
|
), |
|
|
|
|
|
|
|
]) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
console.log('result', result) |
|
|
|
|
|
|
|
return await syncCellData?.(activeCell) |
|
|
|
return await syncCellData?.(activeCell) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|