From b67966a9fb9f93df6e156bde6c879b618d225dec Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Fri, 27 Dec 2024 09:23:53 +0100 Subject: [PATCH 1/4] Initial typing --- .../stores/{clipboard.js => clipboard.ts} | 60 +++++++++++++------ .../src/components/grid/stores/index.ts | 6 +- .../src/components/grid/stores/ui.ts | 2 + 3 files changed, 49 insertions(+), 19 deletions(-) rename packages/frontend-core/src/components/grid/stores/{clipboard.js => clipboard.ts} (81%) diff --git a/packages/frontend-core/src/components/grid/stores/clipboard.js b/packages/frontend-core/src/components/grid/stores/clipboard.ts similarity index 81% rename from packages/frontend-core/src/components/grid/stores/clipboard.js rename to packages/frontend-core/src/components/grid/stores/clipboard.ts index 1cb75b38b1..a4d117f219 100644 --- a/packages/frontend-core/src/components/grid/stores/clipboard.js +++ b/packages/frontend-core/src/components/grid/stores/clipboard.ts @@ -1,9 +1,30 @@ -import { derived, writable, get } from "svelte/store" +import { derived, writable, get, Writable, Readable } from "svelte/store" import { Helpers } from "@budibase/bbui" import { parseCellID, getCellID } from "../lib/utils" import { NewRowID } from "../lib/constants" +import { Store as StoreContext } from "." -export const createStores = () => { +interface ClipboardStore { + clipboard: Writable<{ value: any; multiCellCopy: boolean }> +} + +interface ClipboardDerivedStore { + copyAllowed: Readable + pasteAllowed: Readable +} + +interface ClipboardActions { + clipboard: ClipboardStore["clipboard"] & { + actions: { + copy: any + paste: any + } + } +} + +export type Store = ClipboardStore & ClipboardDerivedStore & ClipboardActions + +export const createStores = (): ClipboardStore => { const clipboard = writable({ value: null, multiCellCopy: false, @@ -13,7 +34,7 @@ export const createStores = () => { } } -export const deriveStores = context => { +export const deriveStores = (context: StoreContext): ClipboardDerivedStore => { const { clipboard, focusedCellAPI, selectedCellCount, config, focusedRowId } = context @@ -60,7 +81,7 @@ export const deriveStores = context => { } } -export const createActions = context => { +export const createActions = (context: StoreContext): ClipboardActions => { const { clipboard, focusedCellAPI, @@ -92,11 +113,11 @@ export const createActions = context => { const $rowChangeCache = get(rowChangeCache) // Extract value of each selected cell, accounting for the change cache - let value = [] - for (let row of $selectedCells) { + const value = [] + for (const row of $selectedCells) { const rowValues = [] - for (let cellId of row) { - const { rowId, field } = parseCellID(cellId) + for (const cellId of row) { + const { rowId = "", field = "" } = parseCellID(cellId) const row = { ...$rowLookupMap[rowId], ...$rowChangeCache[rowId], @@ -113,7 +134,7 @@ export const createActions = context => { }) } else { // Single value to copy - const value = $focusedCellAPI.getValue() + const value = $focusedCellAPI?.getValue() clipboard.set({ value, multiCellCopy, @@ -130,7 +151,7 @@ export const createActions = context => { } // Pastes the previously copied value(s) into the selected cell(s) - const paste = async progressCallback => { + const paste = async (progressCallback: () => void) => { if (!get(pasteAllowed)) { return } @@ -166,8 +187,8 @@ export const createActions = context => { const { rowId, field } = parseCellID($focusedCellId) const $rowLookupMap = get(rowLookupMap) const $columnLookupMap = get(columnLookupMap) - const rowIdx = $rowLookupMap[rowId].__idx - const colIdx = $columnLookupMap[field].__idx + const rowIdx = $rowLookupMap[rowId!].__idx + const colIdx = $columnLookupMap[field!].__idx || 0 // Get limits of how many rows and columns we're able to paste into const $rows = get(rows) @@ -187,7 +208,7 @@ export const createActions = context => { // Paste into target cell range if (targetCellId === $focusedCellId) { // Single cell edge case - get(focusedCellAPI).setValue(value[0][0]) + get(focusedCellAPI)?.setValue(value[0][0]) } else { // Select the new cells to paste into, then paste selectedCells.actions.selectRange($focusedCellId, targetCellId) @@ -201,13 +222,16 @@ export const createActions = context => { await pasteIntoSelectedCells(newValue, progressCallback) } else { // Single to single - just update the cell's value - get(focusedCellAPI).setValue(value) + get(focusedCellAPI)?.setValue(value) } } } // Paste the specified value into the currently selected cells - const pasteIntoSelectedCells = async (value, progressCallback) => { + const pasteIntoSelectedCells = async ( + value: string[][], + progressCallback: () => any + ) => { const $selectedCells = get(selectedCells) // Find the extent at which we can paste @@ -215,11 +239,13 @@ export const createActions = context => { const colExtent = Math.min(value[0].length, $selectedCells[0].length) // Build change map - let changeMap = {} + let changeMap: Record> = {} for (let rowIdx = 0; rowIdx < rowExtent; rowIdx++) { for (let colIdx = 0; colIdx < colExtent; colIdx++) { const cellId = $selectedCells[rowIdx][colIdx] - const { rowId, field } = parseCellID(cellId) + let { rowId, field } = parseCellID(cellId) + rowId = rowId! + field = field! if (!changeMap[rowId]) { changeMap[rowId] = {} } diff --git a/packages/frontend-core/src/components/grid/stores/index.ts b/packages/frontend-core/src/components/grid/stores/index.ts index 81e0e824b0..b5676ca241 100644 --- a/packages/frontend-core/src/components/grid/stores/index.ts +++ b/packages/frontend-core/src/components/grid/stores/index.ts @@ -1,4 +1,4 @@ -import { Writable } from "svelte/store" +import { Readable, Writable } from "svelte/store" import type { APIClient } from "../../../api/types" import * as Bounds from "./bounds" @@ -65,7 +65,8 @@ export type Store = BaseStore & Users.Store & Menu.Store & Filter.Store & - UI.Store & { + UI.Store & + Clipboard.Store & { // TODO while typing the rest of stores fetch: Writable sort: Writable @@ -83,6 +84,7 @@ export type Store = BaseStore & rowLookupMap: Writable width: Writable fixedRowHeight: Writable + rowChangeCache: Readable } export const attachStores = (context: Store): Store => { diff --git a/packages/frontend-core/src/components/grid/stores/ui.ts b/packages/frontend-core/src/components/grid/stores/ui.ts index 23710668b7..7196723064 100644 --- a/packages/frontend-core/src/components/grid/stores/ui.ts +++ b/packages/frontend-core/src/components/grid/stores/ui.ts @@ -13,6 +13,8 @@ export interface UIStore { focusedCellId: Writable focusedCellAPI: Writable<{ isReadonly: () => boolean + getValue: () => string + setValue: (val: string) => void } | null> selectedRows: Writable> hoveredRowId: Writable From b600a2763d9b088dd48cb1756632acecd2127e53 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Fri, 27 Dec 2024 09:27:40 +0100 Subject: [PATCH 2/4] Type --- .../frontend-core/src/components/grid/stores/clipboard.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/frontend-core/src/components/grid/stores/clipboard.ts b/packages/frontend-core/src/components/grid/stores/clipboard.ts index a4d117f219..4dce0cf47a 100644 --- a/packages/frontend-core/src/components/grid/stores/clipboard.ts +++ b/packages/frontend-core/src/components/grid/stores/clipboard.ts @@ -16,8 +16,8 @@ interface ClipboardDerivedStore { interface ClipboardActions { clipboard: ClipboardStore["clipboard"] & { actions: { - copy: any - paste: any + copy: () => void + paste: (progressCallback: () => void) => Promise } } } From 296d5dfb55c698b335bed762e738a1a7e9f6d16c Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Fri, 27 Dec 2024 09:34:27 +0100 Subject: [PATCH 3/4] Type clipboard store --- .../src/components/grid/stores/clipboard.ts | 18 ++++++++++++++---- .../src/components/grid/stores/ui.ts | 2 +- 2 files changed, 15 insertions(+), 5 deletions(-) diff --git a/packages/frontend-core/src/components/grid/stores/clipboard.ts b/packages/frontend-core/src/components/grid/stores/clipboard.ts index 4dce0cf47a..57688e18a3 100644 --- a/packages/frontend-core/src/components/grid/stores/clipboard.ts +++ b/packages/frontend-core/src/components/grid/stores/clipboard.ts @@ -4,8 +4,18 @@ import { parseCellID, getCellID } from "../lib/utils" import { NewRowID } from "../lib/constants" import { Store as StoreContext } from "." +type ClipboardStoreData = + | { + value: string[][] + multiCellCopy: true + } + | { + value: string | null | undefined + multiCellCopy: false + } + interface ClipboardStore { - clipboard: Writable<{ value: any; multiCellCopy: boolean }> + clipboard: Writable } interface ClipboardDerivedStore { @@ -25,7 +35,7 @@ interface ClipboardActions { export type Store = ClipboardStore & ClipboardDerivedStore & ClipboardActions export const createStores = (): ClipboardStore => { - const clipboard = writable({ + const clipboard = writable({ value: null, multiCellCopy: false, }) @@ -218,11 +228,11 @@ export const createActions = (context: StoreContext): ClipboardActions => { } else { if (multiCellPaste) { // Single to multi - duplicate value to all selected cells - const newValue = get(selectedCells).map(row => row.map(() => value)) + const newValue = get(selectedCells).map(row => row.map(() => value!)) await pasteIntoSelectedCells(newValue, progressCallback) } else { // Single to single - just update the cell's value - get(focusedCellAPI)?.setValue(value) + get(focusedCellAPI)?.setValue(value ?? null) } } } diff --git a/packages/frontend-core/src/components/grid/stores/ui.ts b/packages/frontend-core/src/components/grid/stores/ui.ts index 7196723064..8988137e0b 100644 --- a/packages/frontend-core/src/components/grid/stores/ui.ts +++ b/packages/frontend-core/src/components/grid/stores/ui.ts @@ -14,7 +14,7 @@ export interface UIStore { focusedCellAPI: Writable<{ isReadonly: () => boolean getValue: () => string - setValue: (val: string) => void + setValue: (val: string | null) => void } | null> selectedRows: Writable> hoveredRowId: Writable From cab40c8da1b1fffd7a5363be913a0c396f934bd6 Mon Sep 17 00:00:00 2001 From: Adria Navarro Date: Mon, 30 Dec 2024 22:21:23 +0100 Subject: [PATCH 4/4] Update clipboard values to any --- .../frontend-core/src/components/grid/stores/clipboard.ts | 4 ++-- packages/frontend-core/src/components/grid/stores/ui.ts | 4 ++-- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/packages/frontend-core/src/components/grid/stores/clipboard.ts b/packages/frontend-core/src/components/grid/stores/clipboard.ts index 57688e18a3..3489e5c847 100644 --- a/packages/frontend-core/src/components/grid/stores/clipboard.ts +++ b/packages/frontend-core/src/components/grid/stores/clipboard.ts @@ -6,11 +6,11 @@ import { Store as StoreContext } from "." type ClipboardStoreData = | { - value: string[][] + value: any[][] multiCellCopy: true } | { - value: string | null | undefined + value: any | null | undefined multiCellCopy: false } diff --git a/packages/frontend-core/src/components/grid/stores/ui.ts b/packages/frontend-core/src/components/grid/stores/ui.ts index 8988137e0b..7d6c6dcac9 100644 --- a/packages/frontend-core/src/components/grid/stores/ui.ts +++ b/packages/frontend-core/src/components/grid/stores/ui.ts @@ -13,8 +13,8 @@ export interface UIStore { focusedCellId: Writable focusedCellAPI: Writable<{ isReadonly: () => boolean - getValue: () => string - setValue: (val: string | null) => void + getValue: () => any + setValue: (val: any) => void } | null> selectedRows: Writable> hoveredRowId: Writable