From ee984b6a191c52722d4138df7b3c742fff6d4d77 Mon Sep 17 00:00:00 2001 From: Andrew Kingston Date: Tue, 16 Jul 2024 14:01:50 +0100 Subject: [PATCH] Update table conditions to apply both row and cell conditions via same configuration --- packages/bbui/src/Form/Core/Select.svelte | 4 +- .../controls/CellConditionEditor.svelte | 25 +++- packages/client/manifest.json | 12 -- .../src/components/app/GridBlock.svelte | 2 - .../src/components/grid/cells/DataCell.svelte | 4 +- .../components/grid/cells/GutterCell.svelte | 2 +- .../src/components/grid/stores/conditions.js | 119 ++++++------------ .../src/components/grid/stores/viewport.js | 18 +-- 8 files changed, 71 insertions(+), 115 deletions(-) diff --git a/packages/bbui/src/Form/Core/Select.svelte b/packages/bbui/src/Form/Core/Select.svelte index 0111bd6bcf..3806281438 100644 --- a/packages/bbui/src/Form/Core/Select.svelte +++ b/packages/bbui/src/Form/Core/Select.svelte @@ -62,7 +62,9 @@ return placeholder || "Choose an option" } - return getFieldAttribute(getOptionLabel, value, options) + return ( + getFieldAttribute(getOptionLabel, value, options) || "Choose an option" + ) } const selectOption = value => { diff --git a/packages/builder/src/components/design/settings/controls/CellConditionEditor.svelte b/packages/builder/src/components/design/settings/controls/CellConditionEditor.svelte index b901437cba..69b65fb449 100644 --- a/packages/builder/src/components/design/settings/controls/CellConditionEditor.svelte +++ b/packages/builder/src/components/design/settings/controls/CellConditionEditor.svelte @@ -25,13 +25,23 @@ const dispatch = createEventDispatcher() const flipDuration = 130 + const targetOptions = [ + { + label: "Cell", + value: "cell", + }, + { + label: "Row", + value: "row", + }, + ] const conditionOptions = [ { - label: "Update background color", + label: "Background color", value: "backgroundColor", }, { - label: "Update text color", + label: "Text color", value: "textColor", }, ] @@ -68,6 +78,7 @@ const addCondition = () => { const condition = { id: generate(), + target: targetOptions[0].value, metadataKey: conditionOptions[0].value, operator: Constants.OperatorOptions.Equals.value, valueType: FieldType.STRING, @@ -123,7 +134,7 @@
- Update the appearance of cells based on their value. + Update the appearance of cells and rows based on their value. {#if tempValue.length}
+ Update +
{#if $$slots.default} diff --git a/packages/frontend-core/src/components/grid/stores/conditions.js b/packages/frontend-core/src/components/grid/stores/conditions.js index 04eb10b89b..ad4fc9fa84 100644 --- a/packages/frontend-core/src/components/grid/stores/conditions.js +++ b/packages/frontend-core/src/components/grid/stores/conditions.js @@ -1,15 +1,13 @@ import { writable, get } from "svelte/store" import { derivedMemo, QueryUtils } from "../../../utils" -import { OperatorOptions } from "@budibase/shared-core" -import { FieldType } from "@budibase/types" -import { processString, processStringSync } from "@budibase/string-templates" export const createStores = () => { - const cellMetadata = writable({}) - const rowMetadata = writable({}) + const metadata = writable({}) + + metadata.subscribe(console.log) + return { - cellMetadata, - rowMetadata, + metadata, } } @@ -18,7 +16,7 @@ export const deriveStores = context => { // Derive and memoize the cell conditions present in our columns so that we // only recompute condition metadata when absolutely necessary - const cellConditions = derivedMemo(columns, $columns => { + const conditions = derivedMemo(columns, $columns => { let newConditions = [] for (let column of $columns) { for (let condition of column.conditions || []) { @@ -33,73 +31,43 @@ export const deriveStores = context => { }) return { - cellConditions, + conditions, } } export const initialise = context => { - const { cellMetadata, cellConditions, rowConditions, rowMetadata, rows } = - context + const { metadata, conditions, rows } = context // Recompute all cell metadata if cell conditions change - cellConditions.subscribe($conditions => { - let metadata = {} + conditions.subscribe($conditions => { + // Build new metadata + let newMetadata = {} if ($conditions?.length) { for (let row of get(rows)) { - metadata[row._id] = evaluateCellConditions(row, $conditions) + newMetadata[row._id] = evaluateConditions(row, $conditions) } } - cellMetadata.set(metadata) - }) - - // Recompute all row metadata if row conditions change - rowConditions.subscribe($conditions => { - let metadata = {} - if ($conditions?.length) { - for (let row of get(rows)) { - metadata[row._id] = evaluateRowConditions(row, $conditions) - } - } - rowMetadata.set(metadata) + metadata.set(newMetadata) }) // Recompute metadata for specific rows when they change rows.subscribe($rows => { - const $cellConditions = get(cellConditions) - const $rowConditions = get(rowConditions) - const processCells = $cellConditions?.length > 0 - const processRows = $rowConditions?.length > 0 - if (!processCells && !processRows) { + const $conditions = get(conditions) + if (!$conditions?.length) { return } - const $cellMetadata = get(cellMetadata) - const $rowMetadata = get(rowMetadata) - let cellUpdates = {} - let rowUpdates = {} + const $metadata = get(metadata) + let metadataUpdates = {} for (let row of $rows) { - // Process cell metadata - if (processCells) { - if (!row._rev || $cellMetadata[row._id]?.version !== row._rev) { - cellUpdates[row._id] = evaluateCellConditions(row, $cellConditions) - } - } - // Process row metadata - if (processRows) { - if (!row._rev || $rowMetadata[row._id]?.version !== row._rev) { - rowUpdates[row._id] = evaluateRowConditions(row, $rowConditions) - } + if (!row._rev || $metadata[row._id]?.version !== row._rev) { + console.log("reevalute", row._id) + metadataUpdates[row._id] = evaluateConditions(row, $conditions) } } - if (Object.keys(cellUpdates).length) { - cellMetadata.update(state => ({ + if (Object.keys(metadataUpdates).length) { + metadata.update(state => ({ ...state, - ...cellUpdates, - })) - } - if (Object.keys(rowUpdates).length) { - rowMetadata.update(state => ({ - ...state, - ...rowUpdates, + ...metadataUpdates, })) } }) @@ -107,8 +75,12 @@ export const initialise = context => { // Evaluates an array of cell conditions against a certain row and returns the // resultant metadata -const evaluateCellConditions = (row, conditions) => { - let metadata = { version: row._rev } +const evaluateConditions = (row, conditions) => { + let metadata = { + version: row._rev, + row: {}, + cell: {}, + } for (let condition of conditions) { try { let { @@ -118,6 +90,7 @@ const evaluateCellConditions = (row, conditions) => { operator, metadataKey, metadataValue, + target, } = condition let value = row[column] @@ -147,29 +120,17 @@ const evaluateCellConditions = (row, conditions) => { const query = QueryUtils.buildQuery([luceneFilter]) const result = QueryUtils.runQuery([{ value }], query) if (result.length > 0) { - if (!metadata[column]) { - metadata[column] = {} + if (target === "row") { + metadata.row = { + ...metadata.row, + [metadataKey]: metadataValue, + } + } else { + metadata.cell[column] = { + ...metadata.cell[column], + [metadataKey]: metadataValue, + } } - metadata[column][metadataKey] = metadataValue - } - } catch { - // Swallow - } - } - return metadata -} - -// Evaluates an array of row conditions against a certain row and returns the -// resultant metadata -const evaluateRowConditions = (row, conditions) => { - let metadata = { version: row._rev } - for (let condition of conditions) { - try { - const { metadataKey, metadataValue, value } = condition - console.log("JS") - const result = processStringSync(value, { row }) - if (result === true) { - metadata[metadataKey] = metadataValue } } catch { // Swallow diff --git a/packages/frontend-core/src/components/grid/stores/viewport.js b/packages/frontend-core/src/components/grid/stores/viewport.js index fc5960f14c..0797f38b55 100644 --- a/packages/frontend-core/src/components/grid/stores/viewport.js +++ b/packages/frontend-core/src/components/grid/stores/viewport.js @@ -11,8 +11,7 @@ export const deriveStores = context => { width, height, rowChangeCache, - cellMetadata, - rowMetadata, + metadata, } = context // Derive visible rows @@ -31,29 +30,20 @@ export const deriveStores = context => { } ) const renderedRows = derived( - [ - rows, - scrolledRowCount, - visualRowCapacity, - rowChangeCache, - cellMetadata, - rowMetadata, - ], + [rows, scrolledRowCount, visualRowCapacity, rowChangeCache, metadata], ([ $rows, $scrolledRowCount, $visualRowCapacity, $rowChangeCache, - $cellMetadata, - $rowMetadata, + $metadata, ]) => { return $rows .slice($scrolledRowCount, $scrolledRowCount + $visualRowCapacity) .map(row => ({ ...row, ...$rowChangeCache[row._id], - __metadata: $rowMetadata[row._id], - __cellMetadata: $cellMetadata[row._id], + __metadata: $metadata[row._id], })) } )