@marimo-team/frontend 0.23.4-dev0 → 0.23.4-dev1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/assets/{CellStatus-BTL_sgUY.js → CellStatus-PIeZtHEl.js} +1 -1
- package/dist/assets/JsonOutput-BVAcY5xS.js +49 -0
- package/dist/assets/{MarimoErrorOutput-BBD405a6.js → MarimoErrorOutput-B9_RO5eX.js} +2 -2
- package/dist/assets/{RenderHTML-BT9obKLc.js → RenderHTML-DR6lqJGS.js} +1 -1
- package/dist/assets/{add-cell-with-ai-qEfAzC8R.js → add-cell-with-ai-DjfY8mF1.js} +9 -9
- package/dist/assets/{add-connection-dialog-HCShSlr3.js → add-connection-dialog-gd2yD9Qo.js} +2 -2
- package/dist/assets/{agent-panel-CoBHxHpJ.js → agent-panel-TetMUXWP.js} +3 -3
- package/dist/assets/{ai-model-dropdown-CYx9ZfdS.js → ai-model-dropdown-By4Ohb0R.js} +3 -3
- package/dist/assets/{app-config-button-DC5UlXZA.js → app-config-button-YS8lpoUa.js} +1 -1
- package/dist/assets/{cell-editor-BMjfXh0J.js → cell-editor-C6vZfKyj.js} +10 -10
- package/dist/assets/{cell-link-BA7Demf0.js → cell-link-d38hFEon.js} +1 -1
- package/dist/assets/{cells-CJlo_hG2.js → cells-A-ljPpuj.js} +55 -55
- package/dist/assets/{chat-display-DbztpRD4.js → chat-display-Be0U92hE.js} +1 -1
- package/dist/assets/{chat-panel-H-xFzATI.js → chat-panel-BEz970S3.js} +1 -1
- package/dist/assets/{chat-ui-CFqAy2ck.js → chat-ui-CdW7awFa.js} +1 -1
- package/dist/assets/{column-preview-DzN2QumC.js → column-preview-_5uYiy74.js} +1 -1
- package/dist/assets/{command-DbT_zkRP.js → command-CzIDL1VI.js} +1 -1
- package/dist/assets/{command-palette-DSSR50KF.js → command-palette-DfotChyr.js} +1 -1
- package/dist/assets/{common-CquNY-ok.js → common-DC9LEIym.js} +1 -1
- package/dist/assets/{components-BXKuiPiF.js → components-BDnRtsA5.js} +1 -1
- package/dist/assets/{components-BiBOcq1x.js → components-FepcpFGL.js} +1 -1
- package/dist/assets/{datasource-BKDU-4D5.js → datasource-BCnGIjqn.js} +2 -2
- package/dist/assets/{dependency-graph-panel-ayo-Mm2j.js → dependency-graph-panel-Yk1Ch_Gj.js} +1 -1
- package/dist/assets/{documentation-panel-ph6755dz.js → documentation-panel-BMrGuByB.js} +1 -1
- package/dist/assets/{download-mzsKQgiy.js → download-Iyng0xCz.js} +4 -4
- package/dist/assets/{edit-page-Bxa5DWoE.js → edit-page-C-LIP5uT.js} +6 -6
- package/dist/assets/{error-panel-BfG_JeMA.js → error-panel-CHSsnRFF.js} +1 -1
- package/dist/assets/{file-explorer-panel-D7-d0Puf.js → file-explorer-panel-Dlwspu2Y.js} +3 -3
- package/dist/assets/{file-icons-BNrh8MRG.js → file-icons-DTNZv05h.js} +1 -1
- package/dist/assets/{floating-outline-BMB4_phA.js → floating-outline-CSNGDMAm.js} +1 -1
- package/dist/assets/{focus-BX3gXJxx.js → focus-Baz7rlo0.js} +1 -1
- package/dist/assets/form-Dr-_EXr8.js +2 -0
- package/dist/assets/{home-page-B1ELDhn8.js → home-page-QvO22-4u.js} +1 -1
- package/dist/assets/{hooks-b3J1eFE-.js → hooks-CNPv5U-N.js} +1 -1
- package/dist/assets/{html-to-image-Cz8lDF-Y.js → html-to-image-tbPEMBJg.js} +1 -1
- package/dist/assets/{index-BjiE1T38.js → index-BFivsT-R.js} +8 -8
- package/dist/assets/index-DH5UUBE4.css +2 -0
- package/dist/assets/{kiosk-mode-Dxbs5i5g.js → kiosk-mode-cWTXpegb.js} +1 -1
- package/dist/assets/{layout-lxwMDUgI.js → layout-DDh3d8oW.js} +5 -5
- package/dist/assets/{logs-panel-BW6S5AI1.js → logs-panel-Dgkr6V7q.js} +1 -1
- package/dist/assets/{markdown-renderer-CeHY2KoQ.js → markdown-renderer-CFrgiLqu.js} +1 -1
- package/dist/assets/{name-cell-input-B1BCybeo.js → name-cell-input-C5LXTI3T.js} +1 -1
- package/dist/assets/{outline-panel-DgMz83Mx.js → outline-panel-0lyts6z4.js} +1 -1
- package/dist/assets/{packages-panel-CbjavFh2.js → packages-panel-BhI1lbfw.js} +1 -1
- package/dist/assets/{panels-BPzk3EbR.js → panels-CbRenLjk.js} +1 -1
- package/dist/assets/{process-output-CggRIZ6s.js → process-output-BwZ5VxXO.js} +1 -1
- package/dist/assets/{readonly-python-code-Ccd8HM-7.js → readonly-python-code-B0nrQE3U.js} +1 -1
- package/dist/assets/{reveal-component-BamXsTkr.js → reveal-component-CLjsFHmT.js} +1 -1
- package/dist/assets/{run-page-C5cZvltI.js → run-page-C1Wd8dr8.js} +1 -1
- package/dist/assets/{scratchpad-panel-PMFH4Ktp.js → scratchpad-panel-DpqODhZ3.js} +1 -1
- package/dist/assets/{session-panel-CUTSr9rt.js → session-panel-BvCIuxgU.js} +1 -1
- package/dist/assets/{slide-form-Bi8ZYBHP.js → slide-form-CXqvfwGg.js} +1 -1
- package/dist/assets/{snippets-panel-CdhDsAII.js → snippets-panel-COBTGJcr.js} +1 -1
- package/dist/assets/{state-C5AUgyZT.js → state-Bsz477is.js} +1 -1
- package/dist/assets/{state-DHlRrwyY.js → state-LyvUXW6A.js} +2 -2
- package/dist/assets/{textarea-BXPC1-kb.js → textarea-Dd0InTQJ.js} +1 -1
- package/dist/assets/{tracing-wSufGcF9.js → tracing-DjZGgWTo.js} +1 -1
- package/dist/assets/{tracing-panel-BO6MMAXa.js → tracing-panel-8eFzEjG0.js} +2 -2
- package/dist/assets/{useAddCell-CqBbGhrY.js → useAddCell-De9xD63_.js} +1 -1
- package/dist/assets/{useCellActionButton-DYwx4pGE.js → useCellActionButton-Bjxun3wK.js} +1 -1
- package/dist/assets/{useDeleteCell-BJlGbsbV.js → useDeleteCell-C2Stbb6Q.js} +1 -1
- package/dist/assets/{useDependencyPanelTab-rkAIQ6zL.js → useDependencyPanelTab-DFZNQ3a8.js} +1 -1
- package/dist/assets/{useNotebookActions-Dga3qEHF.js → useNotebookActions-F5fGMe9N.js} +1 -1
- package/dist/assets/{useRunCells-B531RIUE.js → useRunCells-DU1tawR-.js} +1 -1
- package/dist/assets/{useSplitCell-BnuJW5uo.js → useSplitCell-BZPuhd8z.js} +1 -1
- package/dist/index.html +25 -25
- package/package.json +1 -1
- package/src/components/data-table/data-table.tsx +11 -2
- package/src/components/data-table/filter-by-values-picker.tsx +238 -0
- package/src/components/data-table/filter-pill-editor.tsx +470 -0
- package/src/components/data-table/filter-pills.tsx +177 -41
- package/src/components/ui/checkbox.tsx +8 -4
- package/src/components/ui/combobox.tsx +3 -0
- package/src/css/app/Cell.css +4 -0
- package/src/plugins/impl/DataTablePlugin.tsx +1 -0
- package/src/plugins/impl/data-frames/forms/__tests__/__snapshots__/form.test.tsx.snap +15 -15
- package/src/utils/sets.ts +13 -0
- package/dist/assets/JsonOutput-DXnOS_Hk.js +0 -49
- package/dist/assets/form-CPDlIjdV.js +0 -2
- package/dist/assets/index-BYLYJcAY.css +0 -2
package/dist/index.html
CHANGED
|
@@ -66,7 +66,7 @@
|
|
|
66
66
|
<marimo-server-token data-token="{{ server_token }}" hidden></marimo-server-token>
|
|
67
67
|
<!-- /TODO -->
|
|
68
68
|
<title>{{ title }}</title>
|
|
69
|
-
<script type="module" crossorigin src="./assets/index-
|
|
69
|
+
<script type="module" crossorigin src="./assets/index-BFivsT-R.js"></script>
|
|
70
70
|
<link rel="modulepreload" crossorigin href="./assets/preload-helper-D2MJg03u.js">
|
|
71
71
|
<link rel="modulepreload" crossorigin href="./assets/chunk-LvLJmgfZ.js">
|
|
72
72
|
<link rel="modulepreload" crossorigin href="./assets/react-Bj1aDYRI.js">
|
|
@@ -133,7 +133,7 @@
|
|
|
133
133
|
<link rel="modulepreload" crossorigin href="./assets/debounce-DhnxH9Rh.js">
|
|
134
134
|
<link rel="modulepreload" crossorigin href="./assets/database-zap-BUVn5HoR.js">
|
|
135
135
|
<link rel="modulepreload" crossorigin href="./assets/main-B0OX4z33.js">
|
|
136
|
-
<link rel="modulepreload" crossorigin href="./assets/cells-
|
|
136
|
+
<link rel="modulepreload" crossorigin href="./assets/cells-A-ljPpuj.js">
|
|
137
137
|
<link rel="modulepreload" crossorigin href="./assets/ErrorBoundary-DzYV_VeY.js">
|
|
138
138
|
<link rel="modulepreload" crossorigin href="./assets/kbd-CGShmG7L.js">
|
|
139
139
|
<link rel="modulepreload" crossorigin href="./assets/useInstallPackage-afgh7PwB.js">
|
|
@@ -147,33 +147,33 @@
|
|
|
147
147
|
<link rel="modulepreload" crossorigin href="./assets/usePress-CtfZXGno.js">
|
|
148
148
|
<link rel="modulepreload" crossorigin href="./assets/input-DyPS_GiK.js">
|
|
149
149
|
<link rel="modulepreload" crossorigin href="./assets/ImperativeModal-BBqcKmmk.js">
|
|
150
|
-
<link rel="modulepreload" crossorigin href="./assets/cell-link-
|
|
150
|
+
<link rel="modulepreload" crossorigin href="./assets/cell-link-d38hFEon.js">
|
|
151
151
|
<link rel="modulepreload" crossorigin href="./assets/multi-map-rafH3cg3.js">
|
|
152
152
|
<link rel="modulepreload" crossorigin href="./assets/alert-cDholnQq.js">
|
|
153
153
|
<link rel="modulepreload" crossorigin href="./assets/chevron-right-CG5QYXYk.js">
|
|
154
154
|
<link rel="modulepreload" crossorigin href="./assets/dropdown-menu-D1A3cFC8.js">
|
|
155
155
|
<link rel="modulepreload" crossorigin href="./assets/links-C-rLiK3d.js">
|
|
156
|
-
<link rel="modulepreload" crossorigin href="./assets/useRunCells-
|
|
156
|
+
<link rel="modulepreload" crossorigin href="./assets/useRunCells-DU1tawR-.js">
|
|
157
157
|
<link rel="modulepreload" crossorigin href="./assets/copy-B5qyZn5s.js">
|
|
158
158
|
<link rel="modulepreload" crossorigin href="./assets/copy-Dk_3y0H-.js">
|
|
159
159
|
<link rel="modulepreload" crossorigin href="./assets/copy-icon-C67c9EwB.js">
|
|
160
|
-
<link rel="modulepreload" crossorigin href="./assets/RenderHTML-
|
|
161
|
-
<link rel="modulepreload" crossorigin href="./assets/datasource-
|
|
162
|
-
<link rel="modulepreload" crossorigin href="./assets/state-
|
|
160
|
+
<link rel="modulepreload" crossorigin href="./assets/RenderHTML-DR6lqJGS.js">
|
|
161
|
+
<link rel="modulepreload" crossorigin href="./assets/datasource-BCnGIjqn.js">
|
|
162
|
+
<link rel="modulepreload" crossorigin href="./assets/state-Bsz477is.js">
|
|
163
163
|
<link rel="modulepreload" crossorigin href="./assets/sparkles-CZ5WmLPA.js">
|
|
164
|
-
<link rel="modulepreload" crossorigin href="./assets/MarimoErrorOutput-
|
|
164
|
+
<link rel="modulepreload" crossorigin href="./assets/MarimoErrorOutput-B9_RO5eX.js">
|
|
165
165
|
<link rel="modulepreload" crossorigin href="./assets/spinner-Bhir8k53.js">
|
|
166
|
-
<link rel="modulepreload" crossorigin href="./assets/html-to-image-
|
|
167
|
-
<link rel="modulepreload" crossorigin href="./assets/focus-
|
|
166
|
+
<link rel="modulepreload" crossorigin href="./assets/html-to-image-tbPEMBJg.js">
|
|
167
|
+
<link rel="modulepreload" crossorigin href="./assets/focus-Baz7rlo0.js">
|
|
168
168
|
<link rel="modulepreload" crossorigin href="./assets/useAsyncData-dS4Ne6pU.js">
|
|
169
169
|
<link rel="modulepreload" crossorigin href="./assets/LazyAnyLanguageCodeMirror-0gZO1asr.js">
|
|
170
170
|
<link rel="modulepreload" crossorigin href="./assets/micromark-factory-space-WzovnJik.js">
|
|
171
171
|
<link rel="modulepreload" crossorigin href="./assets/chunk-5FQGJX7Z-B4RvNAGm.js">
|
|
172
|
-
<link rel="modulepreload" crossorigin href="./assets/markdown-renderer-
|
|
173
|
-
<link rel="modulepreload" crossorigin href="./assets/command-
|
|
172
|
+
<link rel="modulepreload" crossorigin href="./assets/markdown-renderer-CFrgiLqu.js">
|
|
173
|
+
<link rel="modulepreload" crossorigin href="./assets/command-CzIDL1VI.js">
|
|
174
174
|
<link rel="modulepreload" crossorigin href="./assets/popover-UExmgBsf.js">
|
|
175
175
|
<link rel="modulepreload" crossorigin href="./assets/errors-CZb6hI2x.js">
|
|
176
|
-
<link rel="modulepreload" crossorigin href="./assets/download-
|
|
176
|
+
<link rel="modulepreload" crossorigin href="./assets/download-Iyng0xCz.js">
|
|
177
177
|
<link rel="modulepreload" crossorigin href="./assets/table-23NNA1s9.js">
|
|
178
178
|
<link rel="modulepreload" crossorigin href="./assets/useIframeCapabilities-CryCjoyY.js">
|
|
179
179
|
<link rel="modulepreload" crossorigin href="./assets/error-banner-CTBwdcdk.js">
|
|
@@ -189,28 +189,29 @@
|
|
|
189
189
|
<link rel="modulepreload" crossorigin href="./assets/table-6NxtGaCm.js">
|
|
190
190
|
<link rel="modulepreload" crossorigin href="./assets/ellipsis-pk06Lq82.js">
|
|
191
191
|
<link rel="modulepreload" crossorigin href="./assets/message-circle-CWm2KnSx.js">
|
|
192
|
+
<link rel="modulepreload" crossorigin href="./assets/trash-2-D280Xiwg.js">
|
|
192
193
|
<link rel="modulepreload" crossorigin href="./assets/react-resizable-panels.browser.esm-Bcm5njwd.js">
|
|
193
|
-
<link rel="modulepreload" crossorigin href="./assets/JsonOutput-
|
|
194
|
+
<link rel="modulepreload" crossorigin href="./assets/JsonOutput-BVAcY5xS.js">
|
|
194
195
|
<link rel="modulepreload" crossorigin href="./assets/chart-no-axes-column-nqk474t8.js">
|
|
195
196
|
<link rel="modulepreload" crossorigin href="./assets/square-function-D1dlJvD8.js">
|
|
196
197
|
<link rel="modulepreload" crossorigin href="./assets/spec-Cq8FVoTf.js">
|
|
197
198
|
<link rel="modulepreload" crossorigin href="./assets/ellipsis-vertical-gxgNyR5G.js">
|
|
198
199
|
<link rel="modulepreload" crossorigin href="./assets/refresh-cw-DHwG4Mac.js">
|
|
199
200
|
<link rel="modulepreload" crossorigin href="./assets/tree-actions-oMCx6WNc.js">
|
|
200
|
-
<link rel="modulepreload" crossorigin href="./assets/components-
|
|
201
|
-
<link rel="modulepreload" crossorigin href="./assets/column-preview-
|
|
201
|
+
<link rel="modulepreload" crossorigin href="./assets/components-FepcpFGL.js">
|
|
202
|
+
<link rel="modulepreload" crossorigin href="./assets/column-preview-_5uYiy74.js">
|
|
202
203
|
<link rel="modulepreload" crossorigin href="./assets/icons-BRopQwI3.js">
|
|
203
|
-
<link rel="modulepreload" crossorigin href="./assets/floating-outline-
|
|
204
|
-
<link rel="modulepreload" crossorigin href="./assets/useAddCell-
|
|
204
|
+
<link rel="modulepreload" crossorigin href="./assets/floating-outline-CSNGDMAm.js">
|
|
205
|
+
<link rel="modulepreload" crossorigin href="./assets/useAddCell-De9xD63_.js">
|
|
205
206
|
<link rel="modulepreload" crossorigin href="./assets/objectWithoutPropertiesLoose-smPWkHxB.js">
|
|
206
207
|
<link rel="modulepreload" crossorigin href="./assets/esm-G1JMtvxe.js">
|
|
207
208
|
<link rel="modulepreload" crossorigin href="./assets/eye-off-BT-KOYV5.js">
|
|
208
209
|
<link rel="modulepreload" crossorigin href="./assets/plus-BgB18UzY.js">
|
|
209
|
-
<link rel="modulepreload" crossorigin href="./assets/readonly-python-code-
|
|
210
|
+
<link rel="modulepreload" crossorigin href="./assets/readonly-python-code-B0nrQE3U.js">
|
|
210
211
|
<link rel="modulepreload" crossorigin href="./assets/file-headphone-B3fuktN0.js">
|
|
211
212
|
<link rel="modulepreload" crossorigin href="./assets/file-HTLbeC2b.js">
|
|
212
213
|
<link rel="modulepreload" crossorigin href="./assets/image-DQHXdEQn.js">
|
|
213
|
-
<link rel="modulepreload" crossorigin href="./assets/file-icons-
|
|
214
|
+
<link rel="modulepreload" crossorigin href="./assets/file-icons-DTNZv05h.js">
|
|
214
215
|
<link rel="modulepreload" crossorigin href="./assets/switch-DlgpDZMk.js">
|
|
215
216
|
<link rel="modulepreload" crossorigin href="./assets/events-Qeh-bHlj.js">
|
|
216
217
|
<link rel="modulepreload" crossorigin href="./assets/globals-xVhZei_S.js">
|
|
@@ -219,13 +220,12 @@
|
|
|
219
220
|
<link rel="modulepreload" crossorigin href="./assets/memoize-Tp7rARFe.js">
|
|
220
221
|
<link rel="modulepreload" crossorigin href="./assets/get-C-qh_et5.js">
|
|
221
222
|
<link rel="modulepreload" crossorigin href="./assets/_baseSet-CxV9N1bc.js">
|
|
222
|
-
<link rel="modulepreload" crossorigin href="./assets/state-
|
|
223
|
+
<link rel="modulepreload" crossorigin href="./assets/state-LyvUXW6A.js">
|
|
223
224
|
<link rel="modulepreload" crossorigin href="./assets/label-DTR8T0AE.js">
|
|
224
|
-
<link rel="modulepreload" crossorigin href="./assets/textarea-
|
|
225
|
+
<link rel="modulepreload" crossorigin href="./assets/textarea-Dd0InTQJ.js">
|
|
225
226
|
<link rel="modulepreload" crossorigin href="./assets/radio-group-BtBoRbGH.js">
|
|
226
227
|
<link rel="modulepreload" crossorigin href="./assets/refresh-ccw-C-n2VFP5.js">
|
|
227
|
-
<link rel="modulepreload" crossorigin href="./assets/
|
|
228
|
-
<link rel="modulepreload" crossorigin href="./assets/form-CPDlIjdV.js">
|
|
228
|
+
<link rel="modulepreload" crossorigin href="./assets/form-Dr-_EXr8.js">
|
|
229
229
|
<link rel="modulepreload" crossorigin href="./assets/renderShortcut-CkNNAheg.js">
|
|
230
230
|
<link rel="modulepreload" crossorigin href="./assets/field-B8kJgr2A.js">
|
|
231
231
|
<link rel="modulepreload" crossorigin href="./assets/RSPContexts-BeHIgT4C.js">
|
|
@@ -246,7 +246,7 @@
|
|
|
246
246
|
<link rel="stylesheet" crossorigin href="./assets/cells-jmgGt1lS.css">
|
|
247
247
|
<link rel="stylesheet" crossorigin href="./assets/markdown-renderer-DdDKmWlR.css">
|
|
248
248
|
<link rel="stylesheet" crossorigin href="./assets/JsonOutput-B7vuddcd.css">
|
|
249
|
-
<link rel="stylesheet" crossorigin href="./assets/index-
|
|
249
|
+
<link rel="stylesheet" crossorigin href="./assets/index-DH5UUBE4.css">
|
|
250
250
|
</head>
|
|
251
251
|
<body>
|
|
252
252
|
<div id="root"></div>
|
package/package.json
CHANGED
|
@@ -22,7 +22,10 @@ import React, { memo } from "react";
|
|
|
22
22
|
import { useLocale } from "react-aria";
|
|
23
23
|
|
|
24
24
|
import { Table } from "@/components/ui/table";
|
|
25
|
-
import type {
|
|
25
|
+
import type {
|
|
26
|
+
CalculateTopKRows,
|
|
27
|
+
GetRowIds,
|
|
28
|
+
} from "@/plugins/impl/DataTablePlugin";
|
|
26
29
|
import { cn } from "@/utils/cn";
|
|
27
30
|
import {
|
|
28
31
|
PANEL_TYPES,
|
|
@@ -89,6 +92,7 @@ interface DataTableProps<TData> extends Partial<ExportActionProps> {
|
|
|
89
92
|
showFilters?: boolean;
|
|
90
93
|
filters?: ColumnFiltersState;
|
|
91
94
|
onFiltersChange?: OnChangeFn<ColumnFiltersState>;
|
|
95
|
+
calculateTopKRows?: CalculateTopKRows;
|
|
92
96
|
reloading?: boolean;
|
|
93
97
|
// Columns
|
|
94
98
|
freezeColumnsLeft?: string[];
|
|
@@ -139,6 +143,7 @@ const DataTableInternal = <TData,>({
|
|
|
139
143
|
showFilters = false,
|
|
140
144
|
filters,
|
|
141
145
|
onFiltersChange,
|
|
146
|
+
calculateTopKRows,
|
|
142
147
|
reloading,
|
|
143
148
|
freezeColumnsLeft,
|
|
144
149
|
freezeColumnsRight,
|
|
@@ -282,7 +287,11 @@ const DataTableInternal = <TData,>({
|
|
|
282
287
|
|
|
283
288
|
return (
|
|
284
289
|
<div className={cn(wrapperClassName, "flex flex-col space-y-1")}>
|
|
285
|
-
<FilterPills
|
|
290
|
+
<FilterPills
|
|
291
|
+
filters={filters}
|
|
292
|
+
table={table}
|
|
293
|
+
calculateTopKRows={calculateTopKRows}
|
|
294
|
+
/>
|
|
286
295
|
<CellSelectionProvider>
|
|
287
296
|
<div
|
|
288
297
|
part="table-wrapper"
|
|
@@ -0,0 +1,238 @@
|
|
|
1
|
+
/* Copyright 2026 Marimo. All rights reserved. */
|
|
2
|
+
"use no memo";
|
|
3
|
+
|
|
4
|
+
import type { Column } from "@tanstack/react-table";
|
|
5
|
+
import { ChevronDownIcon } from "lucide-react";
|
|
6
|
+
import { useMemo, useState } from "react";
|
|
7
|
+
import { useAsyncData } from "@/hooks/useAsyncData";
|
|
8
|
+
import { ErrorBanner } from "@/plugins/impl/common/error-banner";
|
|
9
|
+
import type { CalculateTopKRows } from "@/plugins/impl/DataTablePlugin";
|
|
10
|
+
import { cn } from "@/utils/cn";
|
|
11
|
+
import { Logger } from "@/utils/Logger";
|
|
12
|
+
import { Sets } from "@/utils/sets";
|
|
13
|
+
import { smartMatch } from "@/utils/smartMatch";
|
|
14
|
+
import { Spinner } from "../icons/spinner";
|
|
15
|
+
import { Button } from "../ui/button";
|
|
16
|
+
import { Checkbox } from "../ui/checkbox";
|
|
17
|
+
import {
|
|
18
|
+
Command,
|
|
19
|
+
CommandEmpty,
|
|
20
|
+
CommandInput,
|
|
21
|
+
CommandItem,
|
|
22
|
+
CommandList,
|
|
23
|
+
} from "../ui/command";
|
|
24
|
+
import { Popover, PopoverContent, PopoverTrigger } from "../ui/popover";
|
|
25
|
+
import { SentinelCell } from "./sentinel-cell";
|
|
26
|
+
import { detectSentinel, stringifyUnknownValue } from "./utils";
|
|
27
|
+
|
|
28
|
+
const TOP_K_ROWS = 30;
|
|
29
|
+
|
|
30
|
+
interface Props<TData, TValue> {
|
|
31
|
+
column: Column<TData, TValue>;
|
|
32
|
+
calculateTopKRows?: CalculateTopKRows;
|
|
33
|
+
chosenValues: unknown[];
|
|
34
|
+
onChange: (values: unknown[]) => void;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export const FilterByValuesPicker = <TData, TValue>({
|
|
38
|
+
column,
|
|
39
|
+
calculateTopKRows,
|
|
40
|
+
chosenValues,
|
|
41
|
+
onChange,
|
|
42
|
+
}: Props<TData, TValue>) => {
|
|
43
|
+
const [open, setOpen] = useState(false);
|
|
44
|
+
|
|
45
|
+
const chosenValuesSet = useMemo(() => new Set(chosenValues), [chosenValues]);
|
|
46
|
+
|
|
47
|
+
const selectedValuesStr = useMemo(() => {
|
|
48
|
+
if (chosenValuesSet.size === 0) {
|
|
49
|
+
return "Select values…";
|
|
50
|
+
}
|
|
51
|
+
const items = [...chosenValuesSet].map((v) =>
|
|
52
|
+
stringifyUnknownValue({ value: v }),
|
|
53
|
+
);
|
|
54
|
+
return `[${items.join(", ")}]`;
|
|
55
|
+
}, [chosenValuesSet]);
|
|
56
|
+
|
|
57
|
+
return (
|
|
58
|
+
<Popover open={open} onOpenChange={setOpen}>
|
|
59
|
+
<PopoverTrigger asChild={true}>
|
|
60
|
+
<Button
|
|
61
|
+
variant="outline"
|
|
62
|
+
size="xs"
|
|
63
|
+
className="h-6 mb-1 w-full justify-between font-normal"
|
|
64
|
+
>
|
|
65
|
+
<span
|
|
66
|
+
className={cn(
|
|
67
|
+
"truncate",
|
|
68
|
+
chosenValuesSet.size === 0 && "text-muted-foreground",
|
|
69
|
+
)}
|
|
70
|
+
>
|
|
71
|
+
{selectedValuesStr}
|
|
72
|
+
</span>
|
|
73
|
+
<ChevronDownIcon className="h-4 w-4 opacity-50 shrink-0" />
|
|
74
|
+
</Button>
|
|
75
|
+
</PopoverTrigger>
|
|
76
|
+
<PopoverContent className="w-80 p-0">
|
|
77
|
+
<PickerBody
|
|
78
|
+
column={column}
|
|
79
|
+
calculateTopKRows={calculateTopKRows}
|
|
80
|
+
chosenValues={chosenValuesSet}
|
|
81
|
+
onChange={onChange}
|
|
82
|
+
/>
|
|
83
|
+
</PopoverContent>
|
|
84
|
+
</Popover>
|
|
85
|
+
);
|
|
86
|
+
};
|
|
87
|
+
|
|
88
|
+
interface PickerBodyProps<TData, TValue> {
|
|
89
|
+
column: Column<TData, TValue>;
|
|
90
|
+
calculateTopKRows?: CalculateTopKRows;
|
|
91
|
+
chosenValues: Set<unknown>;
|
|
92
|
+
onChange: (values: unknown[]) => void;
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
const PickerBody = <TData, TValue>({
|
|
96
|
+
column,
|
|
97
|
+
calculateTopKRows,
|
|
98
|
+
chosenValues,
|
|
99
|
+
onChange,
|
|
100
|
+
}: PickerBodyProps<TData, TValue>) => {
|
|
101
|
+
const [query, setQuery] = useState<string>("");
|
|
102
|
+
|
|
103
|
+
const { data, isPending, error } = useAsyncData(async () => {
|
|
104
|
+
if (!calculateTopKRows) {
|
|
105
|
+
return null;
|
|
106
|
+
}
|
|
107
|
+
const res = await calculateTopKRows({ column: column.id, k: TOP_K_ROWS });
|
|
108
|
+
return res.data;
|
|
109
|
+
}, [calculateTopKRows, column.id]);
|
|
110
|
+
|
|
111
|
+
const filteredData = useMemo(() => {
|
|
112
|
+
if (!data) {
|
|
113
|
+
return [];
|
|
114
|
+
}
|
|
115
|
+
try {
|
|
116
|
+
// try to do includes and also smart match for prefixes
|
|
117
|
+
return data.filter(([value, _count]) => {
|
|
118
|
+
if (value === undefined) {
|
|
119
|
+
return false;
|
|
120
|
+
}
|
|
121
|
+
const str = String(value);
|
|
122
|
+
return (
|
|
123
|
+
smartMatch(query, str) ||
|
|
124
|
+
str.toLowerCase().includes(query.toLowerCase())
|
|
125
|
+
);
|
|
126
|
+
});
|
|
127
|
+
} catch (error_) {
|
|
128
|
+
Logger.error("Error filtering data", error_);
|
|
129
|
+
return [];
|
|
130
|
+
}
|
|
131
|
+
}, [data, query]);
|
|
132
|
+
|
|
133
|
+
const handleToggle = (value: unknown) => {
|
|
134
|
+
onChange([...Sets.toggle(chosenValues, value)]);
|
|
135
|
+
};
|
|
136
|
+
|
|
137
|
+
const allVisibleChecked =
|
|
138
|
+
filteredData.length > 0 &&
|
|
139
|
+
filteredData.every(([value]) => chosenValues.has(value));
|
|
140
|
+
|
|
141
|
+
const selectAllState: boolean | "indeterminate" = allVisibleChecked
|
|
142
|
+
? true
|
|
143
|
+
: chosenValues.size > 0
|
|
144
|
+
? "indeterminate"
|
|
145
|
+
: false;
|
|
146
|
+
|
|
147
|
+
const handleToggleAll = () => {
|
|
148
|
+
if (!data) {
|
|
149
|
+
return;
|
|
150
|
+
}
|
|
151
|
+
const next = new Set(chosenValues);
|
|
152
|
+
if (allVisibleChecked) {
|
|
153
|
+
for (const [value] of filteredData) {
|
|
154
|
+
next.delete(value);
|
|
155
|
+
}
|
|
156
|
+
} else {
|
|
157
|
+
for (const [value] of filteredData) {
|
|
158
|
+
next.add(value);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
onChange([...next]);
|
|
162
|
+
};
|
|
163
|
+
|
|
164
|
+
if (isPending) {
|
|
165
|
+
return <Spinner size="medium" className="mx-auto mt-12 mb-10" />;
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
if (error) {
|
|
169
|
+
return <ErrorBanner error={error} className="my-10 mx-4" />;
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
if (!data) {
|
|
173
|
+
return (
|
|
174
|
+
<div className="py-6 px-4 text-sm text-muted-foreground text-center">
|
|
175
|
+
No values available
|
|
176
|
+
</div>
|
|
177
|
+
);
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
return (
|
|
181
|
+
<Command className="text-sm outline-hidden" shouldFilter={false}>
|
|
182
|
+
<CommandInput
|
|
183
|
+
placeholder={`Search among the top ${data.length} values`}
|
|
184
|
+
autoFocus={true}
|
|
185
|
+
onValueChange={(value) => setQuery(value.trim())}
|
|
186
|
+
/>
|
|
187
|
+
<CommandEmpty>No results found.</CommandEmpty>
|
|
188
|
+
<CommandList>
|
|
189
|
+
{filteredData.length > 0 && (
|
|
190
|
+
<CommandItem
|
|
191
|
+
value="__select-all__"
|
|
192
|
+
className="border-b rounded-none px-3"
|
|
193
|
+
onSelect={handleToggleAll}
|
|
194
|
+
>
|
|
195
|
+
<Checkbox
|
|
196
|
+
checked={selectAllState}
|
|
197
|
+
aria-label="Select all"
|
|
198
|
+
className="mr-3 h-3.5 w-3.5"
|
|
199
|
+
/>
|
|
200
|
+
<span className="font-bold flex-1">{column.id}</span>
|
|
201
|
+
<span className="font-bold">Count</span>
|
|
202
|
+
</CommandItem>
|
|
203
|
+
)}
|
|
204
|
+
{filteredData.map(([value, count]) => {
|
|
205
|
+
const isSelected = chosenValues.has(value);
|
|
206
|
+
const valueString = stringifyUnknownValue({ value });
|
|
207
|
+
const sentinel = detectSentinel(
|
|
208
|
+
value,
|
|
209
|
+
column.columnDef.meta?.dataType,
|
|
210
|
+
);
|
|
211
|
+
return (
|
|
212
|
+
<CommandItem
|
|
213
|
+
key={valueString}
|
|
214
|
+
value={valueString}
|
|
215
|
+
className="not-last:border-b rounded-none px-3"
|
|
216
|
+
onSelect={() => handleToggle(value)}
|
|
217
|
+
>
|
|
218
|
+
<Checkbox
|
|
219
|
+
checked={isSelected}
|
|
220
|
+
aria-label="Select row"
|
|
221
|
+
className="mr-3 h-3.5 w-3.5"
|
|
222
|
+
/>
|
|
223
|
+
<span className="flex-1 overflow-hidden max-h-20 line-clamp-3">
|
|
224
|
+
{sentinel ? <SentinelCell sentinel={sentinel} /> : valueString}
|
|
225
|
+
</span>
|
|
226
|
+
<span className="ml-3">{count}</span>
|
|
227
|
+
</CommandItem>
|
|
228
|
+
);
|
|
229
|
+
})}
|
|
230
|
+
</CommandList>
|
|
231
|
+
{data.length === TOP_K_ROWS && (
|
|
232
|
+
<span className="text-xs text-muted-foreground py-1.5 text-center">
|
|
233
|
+
Only showing the top {TOP_K_ROWS} values
|
|
234
|
+
</span>
|
|
235
|
+
)}
|
|
236
|
+
</Command>
|
|
237
|
+
);
|
|
238
|
+
};
|