Refactor stores to fix dependency issues, use modals for adding rows, simplify sheet

This commit is contained in:
Andrew Kingston 2023-04-11 12:17:08 +01:00
parent 81a28eb4da
commit fe70793e75
26 changed files with 533 additions and 490 deletions

View File

@ -4,7 +4,7 @@
import { getCellRenderer } from "../lib/renderers" import { getCellRenderer } from "../lib/renderers"
import { derived, writable } from "svelte/store" import { derived, writable } from "svelte/store"
const { rows, focusedCellId, menu, sheetAPI, config, validation } = const { rows, focusedCellId, focusedCellAPI, menu, config, validation } =
getContext("sheet") getContext("sheet")
export let rowSelected export let rowSelected
@ -25,13 +25,6 @@
let api let api
$: {
// Wipe error if row is unfocused
if (!rowFocused && $error) {
validation.actions.setError(cellId, null)
}
}
// Get the error for this cell if the row is focused // Get the error for this cell if the row is focused
$: error = getErrorStore(rowFocused, cellId) $: error = getErrorStore(rowFocused, cellId)
@ -40,8 +33,8 @@
// Register this cell API if the row is focused // Register this cell API if the row is focused
$: { $: {
if (rowFocused) { if (focused) {
sheetAPI.actions.registerCellAPI(cellId, cellAPI) focusedCellAPI.set(cellAPI)
} }
} }
@ -57,30 +50,9 @@
blur: () => api?.blur(), blur: () => api?.blur(),
onKeyDown: (...params) => api?.onKeyDown(...params), onKeyDown: (...params) => api?.onKeyDown(...params),
isReadonly: () => readonly, isReadonly: () => readonly,
isRequired: () => !!column.schema.constraints?.presence,
validate: value => {
// Validate the current value if no new value is provided
if (value === undefined) {
value = row[column.name]
}
let newError = null
if (cellAPI.isReadonly() && !(value == null || value === "")) {
// Ensure cell isn't readonly
newError = "Auto columns can't be edited"
} else if (cellAPI.isRequired() && (value == null || value === "")) {
// Sanity check required fields
newError = "Required field"
} else {
newError = null
}
validation.actions.setError(cellId, newError)
return newError
},
updateValue: value => { updateValue: value => {
cellAPI.validate(value) validation.actions.setError(cellId, null)
if (!$error) {
updateRow(row._id, column.name, value) updateRow(row._id, column.name, value)
}
}, },
} }
</script> </script>

View File

@ -0,0 +1,16 @@
<script>
import { ActionButton } from "@budibase/bbui"
import { getContext } from "svelte"
const { config, dispatch } = getContext("sheet")
</script>
<ActionButton
icon="TableColumnAddRight"
quiet
size="M"
on:click={() => dispatch("add-column")}
disabled={!$config.allowAddColumns}
>
Create column
</ActionButton>

View File

@ -2,15 +2,15 @@
import { ActionButton } from "@budibase/bbui" import { ActionButton } from "@budibase/bbui"
import { getContext } from "svelte" import { getContext } from "svelte"
const { dispatch, columns, stickyColumn } = getContext("sheet") const { dispatch, columns, stickyColumn, config } = getContext("sheet")
</script> </script>
<ActionButton <ActionButton
icon="Add" icon="TableRowAddBottom"
quiet quiet
size="M" size="M"
on:click={() => dispatch("add-row-inline")} on:click={() => dispatch("add-row")}
disabled={!$columns.length && !$stickyColumn} disabled={!$config.allowAddRows || (!$columns.length && !$stickyColumn)}
> >
Create row Create row
</ActionButton> </ActionButton>

View File

@ -2,14 +2,8 @@
import { getContext } from "svelte" import { getContext } from "svelte"
import SheetScrollWrapper from "./SheetScrollWrapper.svelte" import SheetScrollWrapper from "./SheetScrollWrapper.svelte"
import HeaderCell from "../cells/HeaderCell.svelte" import HeaderCell from "../cells/HeaderCell.svelte"
import { Icon } from "@budibase/bbui"
const { renderedColumns, dispatch, config, ui } = getContext("sheet") const { renderedColumns } = getContext("sheet")
const addColumn = () => {
ui.actions.blur()
dispatch("add-column")
}
</script> </script>
<div class="header"> <div class="header">
@ -20,11 +14,6 @@
{/each} {/each}
</div> </div>
</SheetScrollWrapper> </SheetScrollWrapper>
{#if $config.allowAddColumns}
<div class="new-column" on:click={addColumn}>
<Icon size="S" name="Add" />
</div>
{/if}
</div> </div>
<style> <style>
@ -38,20 +27,4 @@
.row { .row {
display: flex; display: flex;
} }
.new-column {
position: absolute;
right: 0;
top: 0;
height: var(--row-height);
background: var(--spectrum-global-color-gray-100);
display: grid;
place-items: center;
width: 46px;
border-left: var(--cell-border);
border-bottom: var(--cell-border);
}
.new-column:hover {
cursor: pointer;
background: var(--spectrum-global-color-gray-200);
}
</style> </style>

View File

@ -1,7 +1,7 @@
<script> <script>
import SheetCell from "../cells/SheetCell.svelte" import SheetCell from "../cells/SheetCell.svelte"
import { getContext, onMount, tick } from "svelte" import { getContext, onMount } from "svelte"
import { Icon, Button } from "@budibase/bbui" import { Icon, Button, clickOutside } from "@budibase/bbui"
import SheetScrollWrapper from "./SheetScrollWrapper.svelte" import SheetScrollWrapper from "./SheetScrollWrapper.svelte"
import DataCell from "../cells/DataCell.svelte" import DataCell from "../cells/DataCell.svelte"
import { fly } from "svelte/transition" import { fly } from "svelte/transition"
@ -19,7 +19,7 @@
showHScrollbar, showHScrollbar,
tableId, tableId,
subscribe, subscribe,
sheetAPI, scrollLeft,
} = getContext("sheet") } = getContext("sheet")
let isAdding = false let isAdding = false
@ -30,23 +30,25 @@
$: rowHovered = $hoveredRowId === "new" $: rowHovered = $hoveredRowId === "new"
$: rowFocused = $focusedCellId?.startsWith("new-") $: rowFocused = $focusedCellId?.startsWith("new-")
$: width = gutterWidth + ($stickyColumn?.width || 0) $: width = gutterWidth + ($stickyColumn?.width || 0)
$: scrollLeft = $scroll.left
$: $tableId, (isAdding = false) $: $tableId, (isAdding = false)
const addRow = async () => { const addRow = async () => {
// Create row // Create row
const savedRow = await rows.actions.addRow(newRow, 0) const savedRow = await rows.actions.addRow(newRow, 0)
if (savedRow && firstColumn) { if (savedRow) {
// Select the first cell if possible
if (firstColumn) {
$focusedCellId = `${savedRow._id}-${firstColumn.name}` $focusedCellId = `${savedRow._id}-${firstColumn.name}`
isAdding = false
} }
// Reset scroll // Reset state
isAdding = false
scroll.set({ scroll.set({
left: 0, left: 0,
top: 0, top: 0,
}) })
} }
}
const cancel = () => { const cancel = () => {
isAdding = false isAdding = false
@ -151,17 +153,28 @@
position: absolute; position: absolute;
top: var(--row-height); top: var(--row-height);
left: 0; left: 0;
background: linear-gradient( /*background: linear-gradient(*/
to bottom, /* to bottom,*/
var(--cell-background) 20%, /* var(--cell-background) 20%,*/
transparent 100% /* transparent 100%*/
); /*);*/
width: 100%; width: 100%;
padding-bottom: 100px; padding-bottom: 800px;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
align-items: stretch; align-items: stretch;
} }
.container:before {
position: absolute;
content: "";
left: 0;
top: 0;
height: 100%;
width: 100%;
background: var(--cell-background);
opacity: 0.8;
z-index: -1;
}
.content { .content {
pointer-events: all; pointer-events: all;
background: var(--background); background: var(--background);
@ -227,7 +240,7 @@
display: flex; display: flex;
flex-direction: row; flex-direction: row;
gap: 8px; gap: 8px;
margin: 24px 0 0 16px; margin: 24px 0 0 var(--gutter-width);
pointer-events: all; pointer-events: all;
align-self: flex-start; align-self: flex-start;
} }

View File

@ -3,20 +3,7 @@
import { writable } from "svelte/store" import { writable } from "svelte/store"
import { createEventManagers } from "../lib/events" import { createEventManagers } from "../lib/events"
import { createAPIClient } from "../../../api" import { createAPIClient } from "../../../api"
import { createReorderStores } from "../stores/reorder" import { createStores } from "../stores"
import { createViewportStores } from "../stores/viewport"
import { createRowsStore } from "../stores/rows"
import { createColumnsStores } from "../stores/columns"
import { createScrollStores } from "../stores/scroll"
import { createBoundsStores } from "../stores/bounds"
import { createUIStores } from "../stores/ui"
import { createUserStores } from "../stores/users"
import { createResizeStores } from "../stores/resize"
import { createMenuStores } from "../stores/menu"
import { createMaxScrollStores } from "../stores/max-scroll"
import { createPaginationStores } from "../stores/pagination"
import { createSheetAPIStores } from "../stores/sheet-api"
import { createValidationStores } from "../stores/validation"
import DeleteButton from "../controls/DeleteButton.svelte" import DeleteButton from "../controls/DeleteButton.svelte"
import SheetBody from "./SheetBody.svelte" import SheetBody from "./SheetBody.svelte"
import ResizeOverlay from "../overlays/ResizeOverlay.svelte" import ResizeOverlay from "../overlays/ResizeOverlay.svelte"
@ -28,8 +15,11 @@
import KeyboardManager from "../overlays/KeyboardManager.svelte" import KeyboardManager from "../overlays/KeyboardManager.svelte"
import { clickOutside } from "@budibase/bbui" import { clickOutside } from "@budibase/bbui"
import SheetControls from "./SheetControls.svelte" import SheetControls from "./SheetControls.svelte"
import NewRowTop from "./NewRowTop.svelte"
import { MaxCellRenderHeight } from "../lib/constants" import { MaxCellRenderHeight } from "../lib/constants"
import SortButton from "../controls/SortButton.svelte"
import AddColumnButton from "../controls/AddColumnButton.svelte"
import HideColumnsButton from "../controls/HideColumnsButton.svelte"
import AddRowButton from "../controls/AddRowButton.svelte"
export let API export let API
export let tableId export let tableId
@ -56,7 +46,6 @@
}) })
// Build up spreadsheet context // Build up spreadsheet context
// Stores are listed in order of dependency on each other
let context = { let context = {
API: API || createAPIClient(), API: API || createAPIClient(),
rand, rand,
@ -65,20 +54,7 @@
tableId: tableIdStore, tableId: tableIdStore,
} }
context = { ...context, ...createEventManagers() } context = { ...context, ...createEventManagers() }
context = { ...context, ...createValidationStores(context) } context = { ...context, ...createStores(context) }
context = { ...context, ...createBoundsStores(context) }
context = { ...context, ...createScrollStores(context) }
context = { ...context, ...createRowsStore(context) }
context = { ...context, ...createColumnsStores(context) }
context = { ...context, ...createUIStores(context) }
context = { ...context, ...createSheetAPIStores(context) }
context = { ...context, ...createResizeStores(context) }
context = { ...context, ...createViewportStores(context) }
context = { ...context, ...createMaxScrollStores(context) }
context = { ...context, ...createReorderStores(context) }
context = { ...context, ...createUserStores(context) }
context = { ...context, ...createMenuStores(context) }
context = { ...context, ...createPaginationStores(context) }
// Reference some stores for local use // Reference some stores for local use
const { isResizing, isReordering, ui, loaded, rowHeight } = context const { isResizing, isReordering, ui, loaded, rowHeight } = context
@ -113,8 +89,12 @@
> >
<div class="controls"> <div class="controls">
<div class="controls-left"> <div class="controls-left">
<AddRowButton />
<AddColumnButton />
<SheetControls /> <SheetControls />
<slot name="controls" /> <slot name="controls" />
<HideColumnsButton />
<SortButton />
</div> </div>
<div class="controls-right"> <div class="controls-right">
<DeleteButton /> <DeleteButton />
@ -130,9 +110,6 @@
<SheetBody /> <SheetBody />
</div> </div>
<div class="overlays"> <div class="overlays">
{#if $config.allowAddRows}
<NewRowTop />
{/if}
<ResizeOverlay /> <ResizeOverlay />
<ScrollOverlay /> <ScrollOverlay />
<MenuOverlay /> <MenuOverlay />

View File

@ -2,10 +2,5 @@
import SortButton from "../controls/SortButton.svelte" import SortButton from "../controls/SortButton.svelte"
import HideColumnsButton from "../controls/HideColumnsButton.svelte" import HideColumnsButton from "../controls/HideColumnsButton.svelte"
import AddRowButton from "../controls/AddRowButton.svelte" import AddRowButton from "../controls/AddRowButton.svelte"
import RowHeightButton from "../controls/RowHeightButton.svelte" import AddColumnButton from "../controls/AddColumnButton.svelte"
</script> </script>
<AddRowButton />
<HideColumnsButton />
<SortButton />
<RowHeightButton />

View File

@ -20,9 +20,9 @@
focusedRow, focusedRow,
gutterWidth, gutterWidth,
dispatch, dispatch,
scrollLeft,
} = getContext("sheet") } = getContext("sheet")
$: scrollLeft = $scroll.left
$: rowCount = $rows.length $: rowCount = $rows.length
$: selectedRowCount = Object.values($selectedRows).filter(x => !!x).length $: selectedRowCount = Object.values($selectedRows).filter(x => !!x).length
$: width = gutterWidth + ($stickyColumn?.width || 0) $: width = gutterWidth + ($stickyColumn?.width || 0)
@ -91,6 +91,7 @@
{@const rowSelected = !!$selectedRows[row._id]} {@const rowSelected = !!$selectedRows[row._id]}
{@const rowHovered = $hoveredRowId === row._id} {@const rowHovered = $hoveredRowId === row._id}
{@const rowFocused = $focusedRow?._id === row._id} {@const rowFocused = $focusedRow?._id === row._id}
{@const cellId = `${row._id}-${$stickyColumn.name}`}
<div <div
class="row" class="row"
on:mouseenter={() => ($hoveredRowId = row._id)} on:mouseenter={() => ($hoveredRowId = row._id)}
@ -132,9 +133,7 @@
{/if} {/if}
</div> </div>
</SheetCell> </SheetCell>
{#if $stickyColumn} {#if $stickyColumn}
{@const cellId = `${row._id}-${$stickyColumn.name}`}
<DataCell <DataCell
{rowSelected} {rowSelected}
{rowHovered} {rowHovered}
@ -176,7 +175,7 @@
/*z-index: 1;*/ /*z-index: 1;*/
} }
.sticky-column.scrolled { .sticky-column.scrolled {
/*box-shadow: 0 0 10px 2px rgba(0, 0, 0, 0.1);*/ box-shadow: 0 0 10px 2px rgba(0, 0, 0, 0.1);
} }
/* Don't show borders between cells in the sticky column */ /* Don't show borders between cells in the sticky column */

View File

@ -1,7 +1,6 @@
<script> <script>
import { getContext, onMount } from "svelte" import { getContext, onMount } from "svelte"
import { debounce } from "../../../utils/utils" import { debounce } from "../../../utils/utils"
import { notifications } from "@budibase/bbui"
const { const {
rows, rows,
@ -9,7 +8,7 @@
visibleColumns, visibleColumns,
focusedRow, focusedRow,
stickyColumn, stickyColumn,
selectedCellAPI, focusedCellAPI,
} = getContext("sheet") } = getContext("sheet")
const handleKeyDown = e => { const handleKeyDown = e => {
@ -22,7 +21,7 @@
} }
// Always intercept certain key presses // Always intercept certain key presses
const api = $selectedCellAPI const api = $focusedCellAPI
if (e.key === "Escape") { if (e.key === "Escape") {
api?.blur?.() api?.blur?.()
} else if (e.key === "Tab") { } else if (e.key === "Tab") {
@ -112,17 +111,17 @@
// Debounce to avoid holding down delete and spamming requests // Debounce to avoid holding down delete and spamming requests
const deleteSelectedCell = debounce(() => { const deleteSelectedCell = debounce(() => {
if (!$focusedCellId) { if ($focusedCellAPI?.isReadonly()) {
return return
} }
$selectedCellAPI.updateValue(null) $focusedCellAPI.updateValue(null)
}, 100) }, 100)
const focusSelectedCell = () => { const focusSelectedCell = () => {
if ($selectedCellAPI?.isReadonly()) { if ($focusedCellAPI?.isReadonly()) {
return return
} }
$selectedCellAPI?.focus?.() $focusedCellAPI?.focus?.()
} }
onMount(() => { onMount(() => {

View File

@ -9,9 +9,9 @@
stickyColumn, stickyColumn,
isReordering, isReordering,
gutterWidth, gutterWidth,
scrollLeft,
} = getContext("sheet") } = getContext("sheet")
$: scrollLeft = $scroll.left
$: cutoff = scrollLeft + gutterWidth + ($columns[0]?.width || 0) $: cutoff = scrollLeft + gutterWidth + ($columns[0]?.width || 0)
$: offset = gutterWidth + ($stickyColumn?.width || 0) $: offset = gutterWidth + ($stickyColumn?.width || 0)
$: column = $resize.column $: column = $resize.column

View File

@ -4,7 +4,6 @@
const { const {
scroll, scroll,
bounds,
rowHeight, rowHeight,
contentHeight, contentHeight,
maxScrollTop, maxScrollTop,
@ -13,6 +12,9 @@
screenWidth, screenWidth,
showHScrollbar, showHScrollbar,
showVScrollbar, showVScrollbar,
scrollLeft,
scrollTop,
height,
} = getContext("sheet") } = getContext("sheet")
// Bar config // Bar config
@ -22,32 +24,27 @@
let initialMouse let initialMouse
let initialScroll let initialScroll
// Memoize store primitives to reduce reactive statement invalidations
$: scrollTop = $scroll.top
$: scrollLeft = $scroll.left
$: height = $bounds.height
$: width = $bounds.width
// Calculate V scrollbar size and offset // Calculate V scrollbar size and offset
// Terminology is the same for both axes: // Terminology is the same for both axes:
// renderX - the space available to render the bar in, edge to edge // renderX - the space available to render the bar in, edge to edge
// availX - the space available to render the bar in, until the edge // availX - the space available to render the bar in, until the edge
$: renderHeight = height - 2 * barOffset $: renderHeight = $height - 2 * barOffset
$: barHeight = Math.max(50, (height / $contentHeight) * renderHeight) $: barHeight = Math.max(50, ($height / $contentHeight) * renderHeight)
$: availHeight = renderHeight - barHeight $: availHeight = renderHeight - barHeight
$: barTop = barOffset + $rowHeight + availHeight * (scrollTop / $maxScrollTop) $: barTop =
barOffset + $rowHeight + availHeight * ($scrollTop / $maxScrollTop)
// Calculate H scrollbar size and offset // Calculate H scrollbar size and offset
$: renderWidth = $screenWidth - 2 * barOffset $: renderWidth = $screenWidth - 2 * barOffset
$: barWidth = Math.max(50, ($screenWidth / $contentWidth) * renderWidth) $: barWidth = Math.max(50, ($screenWidth / $contentWidth) * renderWidth)
$: availWidth = renderWidth - barWidth $: availWidth = renderWidth - barWidth
$: barLeft = barOffset + availWidth * (scrollLeft / $maxScrollLeft) $: barLeft = barOffset + availWidth * ($scrollLeft / $maxScrollLeft)
// V scrollbar drag handlers // V scrollbar drag handlers
const startVDragging = e => { const startVDragging = e => {
e.preventDefault() e.preventDefault()
initialMouse = e.clientY initialMouse = e.clientY
initialScroll = scrollTop initialScroll = $scrollTop
document.addEventListener("mousemove", moveVDragging) document.addEventListener("mousemove", moveVDragging)
document.addEventListener("mouseup", stopVDragging) document.addEventListener("mouseup", stopVDragging)
} }
@ -69,7 +66,7 @@
const startHDragging = e => { const startHDragging = e => {
e.preventDefault() e.preventDefault()
initialMouse = e.clientX initialMouse = e.clientX
initialScroll = scrollLeft initialScroll = $scrollLeft
document.addEventListener("mousemove", moveHDragging) document.addEventListener("mousemove", moveHDragging)
document.addEventListener("mouseup", stopHDragging) document.addEventListener("mouseup", stopHDragging)
} }

View File

@ -1,11 +1,16 @@
import { writable } from "svelte/store" import { derived, writable } from "svelte/store"
export const createBoundsStores = () => { export const createStores = () => {
const bounds = writable({ const bounds = writable({
left: 0, left: 0,
top: 0, top: 0,
width: 0, width: 0,
height: 0, height: 0,
}) })
return { bounds }
// Derive height and width as primitives to avoid wasted computation
const width = derived(bounds, $bounds => $bounds.width, 0)
const height = derived(bounds, $bounds => $bounds.height, 0)
return { bounds, height, width }
} }

View File

@ -2,8 +2,7 @@ import { derived, get, writable } from "svelte/store"
export const DefaultColumnWidth = 200 export const DefaultColumnWidth = 200
export const createColumnsStores = context => { export const createStores = () => {
const { table, gutterWidth } = context
const columns = writable([]) const columns = writable([])
const stickyColumn = writable(null) const stickyColumn = writable(null)
@ -36,6 +35,19 @@ export const createColumnsStores = context => {
[] []
) )
return {
columns: {
...columns,
subscribe: enrichedColumns.subscribe,
},
stickyColumn,
visibleColumns,
}
}
export const deriveStores = context => {
const { table, gutterWidth, columns, stickyColumn } = context
// Merge new schema fields with existing schema in order to preserve widths // Merge new schema fields with existing schema in order to preserve widths
table.subscribe($table => { table.subscribe($table => {
const schema = $table?.schema const schema = $table?.schema
@ -109,12 +121,5 @@ export const createColumnsStores = context => {
}) })
}) })
return { return null
columns: {
...columns,
subscribe: enrichedColumns.subscribe,
},
stickyColumn,
visibleColumns,
}
} }

View File

@ -0,0 +1,43 @@
import * as Bounds from "./bounds"
import * as Columns from "./columns"
import * as Menu from "./menu"
import * as Pagination from "./pagination"
import * as Reorder from "./reorder"
import * as Resize from "./resize"
import * as Rows from "./rows"
import * as Scroll from "./scroll"
import * as UI from "./ui"
import * as Users from "./users"
import * as Validation from "./validation"
import * as Viewport from "./viewport"
const DependencyOrderedStores = [
Bounds,
Scroll,
Rows,
Columns,
UI,
Validation,
Resize,
Viewport,
Reorder,
Users,
Menu,
Pagination,
]
export const createStores = context => {
let stores = {}
// Atomic store creation
for (let store of DependencyOrderedStores) {
stores = { ...stores, ...store.createStores?.({ ...context, ...stores }) }
}
// Derived store creation
for (let store of DependencyOrderedStores) {
stores = { ...stores, ...store.deriveStores?.({ ...context, ...stores }) }
}
return stores
}

View File

@ -1,182 +0,0 @@
import { derived, get } from "svelte/store"
import { tick } from "svelte"
export const createMaxScrollStores = context => {
const {
rows,
visibleColumns,
stickyColumn,
bounds,
rowHeight,
scroll,
focusedRow,
focusedCellId,
gutterWidth,
} = context
const padding = 264
// Memoize store primitives
const scrollTop = derived(scroll, $scroll => $scroll.top, 0)
const scrollLeft = derived(scroll, $scroll => $scroll.left, 0)
const stickyColumnWidth = derived(stickyColumn, $col => $col?.width || 0, 0)
// Derive vertical limits
const height = derived(bounds, $bounds => $bounds.height, 0)
const width = derived(bounds, $bounds => $bounds.width, 0)
const contentHeight = derived(
[rows, rowHeight],
([$rows, $rowHeight]) => $rows.length * $rowHeight + padding,
0
)
const maxScrollTop = derived(
[height, contentHeight],
([$height, $contentHeight]) => Math.max($contentHeight - $height, 0),
0
)
// Derive horizontal limits
const contentWidth = derived(
[visibleColumns, stickyColumnWidth],
([$visibleColumns, $stickyColumnWidth]) => {
let width = gutterWidth + padding + $stickyColumnWidth
$visibleColumns.forEach(col => {
width += col.width
})
return width
},
0
)
const screenWidth = derived(
[width, stickyColumnWidth],
([$width, $stickyColumnWidth]) => $width + gutterWidth + $stickyColumnWidth,
0
)
const maxScrollLeft = derived(
[contentWidth, screenWidth],
([$contentWidth, $screenWidth]) => {
return Math.max($contentWidth - $screenWidth, 0)
},
0
)
// Ensure scroll state never goes invalid, which can happen when changing
// rows or tables
const overscrollTop = derived(
[scrollTop, maxScrollTop],
([$scrollTop, $maxScrollTop]) => $scrollTop > $maxScrollTop,
false
)
const overscrollLeft = derived(
[scrollLeft, maxScrollLeft],
([$scrollLeft, $maxScrollLeft]) => $scrollLeft > $maxScrollLeft,
false
)
overscrollTop.subscribe(overscroll => {
if (overscroll) {
scroll.update(state => ({
...state,
top: get(maxScrollTop),
}))
}
})
overscrollLeft.subscribe(overscroll => {
if (overscroll) {
scroll.update(state => ({
...state,
left: get(maxScrollLeft),
}))
}
})
// Ensure the selected cell is visible
focusedCellId.subscribe(async $focusedCellId => {
await tick()
const $focusedRow = get(focusedRow)
const $scroll = get(scroll)
const $bounds = get(bounds)
const $rowHeight = get(rowHeight)
const verticalOffset = $rowHeight * 1.5
// Ensure vertical position is viewable
if ($focusedRow) {
// Ensure row is not below bottom of screen
const rowYPos = $focusedRow.__idx * $rowHeight
const bottomCutoff =
$scroll.top + $bounds.height - $rowHeight - verticalOffset
let delta = rowYPos - bottomCutoff
if (delta > 0) {
scroll.update(state => ({
...state,
top: state.top + delta,
}))
}
// Ensure row is not above top of screen
else {
const delta = $scroll.top - rowYPos + verticalOffset
if (delta > 0) {
scroll.update(state => ({
...state,
top: Math.max(0, state.top - delta),
}))
}
}
}
// Ensure horizontal position is viewable
// Check horizontal position of columns next
const $visibleColumns = get(visibleColumns)
const columnName = $focusedCellId?.split("-")[1]
const column = $visibleColumns.find(col => col.name === columnName)
const horizontalOffset = 24
if (!column) {
return
}
// Ensure column is not cutoff on left edge
let delta = $scroll.left - column.left + horizontalOffset
if (delta > 0) {
scroll.update(state => ({
...state,
left: Math.max(0, state.left - delta),
}))
}
// Ensure column is not cutoff on right edge
else {
const rightEdge = column.left + column.width
const rightBound = $bounds.width + $scroll.left - horizontalOffset
delta = rightEdge - rightBound
if (delta > 0) {
scroll.update(state => ({
...state,
left: state.left + delta,
}))
}
}
})
// Derive whether to show scrollbars or not
const showVScrollbar = derived(
[contentHeight, height],
([$contentHeight, $height]) => {
return $contentHeight > $height
}
)
const showHScrollbar = derived(
[contentWidth, screenWidth],
([$contentWidth, $screenWidth]) => {
return $contentWidth > $screenWidth
}
)
return {
contentHeight,
contentWidth,
screenWidth,
maxScrollTop,
maxScrollLeft,
showHScrollbar,
showVScrollbar,
}
}

View File

@ -1,13 +1,19 @@
import { writable, get } from "svelte/store" import { writable, get } from "svelte/store"
export const createMenuStores = context => { export const createStores = () => {
const { bounds, focusedCellId, stickyColumn, rowHeight } = context
const menu = writable({ const menu = writable({
x: 0, x: 0,
y: 0, y: 0,
visible: false, visible: false,
selectedRow: null, selectedRow: null,
}) })
return {
menu,
}
}
export const deriveStores = context => {
const { menu, bounds, focusedCellId, stickyColumn, rowHeight } = context
const open = (cellId, e) => { const open = (cellId, e) => {
const $bounds = get(bounds) const $bounds = get(bounds)

View File

@ -1,6 +1,6 @@
import { derived } from "svelte/store" import { derived } from "svelte/store"
export const createPaginationStores = context => { export const deriveStores = context => {
const { scrolledRowCount, rows, visualRowCapacity } = context const { scrolledRowCount, rows, visualRowCapacity } = context
// Derive how many rows we have in total // Derive how many rows we have in total

View File

@ -1,7 +1,5 @@
import { get, writable, derived } from "svelte/store" import { get, writable, derived } from "svelte/store"
export const createReorderStores = context => {
const { columns, scroll, bounds, stickyColumn, ui } = context
const reorderInitialState = { const reorderInitialState = {
sourceColumn: null, sourceColumn: null,
targetColumn: null, targetColumn: null,
@ -10,12 +8,22 @@ export const createReorderStores = context => {
scrollLeft: 0, scrollLeft: 0,
sheetLeft: 0, sheetLeft: 0,
} }
export const createStores = () => {
const reorder = writable(reorderInitialState) const reorder = writable(reorderInitialState)
const isReordering = derived( const isReordering = derived(
reorder, reorder,
$reorder => !!$reorder.sourceColumn, $reorder => !!$reorder.sourceColumn,
false false
) )
return {
reorder,
isReordering,
}
}
export const deriveStores = context => {
const { reorder, columns, scroll, bounds, stickyColumn, ui } = context
// Callback when dragging on a colum header and starting reordering // Callback when dragging on a colum header and starting reordering
const startReordering = (column, e) => { const startReordering = (column, e) => {
@ -142,6 +150,5 @@ export const createReorderStores = context => {
moveColumnRight, moveColumnRight,
}, },
}, },
isReordering,
} }
} }

View File

@ -3,8 +3,6 @@ import { DefaultColumnWidth } from "./columns"
export const MinColumnWidth = 100 export const MinColumnWidth = 100
export const createResizeStores = context => {
const { columns, stickyColumn, ui } = context
const initialState = { const initialState = {
initialMouseX: null, initialMouseX: null,
initialWidth: null, initialWidth: null,
@ -13,8 +11,18 @@ export const createResizeStores = context => {
width: 0, width: 0,
left: 0, left: 0,
} }
export const createStores = () => {
const resize = writable(initialState) const resize = writable(initialState)
const isResizing = derived(resize, $resize => $resize.column != null, false) const isResizing = derived(resize, $resize => $resize.column != null, false)
return {
resize,
isResizing,
}
}
export const deriveStores = context => {
const { resize, columns, stickyColumn, ui } = context
// Starts resizing a certain column // Starts resizing a certain column
const startResizing = (column, e) => { const startResizing = (column, e) => {
@ -105,6 +113,5 @@ export const createResizeStores = context => {
resetSize, resetSize,
}, },
}, },
isResizing,
} }
} }

View File

@ -2,18 +2,16 @@ import { writable, derived, get } from "svelte/store"
import { fetchData } from "../../../fetch/fetchData" import { fetchData } from "../../../fetch/fetchData"
import { notifications } from "@budibase/bbui" import { notifications } from "@budibase/bbui"
export const createRowsStore = context => {
const { tableId, API, scroll, validation } = context
const rows = writable([])
const table = writable(null)
const filter = writable([])
const loaded = writable(false)
const instanceLoaded = writable(false)
const fetch = writable(null)
const initialSortState = { const initialSortState = {
column: null, column: null,
order: "ascending", order: "ascending",
} }
export const createStores = () => {
const rows = writable([])
const table = writable(null)
const filter = writable([])
const loaded = writable(false)
const sort = writable(initialSortState) const sort = writable(initialSortState)
// Enrich rows with an index property // Enrich rows with an index property
@ -41,6 +39,37 @@ export const createRowsStore = context => {
{} {}
) )
return {
rows: {
...rows,
subscribe: enrichedRows.subscribe,
},
rowLookupMap,
table,
filter,
loaded,
sort,
}
}
export const deriveStores = context => {
const {
rows,
rowLookupMap,
table,
filter,
loaded,
sort,
tableId,
API,
scroll,
validation,
focusedCellId,
columns,
} = context
const instanceLoaded = writable(false)
const fetch = writable(null)
// Local cache of row IDs to speed up checking if a row exists // Local cache of row IDs to speed up checking if a row exists
let rowCacheMap = {} let rowCacheMap = {}
@ -117,19 +146,32 @@ export const createRowsStore = context => {
// Gets a row by ID // Gets a row by ID
const getRow = id => { const getRow = id => {
const index = get(rowLookupMap)[id] const index = get(rowLookupMap)[id]
return index >= 0 ? get(enrichedRows)[index] : null return index >= 0 ? get(rows)[index] : null
} }
// Handles validation errors from the rows API and updates local validation // Handles validation errors from the rows API and updates local validation
// state, storing error messages against relevant cells // state, storing error messages against relevant cells
const handleValidationError = (rowId, error) => { const handleValidationError = (rowId, error) => {
if (error?.json?.validationErrors) { if (error?.json?.validationErrors) {
for (let column of Object.keys(error.json.validationErrors)) { const keys = Object.keys(error.json.validationErrors)
const $columns = get(columns)
for (let column of keys) {
validation.actions.setError( validation.actions.setError(
`${rowId}-${column}`, `${rowId}-${column}`,
`${column} ${error.json.validationErrors[column]}` `${column} ${error.json.validationErrors[column]}`
) )
// Ensure the column is visible
const index = $columns.findIndex(x => x.name === column)
if (index !== -1 && !$columns[index].visible) {
columns.update(state => {
state[index].visible = true
return state.slice()
})
} }
}
// Focus the first cell with an error
focusedCellId.set(`${rowId}-${keys[0]}`)
} else { } else {
notifications.error(`Error saving row: ${error?.message}`) notifications.error(`Error saving row: ${error?.message}`)
} }
@ -299,7 +341,6 @@ export const createRowsStore = context => {
return { return {
rows: { rows: {
...rows, ...rows,
subscribe: enrichedRows.subscribe,
actions: { actions: {
addRow, addRow,
getRow, getRow,
@ -312,10 +353,5 @@ export const createRowsStore = context => {
refreshTableDefinition, refreshTableDefinition,
}, },
}, },
rowLookupMap,
table,
sort,
filter,
loaded,
} }
} }

View File

@ -1,11 +1,199 @@
import { writable } from "svelte/store" import { writable, derived, get } from "svelte/store"
import { tick } from "svelte"
export const createScrollStores = () => { export const createStores = () => {
const scroll = writable({ const scroll = writable({
left: 0, left: 0,
top: 0, top: 0,
}) })
// Derive height and width as primitives to avoid wasted computation
const scrollTop = derived(scroll, $scroll => $scroll.top, 0)
const scrollLeft = derived(scroll, $scroll => $scroll.left, 0)
return { return {
scroll, scroll,
scrollTop,
scrollLeft,
}
}
export const deriveStores = context => {
const {
scroll,
rows,
visibleColumns,
stickyColumn,
bounds,
rowHeight,
focusedRow,
focusedCellId,
gutterWidth,
} = context
const padding = 264
// Memoize store primitives
const scrollTop = derived(scroll, $scroll => $scroll.top, 0)
const scrollLeft = derived(scroll, $scroll => $scroll.left, 0)
const stickyColumnWidth = derived(stickyColumn, $col => $col?.width || 0, 0)
// Derive vertical limits
const height = derived(bounds, $bounds => $bounds.height, 0)
const width = derived(bounds, $bounds => $bounds.width, 0)
const contentHeight = derived(
[rows, rowHeight],
([$rows, $rowHeight]) => $rows.length * $rowHeight + padding,
0
)
const maxScrollTop = derived(
[height, contentHeight],
([$height, $contentHeight]) => Math.max($contentHeight - $height, 0),
0
)
// Derive horizontal limits
const contentWidth = derived(
[visibleColumns, stickyColumnWidth],
([$visibleColumns, $stickyColumnWidth]) => {
let width = gutterWidth + padding + $stickyColumnWidth
$visibleColumns.forEach(col => {
width += col.width
})
return width
},
0
)
const screenWidth = derived(
[width, stickyColumnWidth],
([$width, $stickyColumnWidth]) => $width + gutterWidth + $stickyColumnWidth,
0
)
const maxScrollLeft = derived(
[contentWidth, screenWidth],
([$contentWidth, $screenWidth]) => {
return Math.max($contentWidth - $screenWidth, 0)
},
0
)
// Ensure scroll state never goes invalid, which can happen when changing
// rows or tables
const overscrollTop = derived(
[scrollTop, maxScrollTop],
([$scrollTop, $maxScrollTop]) => $scrollTop > $maxScrollTop,
false
)
const overscrollLeft = derived(
[scrollLeft, maxScrollLeft],
([$scrollLeft, $maxScrollLeft]) => $scrollLeft > $maxScrollLeft,
false
)
overscrollTop.subscribe(overscroll => {
if (overscroll) {
scroll.update(state => ({
...state,
top: get(maxScrollTop),
}))
}
})
overscrollLeft.subscribe(overscroll => {
if (overscroll) {
scroll.update(state => ({
...state,
left: get(maxScrollLeft),
}))
}
})
// Ensure the selected cell is visible
focusedCellId.subscribe(async $focusedCellId => {
await tick()
const $focusedRow = get(focusedRow)
const $scroll = get(scroll)
const $bounds = get(bounds)
const $rowHeight = get(rowHeight)
const verticalOffset = $rowHeight * 1.5
// Ensure vertical position is viewable
if ($focusedRow) {
// Ensure row is not below bottom of screen
const rowYPos = $focusedRow.__idx * $rowHeight
const bottomCutoff =
$scroll.top + $bounds.height - $rowHeight - verticalOffset
let delta = rowYPos - bottomCutoff
if (delta > 0) {
scroll.update(state => ({
...state,
top: state.top + delta,
}))
}
// Ensure row is not above top of screen
else {
const delta = $scroll.top - rowYPos + verticalOffset
if (delta > 0) {
scroll.update(state => ({
...state,
top: Math.max(0, state.top - delta),
}))
}
}
}
// Ensure horizontal position is viewable
// Check horizontal position of columns next
const $visibleColumns = get(visibleColumns)
const columnName = $focusedCellId?.split("-")[1]
const column = $visibleColumns.find(col => col.name === columnName)
const horizontalOffset = 24
if (!column) {
return
}
// Ensure column is not cutoff on left edge
let delta = $scroll.left - column.left + horizontalOffset
if (delta > 0) {
scroll.update(state => ({
...state,
left: Math.max(0, state.left - delta),
}))
}
// Ensure column is not cutoff on right edge
else {
const rightEdge = column.left + column.width
const rightBound = $bounds.width + $scroll.left - horizontalOffset
delta = rightEdge - rightBound
if (delta > 0) {
scroll.update(state => ({
...state,
left: state.left + delta,
}))
}
}
})
// Derive whether to show scrollbars or not
const showVScrollbar = derived(
[contentHeight, height],
([$contentHeight, $height]) => {
return $contentHeight > $height
}
)
const showHScrollbar = derived(
[contentWidth, screenWidth],
([$contentWidth, $screenWidth]) => {
return $contentWidth > $screenWidth
}
)
return {
contentHeight,
contentWidth,
screenWidth,
maxScrollTop,
maxScrollLeft,
showHScrollbar,
showVScrollbar,
} }
} }

View File

@ -1,66 +0,0 @@
import { derived, get, writable } from "svelte/store"
export const createSheetAPIStores = context => {
const { focusedCellId } = context
const cellAPIs = writable({})
const registerCellAPI = (cellId, api) => {
// Ignore registration if cell is not selected
const [rowId, column] = cellId.split("-")
if (rowId !== "new" && !get(focusedCellId)?.startsWith(rowId)) {
return
}
// Store API
cellAPIs.update(state => ({
...state,
[column]: api,
}))
}
const getCellAPI = column => {
return get(cellAPIs)[column]
}
// Derive the selected cell API
const selectedCellAPI = derived(
[cellAPIs, focusedCellId],
([$apis, $focusedCellId]) => {
if (!$focusedCellId) {
return null
}
const [, column] = $focusedCellId.split("-")
return $apis[column]
},
null
)
const focusedRowAPI = derived(cellAPIs, $apis => {
return {
validate: () => {
let errors = null
for (let [column, api] of Object.entries($apis || {})) {
const error = api.validate()
if (error) {
errors = {
...errors,
[column]: error,
}
}
}
return errors
},
}
})
return {
selectedCellAPI,
focusedRowAPI,
sheetAPI: {
actions: {
registerCellAPI,
getCellAPI,
},
},
}
}

View File

@ -1,19 +1,38 @@
import { writable, get, derived } from "svelte/store" import { writable, get, derived } from "svelte/store"
export const createUIStores = context => { export const createStores = () => {
const { rows, rowLookupMap } = context
const focusedCellId = writable(null) const focusedCellId = writable(null)
const focusedCellAPI = writable(null)
const selectedRows = writable({}) const selectedRows = writable({})
const hoveredRowId = writable(null) const hoveredRowId = writable(null)
const rowHeight = writable(36) const rowHeight = writable(36)
return {
focusedCellId,
focusedCellAPI,
selectedRows,
hoveredRowId,
rowHeight,
}
}
export const deriveStores = context => {
const { focusedCellId, selectedRows, hoveredRowId, rows, rowLookupMap } =
context
// Derive the row that contains the selected cell // Derive the row that contains the selected cell
const focusedRow = derived( const focusedRow = derived(
[focusedCellId, rowLookupMap, rows], [focusedCellId, rowLookupMap, rows],
([$focusedCellId, $rowLookupMap, $rows]) => { ([$focusedCellId, $rowLookupMap, $rows]) => {
const rowId = $focusedCellId?.split("-")[0] const rowId = $focusedCellId?.split("-")[0]
if (rowId === "new") {
// Edge case for new row
return { _id: rowId }
} else {
// All normal rows
const index = $rowLookupMap[rowId] const index = $rowLookupMap[rowId]
return $rows[index] return $rows[index]
}
}, },
null null
) )
@ -80,11 +99,7 @@ export const createUIStores = context => {
}) })
return { return {
focusedCellId,
selectedRows,
hoveredRowId,
focusedRow, focusedRow,
rowHeight,
ui: { ui: {
actions: { actions: {
blur, blur,

View File

@ -1,6 +1,6 @@
import { writable, get, derived } from "svelte/store" import { writable, get, derived } from "svelte/store"
export const createUserStores = () => { export const createStores = () => {
const users = writable([]) const users = writable([])
const userId = writable(null) const userId = writable(null)
@ -55,10 +55,22 @@ export const createUserStores = () => {
[] []
) )
return {
users: {
...users,
subscribe: enrichedUsers.subscribe,
},
userId,
}
}
export const deriveStores = context => {
const { users, userId } = context
// Generate a lookup map of cell ID to the user that has it selected, to make // Generate a lookup map of cell ID to the user that has it selected, to make
// lookups inside sheet cells extremely fast // lookups inside sheet cells extremely fast
const selectedCellMap = derived( const selectedCellMap = derived(
[enrichedUsers, userId], [users, userId],
([$enrichedUsers, $userId]) => { ([$enrichedUsers, $userId]) => {
let map = {} let map = {}
$enrichedUsers.forEach(user => { $enrichedUsers.forEach(user => {
@ -92,15 +104,12 @@ export const createUserStores = () => {
return { return {
users: { users: {
...enrichedUsers, ...users,
set: users.set,
update: users.update,
actions: { actions: {
updateUser, updateUser,
removeUser, removeUser,
}, },
}, },
selectedCellMap, selectedCellMap,
userId,
} }
} }

View File

@ -1,13 +1,9 @@
import { writable, get } from "svelte/store" import { writable, get, derived } from "svelte/store"
export const createValidationStores = () => { export const createStores = () => {
const validation = writable({}) const validation = writable({})
return { const setError = (cellId, error) => {
validation: {
subscribe: validation.subscribe,
actions: {
setError: (cellId, error) => {
if (!cellId) { if (!cellId) {
return return
} }
@ -15,11 +11,42 @@ export const createValidationStores = () => {
...state, ...state,
[cellId]: error, [cellId]: error,
})) }))
}, }
getError: cellId => {
return get(validation)[cellId] return {
}, validation: {
...validation,
actions: {
setError,
}, },
}, },
} }
} }
export const deriveStores = context => {
const { validation, focusedRow, columns, stickyColumn } = context
const focusedRowId = derived(focusedRow, $focusedRow => $focusedRow?._id)
// Store the row ID that was previously focused, so we can remove errors from
// it when we focus a new row
let previousFocusedRowId = null
focusedRowId.subscribe(id => {
// Remove validation errors from previous focused row
if (previousFocusedRowId) {
const $columns = get(columns)
const $stickyColumn = get(stickyColumn)
validation.update(state => {
$columns.forEach(column => {
state[`${previousFocusedRowId}-${column.name}`] = null
})
if ($stickyColumn) {
state[`${previousFocusedRowId}-${$stickyColumn.name}`] = null
}
return state
})
}
// Store row ID
previousFocusedRowId = id
})
}

View File

@ -1,13 +1,15 @@
import { derived, get } from "svelte/store" import { derived, get } from "svelte/store"
export const createViewportStores = context => { export const deriveStores = context => {
const { rowHeight, visibleColumns, rows, scroll, bounds } = context const {
const scrollTop = derived(scroll, $scroll => $scroll.top, 0) rowHeight,
const scrollLeft = derived(scroll, $scroll => $scroll.left, 0) visibleColumns,
rows,
// Derive height and width as primitives to avoid wasted computation scrollTop,
const width = derived(bounds, $bounds => $bounds.width, 0) scrollLeft,
const height = derived(bounds, $bounds => $bounds.height, 0) width,
height,
} = context
// Derive visible rows // Derive visible rows
// Split into multiple stores containing primitives to optimise invalidation // Split into multiple stores containing primitives to optimise invalidation