@stackframe/dashboard-ui-components 2.8.89 → 2.8.92

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.
Files changed (67) hide show
  1. package/dist/components/chart-legend.d.ts +1 -1
  2. package/dist/components/data-grid/data-grid-sizing.d.ts +6 -5
  3. package/dist/components/data-grid/data-grid-sizing.d.ts.map +1 -1
  4. package/dist/components/data-grid/data-grid-sizing.js +9 -28
  5. package/dist/components/data-grid/data-grid-sizing.js.map +1 -1
  6. package/dist/components/data-grid/data-grid.d.ts +17 -237
  7. package/dist/components/data-grid/data-grid.d.ts.map +1 -1
  8. package/dist/components/data-grid/data-grid.js +377 -523
  9. package/dist/components/data-grid/data-grid.js.map +1 -1
  10. package/dist/components/data-grid/data-grid.test.js +82 -0
  11. package/dist/components/data-grid/data-grid.test.js.map +1 -1
  12. package/dist/components/data-grid/index.d.ts +4 -3
  13. package/dist/components/data-grid/index.js +17 -58
  14. package/dist/components/data-grid/state.d.ts +4 -61
  15. package/dist/components/data-grid/state.d.ts.map +1 -1
  16. package/dist/components/data-grid/state.js +13 -160
  17. package/dist/components/data-grid/state.js.map +1 -1
  18. package/dist/components/data-grid/types.d.ts +9 -4
  19. package/dist/components/data-grid/types.d.ts.map +1 -1
  20. package/dist/components/data-grid/use-url-state.d.ts +38 -0
  21. package/dist/components/data-grid/use-url-state.d.ts.map +1 -0
  22. package/dist/components/data-grid/use-url-state.js +214 -0
  23. package/dist/components/data-grid/use-url-state.js.map +1 -0
  24. package/dist/components/data-grid/use-url-state.test.d.ts +1 -0
  25. package/dist/components/data-grid/use-url-state.test.js +91 -0
  26. package/dist/components/data-grid/use-url-state.test.js.map +1 -0
  27. package/dist/components/dialog.d.ts +67 -0
  28. package/dist/components/dialog.d.ts.map +1 -0
  29. package/dist/components/dialog.js +100 -0
  30. package/dist/components/dialog.js.map +1 -0
  31. package/dist/dashboard-ui-components.global.js +10651 -6388
  32. package/dist/dashboard-ui-components.global.js.map +4 -4
  33. package/dist/esm/components/chart-legend.d.ts +1 -1
  34. package/dist/esm/components/data-grid/data-grid-sizing.d.ts +6 -5
  35. package/dist/esm/components/data-grid/data-grid-sizing.d.ts.map +1 -1
  36. package/dist/esm/components/data-grid/data-grid-sizing.js +7 -26
  37. package/dist/esm/components/data-grid/data-grid-sizing.js.map +1 -1
  38. package/dist/esm/components/data-grid/data-grid.d.ts +17 -237
  39. package/dist/esm/components/data-grid/data-grid.d.ts.map +1 -1
  40. package/dist/esm/components/data-grid/data-grid.js +380 -526
  41. package/dist/esm/components/data-grid/data-grid.js.map +1 -1
  42. package/dist/esm/components/data-grid/data-grid.test.js +82 -0
  43. package/dist/esm/components/data-grid/data-grid.test.js.map +1 -1
  44. package/dist/esm/components/data-grid/index.d.ts +4 -3
  45. package/dist/esm/components/data-grid/index.js +4 -3
  46. package/dist/esm/components/data-grid/state.d.ts +4 -61
  47. package/dist/esm/components/data-grid/state.d.ts.map +1 -1
  48. package/dist/esm/components/data-grid/state.js +15 -150
  49. package/dist/esm/components/data-grid/state.js.map +1 -1
  50. package/dist/esm/components/data-grid/types.d.ts +9 -4
  51. package/dist/esm/components/data-grid/types.d.ts.map +1 -1
  52. package/dist/esm/components/data-grid/use-url-state.d.ts +38 -0
  53. package/dist/esm/components/data-grid/use-url-state.d.ts.map +1 -0
  54. package/dist/esm/components/data-grid/use-url-state.js +212 -0
  55. package/dist/esm/components/data-grid/use-url-state.js.map +1 -0
  56. package/dist/esm/components/data-grid/use-url-state.test.d.ts +1 -0
  57. package/dist/esm/components/data-grid/use-url-state.test.js +91 -0
  58. package/dist/esm/components/data-grid/use-url-state.test.js.map +1 -0
  59. package/dist/esm/components/dialog.d.ts +67 -0
  60. package/dist/esm/components/dialog.d.ts.map +1 -0
  61. package/dist/esm/components/dialog.js +92 -0
  62. package/dist/esm/components/dialog.js.map +1 -0
  63. package/dist/esm/index.d.ts +2 -1
  64. package/dist/esm/index.js +2 -1
  65. package/dist/index.d.ts +5 -3
  66. package/dist/index.js +37 -0
  67. package/package.json +4 -3
@@ -0,0 +1,38 @@
1
+ import { DataGridColumnDef, DataGridState } from "./types.js";
2
+
3
+ //#region src/components/data-grid/use-url-state.d.ts
4
+ type UrlStateOptions = {
5
+ /** Disambiguates URL params when multiple grids share a page. Defaults
6
+ * to `"grid"`. Use unique prefixes per-grid (e.g. `"users"`, `"teams"`). */
7
+ paramPrefix?: string;
8
+ /** Overrides for default state used when the URL has no value for a
9
+ * given key. Useful for things like a sensible initial sort (e.g.
10
+ * "newest signups first") that should appear on first load but be
11
+ * overridden when the user navigates to a bookmarked URL. */
12
+ initial?: Partial<Pick<DataGridState, "sorting" | "quickSearch" | "columnVisibility">>;
13
+ };
14
+ /**
15
+ * Drop-in replacement for `useState(() => createDefaultDataGridState(columns))`
16
+ * that persists user view preferences to URL search params, so a view
17
+ * can be bookmarked / shared / restored on reload.
18
+ *
19
+ * **Persisted:** column widths, hidden columns, sort model, quick-search.
20
+ * **Not persisted** (deliberately): pagination scroll position, selection,
21
+ * column pinning/order, date display mode — these are session-scoped.
22
+ *
23
+ * ```tsx
24
+ * const [gridState, setGridState] = useDataGridUrlState(columns, {
25
+ * paramPrefix: "users",
26
+ * initial: { sorting: [{ columnId: "signedUpAt", direction: "desc" }] },
27
+ * });
28
+ * ```
29
+ *
30
+ * URL encoding: `?{prefix}_w=...&{prefix}_h=...&{prefix}_s=...&{prefix}_q=...`.
31
+ * Default values are omitted so URLs stay clean. Updates use
32
+ * `history.replaceState` (not pushState) so back/forward isn't polluted,
33
+ * and `popstate` is observed so external URL changes flow back into state.
34
+ */
35
+ declare function useDataGridUrlState<TRow>(columns: readonly DataGridColumnDef<TRow>[], opts?: UrlStateOptions): [DataGridState, React.Dispatch<React.SetStateAction<DataGridState>>];
36
+ //#endregion
37
+ export { useDataGridUrlState };
38
+ //# sourceMappingURL=use-url-state.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-url-state.d.ts","names":[],"sources":["../../../src/components/data-grid/use-url-state.ts"],"mappings":";;;KAqJK,eAAA;;AA3IY;EA8If,WAAA;;;;;EAKA,OAAA,GAAU,OAAA,CAAQ,IAAA,CAAK,aAAA;AAAA;;;;;;;;AAwBzB;;;;;;;;;;;;;;iBAAgB,mBAAA,MAAA,CACd,OAAA,WAAkB,iBAAA,CAAkB,IAAA,KACpC,IAAA,GAAO,eAAA,IACL,aAAA,EAAe,KAAA,CAAM,QAAA,CAAS,KAAA,CAAM,cAAA,CAAe,aAAA"}
@@ -0,0 +1,214 @@
1
+ "use client";
2
+
3
+ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
4
+ const require_chunk = require('../../chunk-BE-pF4vm.js');
5
+ let react = require("react");
6
+ let __state_js = require("./state.js");
7
+ let __data_grid_sizing_js = require("./data-grid-sizing.js");
8
+
9
+ //#region src/components/data-grid/use-url-state.ts
10
+ function serializeWidths(widths, columns) {
11
+ const parts = [];
12
+ for (const col of columns) {
13
+ const w = widths[col.id];
14
+ if (typeof w !== "number" || !Number.isFinite(w)) continue;
15
+ const defaultW = (0, __data_grid_sizing_js.clampColumnWidth)(col, col.width ?? __data_grid_sizing_js.DEFAULT_COL_WIDTH);
16
+ if (Math.round(w) === Math.round(defaultW)) continue;
17
+ parts.push(`${encodeURIComponent(col.id)}:${Math.round(w)}`);
18
+ }
19
+ return parts.join(",");
20
+ }
21
+ function parseWidths(raw, fallback, columns) {
22
+ if (!raw) return fallback;
23
+ const colMap = new Map(columns.map((c) => [c.id, c]));
24
+ const out = { ...fallback };
25
+ for (const part of raw.split(",")) {
26
+ const colonIdx = part.indexOf(":");
27
+ if (colonIdx <= 0) continue;
28
+ const encodedId = part.slice(0, colonIdx);
29
+ const num = part.slice(colonIdx + 1);
30
+ let id;
31
+ try {
32
+ id = decodeURIComponent(encodedId);
33
+ } catch {
34
+ continue;
35
+ }
36
+ if (!id || !num) continue;
37
+ const col = colMap.get(id);
38
+ if (!col) continue;
39
+ const n = Number(num);
40
+ if (!Number.isFinite(n)) continue;
41
+ out[id] = (0, __data_grid_sizing_js.clampColumnWidth)(col, n);
42
+ }
43
+ return out;
44
+ }
45
+ function serializeHidden(visibility) {
46
+ return Object.entries(visibility).filter(([, v]) => v === false).map(([id]) => encodeURIComponent(id)).join(",");
47
+ }
48
+ function parseHidden(raw, columns) {
49
+ if (!raw) return {};
50
+ const known = new Set(columns.map((c) => c.id));
51
+ const out = {};
52
+ for (const encodedId of raw.split(",")) {
53
+ let id;
54
+ try {
55
+ id = decodeURIComponent(encodedId);
56
+ } catch {
57
+ continue;
58
+ }
59
+ if (id && known.has(id)) out[id] = false;
60
+ }
61
+ return out;
62
+ }
63
+ function serializeSort(sort) {
64
+ return sort.map((s) => `${encodeURIComponent(s.columnId)}:${s.direction === "desc" ? "desc" : "asc"}`).join(",");
65
+ }
66
+ function sortEqual(a, b) {
67
+ if (a.length !== b.length) return false;
68
+ for (let i = 0; i < a.length; i++) if (a[i].columnId !== b[i].columnId || a[i].direction !== b[i].direction) return false;
69
+ return true;
70
+ }
71
+ function parseSort(raw, fallback, columns) {
72
+ if (raw == null) return fallback;
73
+ if (raw === "") return [];
74
+ const known = new Set(columns.map((c) => c.id));
75
+ const out = [];
76
+ for (const part of raw.split(",")) {
77
+ const colonIdx = part.indexOf(":");
78
+ if (colonIdx <= 0) continue;
79
+ const encodedId = part.slice(0, colonIdx);
80
+ const dir = part.slice(colonIdx + 1);
81
+ if (dir !== "asc" && dir !== "desc") continue;
82
+ let id;
83
+ try {
84
+ id = decodeURIComponent(encodedId);
85
+ } catch {
86
+ continue;
87
+ }
88
+ if (!id || !known.has(id)) continue;
89
+ out.push({
90
+ columnId: id,
91
+ direction: dir
92
+ });
93
+ }
94
+ return out;
95
+ }
96
+ function parseQuickSearch(raw, fallback) {
97
+ if (raw == null) return fallback;
98
+ return raw;
99
+ }
100
+ /**
101
+ * Drop-in replacement for `useState(() => createDefaultDataGridState(columns))`
102
+ * that persists user view preferences to URL search params, so a view
103
+ * can be bookmarked / shared / restored on reload.
104
+ *
105
+ * **Persisted:** column widths, hidden columns, sort model, quick-search.
106
+ * **Not persisted** (deliberately): pagination scroll position, selection,
107
+ * column pinning/order, date display mode — these are session-scoped.
108
+ *
109
+ * ```tsx
110
+ * const [gridState, setGridState] = useDataGridUrlState(columns, {
111
+ * paramPrefix: "users",
112
+ * initial: { sorting: [{ columnId: "signedUpAt", direction: "desc" }] },
113
+ * });
114
+ * ```
115
+ *
116
+ * URL encoding: `?{prefix}_w=...&{prefix}_h=...&{prefix}_s=...&{prefix}_q=...`.
117
+ * Default values are omitted so URLs stay clean. Updates use
118
+ * `history.replaceState` (not pushState) so back/forward isn't polluted,
119
+ * and `popstate` is observed so external URL changes flow back into state.
120
+ */
121
+ function useDataGridUrlState(columns, opts) {
122
+ const prefix = opts?.paramPrefix ?? "grid";
123
+ const widthsKey = `${prefix}_w`;
124
+ const hiddenKey = `${prefix}_h`;
125
+ const sortKey = `${prefix}_s`;
126
+ const searchKey = `${prefix}_q`;
127
+ const columnsRef = (0, react.useRef)(columns);
128
+ columnsRef.current = columns;
129
+ const initialRef = (0, react.useRef)(opts?.initial);
130
+ const [state, setState] = (0, react.useState)(() => {
131
+ const base = (0, __state_js.createDefaultDataGridState)(columns);
132
+ const initial = initialRef.current ?? {};
133
+ const baseWithInitial = {
134
+ ...base,
135
+ sorting: initial.sorting ?? base.sorting,
136
+ quickSearch: initial.quickSearch ?? base.quickSearch,
137
+ columnVisibility: initial.columnVisibility ?? base.columnVisibility
138
+ };
139
+ if (typeof window === "undefined") return baseWithInitial;
140
+ const params = new URLSearchParams(window.location.search);
141
+ return {
142
+ ...baseWithInitial,
143
+ columnWidths: parseWidths(params.get(widthsKey), base.columnWidths, columns),
144
+ columnVisibility: params.get(hiddenKey) != null ? parseHidden(params.get(hiddenKey), columns) : baseWithInitial.columnVisibility,
145
+ sorting: parseSort(params.get(sortKey), baseWithInitial.sorting, columns),
146
+ quickSearch: parseQuickSearch(params.get(searchKey), baseWithInitial.quickSearch)
147
+ };
148
+ });
149
+ (0, react.useEffect)(() => {
150
+ if (typeof window === "undefined") return;
151
+ const timer = setTimeout(() => {
152
+ const params = new URLSearchParams(window.location.search);
153
+ const before = params.toString();
154
+ const cols = columnsRef.current;
155
+ const initial = initialRef.current ?? {};
156
+ const widthsStr = serializeWidths(state.columnWidths, cols);
157
+ if (widthsStr) params.set(widthsKey, widthsStr);
158
+ else params.delete(widthsKey);
159
+ const hiddenStr = serializeHidden(state.columnVisibility);
160
+ const initialHidden = initial.columnVisibility ? serializeHidden(initial.columnVisibility) : "";
161
+ if (hiddenStr) params.set(hiddenKey, hiddenStr);
162
+ else if (initialHidden) params.set(hiddenKey, "");
163
+ else params.delete(hiddenKey);
164
+ const initialSort = initial.sorting ?? [];
165
+ if (sortEqual(state.sorting, initialSort)) params.delete(sortKey);
166
+ else params.set(sortKey, serializeSort(state.sorting));
167
+ const initialSearch = initial.quickSearch ?? "";
168
+ if (state.quickSearch === initialSearch) params.delete(searchKey);
169
+ else params.set(searchKey, state.quickSearch);
170
+ const after = params.toString();
171
+ if (before === after) return;
172
+ const url = `${window.location.pathname}${after ? `?${after}` : ""}${window.location.hash}`;
173
+ window.history.replaceState(window.history.state, "", url);
174
+ }, 100);
175
+ return () => clearTimeout(timer);
176
+ }, [
177
+ state.columnWidths,
178
+ state.columnVisibility,
179
+ state.sorting,
180
+ state.quickSearch,
181
+ widthsKey,
182
+ hiddenKey,
183
+ sortKey,
184
+ searchKey
185
+ ]);
186
+ (0, react.useEffect)(() => {
187
+ if (typeof window === "undefined") return;
188
+ const onPop = () => {
189
+ const params = new URLSearchParams(window.location.search);
190
+ const cols = columnsRef.current;
191
+ const initial = initialRef.current ?? {};
192
+ const defaults = (0, __state_js.createDefaultDataGridState)(cols);
193
+ setState((prev) => ({
194
+ ...prev,
195
+ columnWidths: parseWidths(params.get(widthsKey), defaults.columnWidths, cols),
196
+ columnVisibility: params.get(hiddenKey) != null ? parseHidden(params.get(hiddenKey), cols) : initial.columnVisibility ?? defaults.columnVisibility,
197
+ sorting: parseSort(params.get(sortKey), initial.sorting ?? defaults.sorting, cols),
198
+ quickSearch: parseQuickSearch(params.get(searchKey), initial.quickSearch ?? defaults.quickSearch)
199
+ }));
200
+ };
201
+ window.addEventListener("popstate", onPop);
202
+ return () => window.removeEventListener("popstate", onPop);
203
+ }, [
204
+ widthsKey,
205
+ hiddenKey,
206
+ sortKey,
207
+ searchKey
208
+ ]);
209
+ return [state, setState];
210
+ }
211
+
212
+ //#endregion
213
+ exports.useDataGridUrlState = useDataGridUrlState;
214
+ //# sourceMappingURL=use-url-state.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-url-state.js","names":["DEFAULT_COL_WIDTH"],"sources":["../../../src/components/data-grid/use-url-state.ts"],"sourcesContent":["\"use client\";\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { DEFAULT_COL_WIDTH, clampColumnWidth } from \"./data-grid-sizing\";\nimport { createDefaultDataGridState } from \"./state\";\nimport type {\n DataGridColumnDef,\n DataGridSortItem,\n DataGridSortModel,\n DataGridState,\n} from \"./types\";\n\n// ─── URL <-> state encoding ──────────────────────────────────────────\n// Compact, human-readable formats so URLs stay short. Each piece of\n// state gets its own param so unrelated changes don't churn shared keys:\n//\n// ?{prefix}_w=name:200,email:300 column widths (only non-defaults)\n// ?{prefix}_h=createdAt,role hidden column ids\n// ?{prefix}_s=signedUpAt:desc,name:asc multi-column sort\n// ?{prefix}_q=alice quick-search text\n//\n// Column ids are URL-encoded so ids containing `,` `:` or other reserved\n// characters round-trip safely. Without encoding, an id like \"user:name\"\n// would silently break the parser.\n\n// ── widths ─────────────────────────────────────────────────────────\nfunction serializeWidths(\n widths: Record<string, number>,\n columns: readonly DataGridColumnDef<any>[],\n): string {\n const parts: string[] = [];\n for (const col of columns) {\n const w = widths[col.id];\n if (typeof w !== \"number\" || !Number.isFinite(w)) continue;\n const defaultW = clampColumnWidth(col, col.width ?? DEFAULT_COL_WIDTH);\n if (Math.round(w) === Math.round(defaultW)) continue;\n parts.push(`${encodeURIComponent(col.id)}:${Math.round(w)}`);\n }\n return parts.join(\",\");\n}\n\nfunction parseWidths(\n raw: string | null,\n fallback: Record<string, number>,\n columns: readonly DataGridColumnDef<any>[],\n): Record<string, number> {\n if (!raw) return fallback;\n const colMap = new Map(columns.map((c) => [c.id, c]));\n const out: Record<string, number> = { ...fallback };\n for (const part of raw.split(\",\")) {\n // Only split on the FIRST colon — id-side is always pre-encoded so it\n // can't contain a literal `:`, but width-side is always numeric.\n const colonIdx = part.indexOf(\":\");\n if (colonIdx <= 0) continue;\n const encodedId = part.slice(0, colonIdx);\n const num = part.slice(colonIdx + 1);\n let id: string;\n try {\n id = decodeURIComponent(encodedId);\n } catch {\n continue;\n }\n if (!id || !num) continue;\n const col = colMap.get(id);\n if (!col) continue;\n const n = Number(num);\n if (!Number.isFinite(n)) continue;\n out[id] = clampColumnWidth(col, n);\n }\n return out;\n}\n\n// ── hidden columns ─────────────────────────────────────────────────\nfunction serializeHidden(visibility: Record<string, boolean>): string {\n return Object.entries(visibility)\n .filter(([, v]) => v === false)\n .map(([id]) => encodeURIComponent(id))\n .join(\",\");\n}\n\nfunction parseHidden(\n raw: string | null,\n columns: readonly DataGridColumnDef<any>[],\n): Record<string, boolean> {\n if (!raw) return {};\n const known = new Set(columns.map((c) => c.id));\n const out: Record<string, boolean> = {};\n for (const encodedId of raw.split(\",\")) {\n let id: string;\n try {\n id = decodeURIComponent(encodedId);\n } catch {\n continue;\n }\n if (id && known.has(id)) out[id] = false;\n }\n return out;\n}\n\n// ── sort model ─────────────────────────────────────────────────────\nfunction serializeSort(sort: DataGridSortModel): string {\n return sort\n .map((s) => `${encodeURIComponent(s.columnId)}:${s.direction === \"desc\" ? \"desc\" : \"asc\"}`)\n .join(\",\");\n}\n\nfunction sortEqual(a: DataGridSortModel, b: DataGridSortModel): boolean {\n if (a.length !== b.length) return false;\n for (let i = 0; i < a.length; i++) {\n if (a[i].columnId !== b[i].columnId || a[i].direction !== b[i].direction) return false;\n }\n return true;\n}\n\nfunction parseSort(\n raw: string | null,\n fallback: DataGridSortModel,\n columns: readonly DataGridColumnDef<any>[],\n): DataGridSortModel {\n if (raw == null) return fallback;\n if (raw === \"\") return [];\n const known = new Set(columns.map((c) => c.id));\n const out: DataGridSortItem[] = [];\n for (const part of raw.split(\",\")) {\n const colonIdx = part.indexOf(\":\");\n if (colonIdx <= 0) continue;\n const encodedId = part.slice(0, colonIdx);\n const dir = part.slice(colonIdx + 1);\n if (dir !== \"asc\" && dir !== \"desc\") continue;\n let id: string;\n try {\n id = decodeURIComponent(encodedId);\n } catch {\n continue;\n }\n if (!id || !known.has(id)) continue;\n out.push({ columnId: id, direction: dir });\n }\n return out;\n}\n\n// ── quick search ───────────────────────────────────────────────────\nfunction parseQuickSearch(raw: string | null, fallback: string): string {\n if (raw == null) return fallback;\n return raw;\n}\n\n// ─── Hook ────────────────────────────────────────────────────────────\n\ntype UrlStateOptions = {\n /** Disambiguates URL params when multiple grids share a page. Defaults\n * to `\"grid\"`. Use unique prefixes per-grid (e.g. `\"users\"`, `\"teams\"`). */\n paramPrefix?: string,\n /** Overrides for default state used when the URL has no value for a\n * given key. Useful for things like a sensible initial sort (e.g.\n * \"newest signups first\") that should appear on first load but be\n * overridden when the user navigates to a bookmarked URL. */\n initial?: Partial<Pick<DataGridState, \"sorting\" | \"quickSearch\" | \"columnVisibility\">>,\n};\n\n/**\n * Drop-in replacement for `useState(() => createDefaultDataGridState(columns))`\n * that persists user view preferences to URL search params, so a view\n * can be bookmarked / shared / restored on reload.\n *\n * **Persisted:** column widths, hidden columns, sort model, quick-search.\n * **Not persisted** (deliberately): pagination scroll position, selection,\n * column pinning/order, date display mode — these are session-scoped.\n *\n * ```tsx\n * const [gridState, setGridState] = useDataGridUrlState(columns, {\n * paramPrefix: \"users\",\n * initial: { sorting: [{ columnId: \"signedUpAt\", direction: \"desc\" }] },\n * });\n * ```\n *\n * URL encoding: `?{prefix}_w=...&{prefix}_h=...&{prefix}_s=...&{prefix}_q=...`.\n * Default values are omitted so URLs stay clean. Updates use\n * `history.replaceState` (not pushState) so back/forward isn't polluted,\n * and `popstate` is observed so external URL changes flow back into state.\n */\nexport function useDataGridUrlState<TRow>(\n columns: readonly DataGridColumnDef<TRow>[],\n opts?: UrlStateOptions,\n): [DataGridState, React.Dispatch<React.SetStateAction<DataGridState>>] {\n const prefix = opts?.paramPrefix ?? \"grid\";\n const widthsKey = `${prefix}_w`;\n const hiddenKey = `${prefix}_h`;\n const sortKey = `${prefix}_s`;\n const searchKey = `${prefix}_q`;\n\n const columnsRef = useRef(columns);\n columnsRef.current = columns;\n\n // `initial` snapshots are captured once on first render. Re-running them\n // each render would clobber the user's interactions.\n const initialRef = useRef(opts?.initial);\n\n const [state, setState] = useState<DataGridState>(() => {\n const base = createDefaultDataGridState(columns);\n const initial = initialRef.current ?? {};\n const baseWithInitial: DataGridState = {\n ...base,\n sorting: initial.sorting ?? base.sorting,\n quickSearch: initial.quickSearch ?? base.quickSearch,\n columnVisibility: initial.columnVisibility ?? base.columnVisibility,\n };\n if (typeof window === \"undefined\") return baseWithInitial;\n const params = new URLSearchParams(window.location.search);\n return {\n ...baseWithInitial,\n columnWidths: parseWidths(params.get(widthsKey), base.columnWidths, columns),\n columnVisibility: params.get(hiddenKey) != null\n ? parseHidden(params.get(hiddenKey), columns)\n : baseWithInitial.columnVisibility,\n sorting: parseSort(params.get(sortKey), baseWithInitial.sorting, columns),\n quickSearch: parseQuickSearch(params.get(searchKey), baseWithInitial.quickSearch),\n };\n });\n\n // Sync state -> URL. Debounced so that high-frequency state changes\n // (e.g. dragging a column resize handle, typing in the search box)\n // don't fire a URL write per pixel / per keystroke.\n useEffect(() => {\n if (typeof window === \"undefined\") return;\n const timer = setTimeout(() => {\n const params = new URLSearchParams(window.location.search);\n const before = params.toString();\n const cols = columnsRef.current;\n const initial = initialRef.current ?? {};\n\n const widthsStr = serializeWidths(state.columnWidths, cols);\n if (widthsStr) params.set(widthsKey, widthsStr);\n else params.delete(widthsKey);\n\n const hiddenStr = serializeHidden(state.columnVisibility);\n // If the consumer supplied initial visibility, \"no hidden cols\" must\n // be encoded explicitly (empty string) so a bookmark with no `_h=`\n // doesn't silently re-hide a column the user just un-hid.\n const initialHidden = initial.columnVisibility\n ? serializeHidden(initial.columnVisibility)\n : \"\";\n if (hiddenStr) params.set(hiddenKey, hiddenStr);\n else if (initialHidden) params.set(hiddenKey, \"\");\n else params.delete(hiddenKey);\n\n const initialSort = initial.sorting ?? [];\n if (sortEqual(state.sorting, initialSort)) params.delete(sortKey);\n else params.set(sortKey, serializeSort(state.sorting));\n\n const initialSearch = initial.quickSearch ?? \"\";\n if (state.quickSearch === initialSearch) params.delete(searchKey);\n else params.set(searchKey, state.quickSearch);\n\n const after = params.toString();\n if (before === after) return;\n const url = `${window.location.pathname}${after ? `?${after}` : \"\"}${window.location.hash}`;\n window.history.replaceState(window.history.state, \"\", url);\n }, 100);\n return () => clearTimeout(timer);\n }, [\n state.columnWidths,\n state.columnVisibility,\n state.sorting,\n state.quickSearch,\n widthsKey,\n hiddenKey,\n sortKey,\n searchKey,\n ]);\n\n // React to back/forward navigation.\n useEffect(() => {\n if (typeof window === \"undefined\") return;\n const onPop = () => {\n const params = new URLSearchParams(window.location.search);\n const cols = columnsRef.current;\n const initial = initialRef.current ?? {};\n // When the URL no longer has a value for a key, reset to defaults\n // rather than preserving the previous in-memory state — otherwise\n // navigating back to a clean URL leaves stale state.\n const defaults = createDefaultDataGridState(cols);\n setState((prev) => ({\n ...prev,\n columnWidths: parseWidths(params.get(widthsKey), defaults.columnWidths, cols),\n columnVisibility: params.get(hiddenKey) != null\n ? parseHidden(params.get(hiddenKey), cols)\n : (initial.columnVisibility ?? defaults.columnVisibility),\n sorting: parseSort(params.get(sortKey), initial.sorting ?? defaults.sorting, cols),\n quickSearch: parseQuickSearch(params.get(searchKey), initial.quickSearch ?? defaults.quickSearch),\n }));\n };\n window.addEventListener(\"popstate\", onPop);\n return () => window.removeEventListener(\"popstate\", onPop);\n }, [widthsKey, hiddenKey, sortKey, searchKey]);\n\n return [state, setState];\n}\n"],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export { };
@@ -0,0 +1,91 @@
1
+ const require_chunk = require('../../chunk-BE-pF4vm.js');
2
+ let _testing_library_react = require("@testing-library/react");
3
+ let vitest = require("vitest");
4
+ let __use_url_state_js = require("./use-url-state.js");
5
+
6
+ //#region src/components/data-grid/use-url-state.test.tsx
7
+ // @vitest-environment jsdom
8
+ const columns = [{
9
+ id: "name",
10
+ header: "Name",
11
+ accessor: () => "",
12
+ width: 160,
13
+ minWidth: 80,
14
+ type: "string"
15
+ }, {
16
+ id: "email",
17
+ header: "Email",
18
+ accessor: () => "",
19
+ width: 200,
20
+ minWidth: 80,
21
+ type: "string"
22
+ }];
23
+ function setUrl(search) {
24
+ window.history.replaceState(null, "", `/${search ? `?${search}` : ""}`);
25
+ }
26
+ (0, vitest.beforeEach)(() => {
27
+ setUrl("");
28
+ });
29
+ (0, vitest.afterEach)(() => {
30
+ setUrl("");
31
+ });
32
+ (0, vitest.describe)("useDataGridUrlState", () => {
33
+ (0, vitest.it)("initializes from existing URL params", () => {
34
+ setUrl("grid_w=name:240&grid_h=email");
35
+ const { result } = (0, _testing_library_react.renderHook)(() => (0, __use_url_state_js.useDataGridUrlState)(columns));
36
+ const [state] = result.current;
37
+ (0, vitest.expect)(state.columnWidths.name).toBe(240);
38
+ (0, vitest.expect)(state.columnVisibility.email).toBe(false);
39
+ });
40
+ (0, vitest.it)("writes width changes back to the URL (after debounce)", async () => {
41
+ const { result } = (0, _testing_library_react.renderHook)(() => (0, __use_url_state_js.useDataGridUrlState)(columns));
42
+ (0, _testing_library_react.act)(() => {
43
+ const [, setState] = result.current;
44
+ setState((prev) => ({
45
+ ...prev,
46
+ columnWidths: {
47
+ ...prev.columnWidths,
48
+ name: 250
49
+ }
50
+ }));
51
+ });
52
+ await new Promise((resolve) => setTimeout(resolve, 150));
53
+ (0, vitest.expect)(new URLSearchParams(window.location.search).get("grid_w")).toBe("name:250");
54
+ });
55
+ (0, vitest.it)("resets column widths to defaults when a popstate clears the URL", () => {
56
+ setUrl("grid_w=name:300");
57
+ const { result } = (0, _testing_library_react.renderHook)(() => (0, __use_url_state_js.useDataGridUrlState)(columns));
58
+ (0, vitest.expect)(result.current[0].columnWidths.name).toBe(300);
59
+ (0, _testing_library_react.act)(() => {
60
+ setUrl("");
61
+ window.dispatchEvent(new PopStateEvent("popstate"));
62
+ });
63
+ (0, vitest.expect)(result.current[0].columnWidths.name).toBe(160);
64
+ });
65
+ (0, vitest.it)("isolates two grids on the same page via paramPrefix", async () => {
66
+ const { result: a } = (0, _testing_library_react.renderHook)(() => (0, __use_url_state_js.useDataGridUrlState)(columns, { paramPrefix: "users" }));
67
+ const { result: b } = (0, _testing_library_react.renderHook)(() => (0, __use_url_state_js.useDataGridUrlState)(columns, { paramPrefix: "teams" }));
68
+ (0, _testing_library_react.act)(() => {
69
+ a.current[1]((prev) => ({
70
+ ...prev,
71
+ columnWidths: {
72
+ ...prev.columnWidths,
73
+ name: 222
74
+ }
75
+ }));
76
+ });
77
+ await new Promise((resolve) => setTimeout(resolve, 150));
78
+ const params = new URLSearchParams(window.location.search);
79
+ (0, vitest.expect)(params.get("users_w")).toBe("name:222");
80
+ (0, vitest.expect)(params.get("teams_w")).toBeNull();
81
+ (0, vitest.expect)(b.current[0].columnWidths.name).toBe(160);
82
+ });
83
+ (0, vitest.it)("ignores malformed entries in the URL param without throwing", () => {
84
+ setUrl("grid_w=:,name:abc,name:240,bogusid:99");
85
+ const { result } = (0, _testing_library_react.renderHook)(() => (0, __use_url_state_js.useDataGridUrlState)(columns));
86
+ (0, vitest.expect)(result.current[0].columnWidths.name).toBe(240);
87
+ });
88
+ });
89
+
90
+ //#endregion
91
+ //# sourceMappingURL=use-url-state.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-url-state.test.js","names":[],"sources":["../../../src/components/data-grid/use-url-state.test.tsx"],"sourcesContent":["// @vitest-environment jsdom\n\nimport { act, renderHook } from \"@testing-library/react\";\nimport { afterEach, beforeEach, describe, expect, it } from \"vitest\";\nimport type { DataGridColumnDef } from \"./types\";\nimport { useDataGridUrlState } from \"./use-url-state\";\n\ntype Row = { id: string };\n\nconst columns: DataGridColumnDef<Row>[] = [\n { id: \"name\", header: \"Name\", accessor: () => \"\", width: 160, minWidth: 80, type: \"string\" },\n { id: \"email\", header: \"Email\", accessor: () => \"\", width: 200, minWidth: 80, type: \"string\" },\n];\n\nfunction setUrl(search: string) {\n window.history.replaceState(null, \"\", `/${search ? `?${search}` : \"\"}`);\n}\n\nbeforeEach(() => {\n setUrl(\"\");\n});\n\nafterEach(() => {\n setUrl(\"\");\n});\n\ndescribe(\"useDataGridUrlState\", () => {\n it(\"initializes from existing URL params\", () => {\n setUrl(\"grid_w=name:240&grid_h=email\");\n const { result } = renderHook(() => useDataGridUrlState(columns));\n const [state] = result.current;\n expect(state.columnWidths.name).toBe(240);\n expect(state.columnVisibility.email).toBe(false);\n });\n\n it(\"writes width changes back to the URL (after debounce)\", async () => {\n const { result } = renderHook(() => useDataGridUrlState(columns));\n\n act(() => {\n const [, setState] = result.current;\n setState((prev) => ({\n ...prev,\n columnWidths: { ...prev.columnWidths, name: 250 },\n }));\n });\n\n // Debounce is 100ms.\n await new Promise((resolve) => setTimeout(resolve, 150));\n const params = new URLSearchParams(window.location.search);\n expect(params.get(\"grid_w\")).toBe(\"name:250\");\n });\n\n it(\"resets column widths to defaults when a popstate clears the URL\", () => {\n setUrl(\"grid_w=name:300\");\n const { result } = renderHook(() => useDataGridUrlState(columns));\n expect(result.current[0].columnWidths.name).toBe(300);\n\n act(() => {\n setUrl(\"\");\n window.dispatchEvent(new PopStateEvent(\"popstate\"));\n });\n\n // Default for \"name\" column is its declared width of 160.\n expect(result.current[0].columnWidths.name).toBe(160);\n });\n\n it(\"isolates two grids on the same page via paramPrefix\", async () => {\n const { result: a } = renderHook(() => useDataGridUrlState(columns, { paramPrefix: \"users\" }));\n const { result: b } = renderHook(() => useDataGridUrlState(columns, { paramPrefix: \"teams\" }));\n\n act(() => {\n a.current[1]((prev) => ({ ...prev, columnWidths: { ...prev.columnWidths, name: 222 } }));\n });\n await new Promise((resolve) => setTimeout(resolve, 150));\n\n const params = new URLSearchParams(window.location.search);\n expect(params.get(\"users_w\")).toBe(\"name:222\");\n expect(params.get(\"teams_w\")).toBeNull();\n expect(b.current[0].columnWidths.name).toBe(160); // unaffected\n });\n\n it(\"ignores malformed entries in the URL param without throwing\", () => {\n setUrl(\"grid_w=:,name:abc,name:240,bogusid:99\");\n const { result } = renderHook(() => useDataGridUrlState(columns));\n // Only the well-formed `name:240` should land; junk is dropped silently.\n expect(result.current[0].columnWidths.name).toBe(240);\n });\n});\n"],"mappings":";;;;;;;AASA,MAAM,UAAoC,CACxC;CAAE,IAAI;CAAQ,QAAQ;CAAQ,gBAAgB;CAAI,OAAO;CAAK,UAAU;CAAI,MAAM;CAAU,EAC5F;CAAE,IAAI;CAAS,QAAQ;CAAS,gBAAgB;CAAI,OAAO;CAAK,UAAU;CAAI,MAAM;CAAU,CAC/F;AAED,SAAS,OAAO,QAAgB;AAC9B,QAAO,QAAQ,aAAa,MAAM,IAAI,IAAI,SAAS,IAAI,WAAW,KAAK;;6BAGxD;AACf,QAAO,GAAG;EACV;4BAEc;AACd,QAAO,GAAG;EACV;qBAEO,6BAA6B;AACpC,gBAAG,8CAA8C;AAC/C,SAAO,+BAA+B;EACtC,MAAM,EAAE,oGAAgD,QAAQ,CAAC;EACjE,MAAM,CAAC,SAAS,OAAO;AACvB,qBAAO,MAAM,aAAa,KAAK,CAAC,KAAK,IAAI;AACzC,qBAAO,MAAM,iBAAiB,MAAM,CAAC,KAAK,MAAM;GAChD;AAEF,gBAAG,yDAAyD,YAAY;EACtE,MAAM,EAAE,oGAAgD,QAAQ,CAAC;AAEjE,wCAAU;GACR,MAAM,GAAG,YAAY,OAAO;AAC5B,aAAU,UAAU;IAClB,GAAG;IACH,cAAc;KAAE,GAAG,KAAK;KAAc,MAAM;KAAK;IAClD,EAAE;IACH;AAGF,QAAM,IAAI,SAAS,YAAY,WAAW,SAAS,IAAI,CAAC;AAExD,qBADe,IAAI,gBAAgB,OAAO,SAAS,OAAO,CAC5C,IAAI,SAAS,CAAC,CAAC,KAAK,WAAW;GAC7C;AAEF,gBAAG,yEAAyE;AAC1E,SAAO,kBAAkB;EACzB,MAAM,EAAE,oGAAgD,QAAQ,CAAC;AACjE,qBAAO,OAAO,QAAQ,GAAG,aAAa,KAAK,CAAC,KAAK,IAAI;AAErD,wCAAU;AACR,UAAO,GAAG;AACV,UAAO,cAAc,IAAI,cAAc,WAAW,CAAC;IACnD;AAGF,qBAAO,OAAO,QAAQ,GAAG,aAAa,KAAK,CAAC,KAAK,IAAI;GACrD;AAEF,gBAAG,uDAAuD,YAAY;EACpE,MAAM,EAAE,QAAQ,+FAA2C,SAAS,EAAE,aAAa,SAAS,CAAC,CAAC;EAC9F,MAAM,EAAE,QAAQ,+FAA2C,SAAS,EAAE,aAAa,SAAS,CAAC,CAAC;AAE9F,wCAAU;AACR,KAAE,QAAQ,IAAI,UAAU;IAAE,GAAG;IAAM,cAAc;KAAE,GAAG,KAAK;KAAc,MAAM;KAAK;IAAE,EAAE;IACxF;AACF,QAAM,IAAI,SAAS,YAAY,WAAW,SAAS,IAAI,CAAC;EAExD,MAAM,SAAS,IAAI,gBAAgB,OAAO,SAAS,OAAO;AAC1D,qBAAO,OAAO,IAAI,UAAU,CAAC,CAAC,KAAK,WAAW;AAC9C,qBAAO,OAAO,IAAI,UAAU,CAAC,CAAC,UAAU;AACxC,qBAAO,EAAE,QAAQ,GAAG,aAAa,KAAK,CAAC,KAAK,IAAI;GAChD;AAEF,gBAAG,qEAAqE;AACtE,SAAO,wCAAwC;EAC/C,MAAM,EAAE,oGAAgD,QAAQ,CAAC;AAEjE,qBAAO,OAAO,QAAQ,GAAG,aAAa,KAAK,CAAC,KAAK,IAAI;GACrD;EACF"}
@@ -0,0 +1,67 @@
1
+ import * as react_jsx_runtime0 from "react/jsx-runtime";
2
+ import React from "react";
3
+ import { Dialog } from "@stackframe/stack-ui";
4
+ import * as _radix_ui_react_dialog0 from "@radix-ui/react-dialog";
5
+
6
+ //#region src/components/dialog.d.ts
7
+ type DesignDialogSize = "sm" | "md" | "lg" | "xl" | "2xl" | "3xl" | "4xl" | "5xl" | "6xl" | "7xl" | "full";
8
+ type DesignDialogVariant = "glassmorphic" | "plain";
9
+ type DesignDialogIcon = React.ElementType<{
10
+ className?: string;
11
+ }>;
12
+ type DesignDialogRootProps = Omit<React.ComponentProps<typeof Dialog>, "children">;
13
+ type DesignDialogProps = {
14
+ trigger?: React.ReactElement;
15
+ size?: DesignDialogSize;
16
+ variant?: DesignDialogVariant;
17
+ icon?: DesignDialogIcon | null;
18
+ title?: React.ReactNode;
19
+ description?: React.ReactNode;
20
+ headerContent?: React.ReactNode;
21
+ customHeader?: React.ReactNode;
22
+ footer?: React.ReactNode;
23
+ noBodyPadding?: boolean;
24
+ hideTopCloseButton?: boolean;
25
+ className?: string;
26
+ overlayClassName?: string;
27
+ headerClassName?: string;
28
+ bodyClassName?: string;
29
+ footerClassName?: string;
30
+ children?: React.ReactNode;
31
+ } & DesignDialogRootProps;
32
+ /**
33
+ * Canonical dashboard modal surface. This wraps the base dialog primitives with
34
+ * a reusable glassmorphic shell and consistent header/body/footer regions.
35
+ */
36
+ declare function DesignDialog({
37
+ trigger,
38
+ size,
39
+ variant,
40
+ icon: Icon,
41
+ title,
42
+ description,
43
+ headerContent,
44
+ customHeader,
45
+ footer,
46
+ noBodyPadding,
47
+ hideTopCloseButton,
48
+ className,
49
+ overlayClassName,
50
+ headerClassName,
51
+ bodyClassName,
52
+ footerClassName,
53
+ children,
54
+ ...dialogRootProps
55
+ }: DesignDialogProps): react_jsx_runtime0.JSX.Element;
56
+ declare const DesignDialogRoot: React.FC<_radix_ui_react_dialog0.DialogProps>;
57
+ declare const DesignDialogTrigger: React.ForwardRefExoticComponent<_radix_ui_react_dialog0.DialogTriggerProps & React.RefAttributes<HTMLButtonElement>>;
58
+ declare const DesignDialogClose: React.ForwardRefExoticComponent<_radix_ui_react_dialog0.DialogCloseProps & React.RefAttributes<HTMLButtonElement>>;
59
+ declare const DesignDialogTitle: React.FC<Omit<_radix_ui_react_dialog0.DialogTitleProps & React.RefAttributes<HTMLHeadingElement>, "ref"> & {
60
+ ref?: React.Ref<HTMLHeadingElement> | undefined;
61
+ }>;
62
+ declare const DesignDialogDescription: React.FC<Omit<_radix_ui_react_dialog0.DialogDescriptionProps & React.RefAttributes<HTMLParagraphElement>, "ref"> & {
63
+ ref?: React.Ref<HTMLParagraphElement> | undefined;
64
+ }>;
65
+ //#endregion
66
+ export { DesignDialog, DesignDialogClose, DesignDialogDescription, DesignDialogProps, DesignDialogRoot, DesignDialogSize, DesignDialogTitle, DesignDialogTrigger, DesignDialogVariant };
67
+ //# sourceMappingURL=dialog.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dialog.d.ts","names":[],"sources":["../../src/components/dialog.tsx"],"mappings":";;;;;;KAgBY,gBAAA;AAAA,KAaA,mBAAA;AAAA,KA0BP,gBAAA,GAAmB,KAAA,CAAM,WAAA;EAAc,SAAA;AAAA;AAAA,KAEvC,qBAAA,GAAwB,IAAA,CAAK,KAAA,CAAM,cAAA,QAAsB,MAAA;AAAA,KAElD,iBAAA;EACV,OAAA,GAAU,KAAA,CAAM,YAAA;EAChB,IAAA,GAAO,gBAAA;EACP,OAAA,GAAU,mBAAA;EACV,IAAA,GAAO,gBAAA;EACP,KAAA,GAAQ,KAAA,CAAM,SAAA;EACd,WAAA,GAAc,KAAA,CAAM,SAAA;EACpB,aAAA,GAAgB,KAAA,CAAM,SAAA;EACtB,YAAA,GAAe,KAAA,CAAM,SAAA;EACrB,MAAA,GAAS,KAAA,CAAM,SAAA;EACf,aAAA;EACA,kBAAA;EACA,SAAA;EACA,gBAAA;EACA,eAAA;EACA,aAAA;EACA,eAAA;EACA,QAAA,GAAW,KAAA,CAAM,SAAA;AAAA,IACf,qBAAA;;;;;iBAMY,YAAA,CAAA;EACd,OAAA;EACA,IAAA;EACA,OAAA;EACA,IAAA,EAAM,IAAA;EACN,KAAA;EACA,WAAA;EACA,aAAA;EACA,YAAA;EACA,MAAA;EACA,aAAA;EACA,kBAAA;EACA,SAAA;EACA,gBAAA;EACA,eAAA;EACA,aAAA;EACA,eAAA;EACA,QAAA;EAAA,GACG;AAAA,GACF,iBAAA,GAAiB,kBAAA,CAAA,GAAA,CAAA,OAAA;AAAA,cAoGP,gBAAA,EAAgB,KAAA,CAAA,EAAA,CAAS,uBAAA,CAAT,WAAA;AAAA,cAChB,mBAAA,EAAmB,KAAA,CAAA,yBAAA,CAAgB,uBAAA,CAAhB,kBAAA,GAAA,KAAA,CAAA,aAAA,CAAA,iBAAA;AAAA,cACnB,iBAAA,EAAiB,KAAA,CAAA,yBAAA,CAAc,uBAAA,CAAd,gBAAA,GAAA,KAAA,CAAA,aAAA,CAAA,iBAAA;AAAA,cACjB,iBAAA,EAAiB,KAAA,CAAA,EAAA,CAAA,IAAA,CAAc,uBAAA,CAAd,gBAAA,GAAA,KAAA,CAAA,aAAA,CAAA,kBAAA;;;cACjB,uBAAA,EAAuB,KAAA,CAAA,EAAA,CAAA,IAAA,CAAoB,uBAAA,CAApB,sBAAA,GAAA,KAAA,CAAA,aAAA,CAAA,oBAAA"}
@@ -0,0 +1,100 @@
1
+ "use client";
2
+
3
+ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
4
+ const require_chunk = require('../chunk-BE-pF4vm.js');
5
+ let _stackframe_stack_ui = require("@stackframe/stack-ui");
6
+ let react_jsx_runtime = require("react/jsx-runtime");
7
+ let react = require("react");
8
+ react = require_chunk.__toESM(react);
9
+
10
+ //#region src/components/dialog.tsx
11
+ const dialogSizeClasses = new Map([
12
+ ["sm", "max-w-sm"],
13
+ ["md", "max-w-md"],
14
+ ["lg", "max-w-lg"],
15
+ ["xl", "max-w-xl"],
16
+ ["2xl", "max-w-2xl"],
17
+ ["3xl", "max-w-3xl"],
18
+ ["4xl", "max-w-4xl"],
19
+ ["5xl", "max-w-5xl"],
20
+ ["6xl", "max-w-6xl"],
21
+ ["7xl", "max-w-7xl"],
22
+ ["full", "max-w-[calc(100vw-2rem)]"]
23
+ ]);
24
+ const dialogSurfaceClasses = new Map([["glassmorphic", "border-0 sm:rounded-2xl bg-background/85 backdrop-blur-2xl ring-1 ring-foreground/[0.06] shadow-[0_24px_48px_-12px_rgba(0,0,0,0.25),0_4px_24px_-8px_rgba(0,0,0,0.12)] dark:bg-background/80 dark:ring-white/[0.06]"], ["plain", "border bg-background shadow-lg sm:rounded-lg"]]);
25
+ const dialogOverlayClasses = new Map([["glassmorphic", "bg-black/50 backdrop-blur-sm"], ["plain", void 0]]);
26
+ /**
27
+ * Canonical dashboard modal surface. This wraps the base dialog primitives with
28
+ * a reusable glassmorphic shell and consistent header/body/footer regions.
29
+ */
30
+ function DesignDialog({ trigger, size = "lg", variant = "glassmorphic", icon: Icon = null, title, description, headerContent, customHeader, footer, noBodyPadding = false, hideTopCloseButton = false, className, overlayClassName, headerClassName, bodyClassName, footerClassName, children, ...dialogRootProps }) {
31
+ const resolvedSizeClass = dialogSizeClasses.get(size) ?? "max-w-lg";
32
+ const resolvedSurfaceClass = dialogSurfaceClasses.get(variant) ?? dialogSurfaceClasses.get("glassmorphic");
33
+ const resolvedOverlayClass = (0, _stackframe_stack_ui.cn)(dialogOverlayClasses.get(variant), overlayClassName);
34
+ const shouldRenderTopHeaderRow = Icon != null || title != null || description != null;
35
+ const shouldRenderHeader = customHeader != null || shouldRenderTopHeaderRow || headerContent != null;
36
+ const shouldRenderBody = react.default.Children.toArray(children).filter(Boolean).length > 0;
37
+ const needsAccessibleTitleFallback = !(title != null) && customHeader == null;
38
+ if (process.env.NODE_ENV !== "production" && needsAccessibleTitleFallback) console.warn("[DesignDialog] Rendered without a `title` or `customHeader`. Every dialog needs an accessible name — pass `title`, or render a `DialogTitle` inside `customHeader`.");
39
+ return /* @__PURE__ */ (0, react_jsx_runtime.jsxs)(_stackframe_stack_ui.Dialog, {
40
+ ...dialogRootProps,
41
+ children: [trigger != null && /* @__PURE__ */ (0, react_jsx_runtime.jsx)(_stackframe_stack_ui.DialogTrigger, {
42
+ asChild: true,
43
+ children: trigger
44
+ }), /* @__PURE__ */ (0, react_jsx_runtime.jsxs)(_stackframe_stack_ui.DialogContent, {
45
+ className: (0, _stackframe_stack_ui.cn)("gap-0 p-0 overflow-hidden", resolvedSizeClass, resolvedSurfaceClass, className),
46
+ overlayProps: resolvedOverlayClass ? { className: resolvedOverlayClass } : void 0,
47
+ noCloseButton: hideTopCloseButton,
48
+ children: [
49
+ needsAccessibleTitleFallback && /* @__PURE__ */ (0, react_jsx_runtime.jsx)(_stackframe_stack_ui.DialogTitle, {
50
+ className: "sr-only",
51
+ children: "Dialog"
52
+ }),
53
+ shouldRenderHeader && /* @__PURE__ */ (0, react_jsx_runtime.jsx)(_stackframe_stack_ui.DialogHeader, {
54
+ className: (0, _stackframe_stack_ui.cn)("px-6 pt-6 pb-4 border-b border-foreground/[0.06]", headerClassName),
55
+ children: customHeader ?? /* @__PURE__ */ (0, react_jsx_runtime.jsxs)(react_jsx_runtime.Fragment, { children: [shouldRenderTopHeaderRow && /* @__PURE__ */ (0, react_jsx_runtime.jsxs)("div", {
56
+ className: (0, _stackframe_stack_ui.cn)("flex items-start gap-3", Icon == null && "gap-0"),
57
+ children: [Icon != null && /* @__PURE__ */ (0, react_jsx_runtime.jsx)("div", {
58
+ className: "h-9 w-9 rounded-xl bg-primary/10 ring-1 ring-primary/15 flex items-center justify-center shrink-0",
59
+ children: /* @__PURE__ */ (0, react_jsx_runtime.jsx)(Icon, { className: "h-4 w-4 text-primary" })
60
+ }), (title != null || description != null) && /* @__PURE__ */ (0, react_jsx_runtime.jsxs)("div", {
61
+ className: "flex-1 min-w-0 space-y-1",
62
+ children: [title != null ? /* @__PURE__ */ (0, react_jsx_runtime.jsx)(_stackframe_stack_ui.DialogTitle, {
63
+ className: "text-base",
64
+ children: title
65
+ }) : null, description != null ? /* @__PURE__ */ (0, react_jsx_runtime.jsx)(_stackframe_stack_ui.DialogDescription, {
66
+ className: "text-xs",
67
+ children: description
68
+ }) : null]
69
+ })]
70
+ }), headerContent != null ? /* @__PURE__ */ (0, react_jsx_runtime.jsx)("div", {
71
+ className: (0, _stackframe_stack_ui.cn)(shouldRenderTopHeaderRow && "mt-4"),
72
+ children: headerContent
73
+ }) : null] })
74
+ }),
75
+ shouldRenderBody && /* @__PURE__ */ (0, react_jsx_runtime.jsx)(_stackframe_stack_ui.DialogBody, {
76
+ className: (0, _stackframe_stack_ui.cn)("mx-0 my-0 w-auto", noBodyPadding ? "px-0 py-0" : "px-6 py-4", bodyClassName),
77
+ children
78
+ }),
79
+ footer != null ? /* @__PURE__ */ (0, react_jsx_runtime.jsx)(_stackframe_stack_ui.DialogFooter, {
80
+ className: (0, _stackframe_stack_ui.cn)("px-6 py-3 border-t border-foreground/[0.06] bg-foreground/[0.02]", footerClassName),
81
+ children: footer
82
+ }) : null
83
+ ]
84
+ })]
85
+ });
86
+ }
87
+ const DesignDialogRoot = _stackframe_stack_ui.Dialog;
88
+ const DesignDialogTrigger = _stackframe_stack_ui.DialogTrigger;
89
+ const DesignDialogClose = _stackframe_stack_ui.DialogClose;
90
+ const DesignDialogTitle = _stackframe_stack_ui.DialogTitle;
91
+ const DesignDialogDescription = _stackframe_stack_ui.DialogDescription;
92
+
93
+ //#endregion
94
+ exports.DesignDialog = DesignDialog;
95
+ exports.DesignDialogClose = DesignDialogClose;
96
+ exports.DesignDialogDescription = DesignDialogDescription;
97
+ exports.DesignDialogRoot = DesignDialogRoot;
98
+ exports.DesignDialogTitle = DesignDialogTitle;
99
+ exports.DesignDialogTrigger = DesignDialogTrigger;
100
+ //# sourceMappingURL=dialog.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dialog.js","names":["React","Dialog","DialogTrigger","DialogContent","DialogTitle","DialogHeader","DialogDescription","DialogBody","DialogFooter","DialogClose"],"sources":["../../src/components/dialog.tsx"],"sourcesContent":["\"use client\";\n\nimport {\n Dialog,\n DialogBody,\n DialogClose,\n DialogContent,\n DialogDescription,\n DialogFooter,\n DialogHeader,\n DialogTitle,\n DialogTrigger,\n cn,\n} from \"@stackframe/stack-ui\";\nimport React from \"react\";\n\nexport type DesignDialogSize =\n | \"sm\"\n | \"md\"\n | \"lg\"\n | \"xl\"\n | \"2xl\"\n | \"3xl\"\n | \"4xl\"\n | \"5xl\"\n | \"6xl\"\n | \"7xl\"\n | \"full\";\n\nexport type DesignDialogVariant = \"glassmorphic\" | \"plain\";\n\nconst dialogSizeClasses = new Map<DesignDialogSize, string>([\n [\"sm\", \"max-w-sm\"],\n [\"md\", \"max-w-md\"],\n [\"lg\", \"max-w-lg\"],\n [\"xl\", \"max-w-xl\"],\n [\"2xl\", \"max-w-2xl\"],\n [\"3xl\", \"max-w-3xl\"],\n [\"4xl\", \"max-w-4xl\"],\n [\"5xl\", \"max-w-5xl\"],\n [\"6xl\", \"max-w-6xl\"],\n [\"7xl\", \"max-w-7xl\"],\n [\"full\", \"max-w-[calc(100vw-2rem)]\"],\n]);\n\nconst dialogSurfaceClasses = new Map<DesignDialogVariant, string>([\n [\"glassmorphic\", \"border-0 sm:rounded-2xl bg-background/85 backdrop-blur-2xl ring-1 ring-foreground/[0.06] shadow-[0_24px_48px_-12px_rgba(0,0,0,0.25),0_4px_24px_-8px_rgba(0,0,0,0.12)] dark:bg-background/80 dark:ring-white/[0.06]\"],\n [\"plain\", \"border bg-background shadow-lg sm:rounded-lg\"],\n]);\n\nconst dialogOverlayClasses = new Map<DesignDialogVariant, string | undefined>([\n [\"glassmorphic\", \"bg-black/50 backdrop-blur-sm\"],\n [\"plain\", undefined],\n]);\n\ntype DesignDialogIcon = React.ElementType<{ className?: string }>;\n\ntype DesignDialogRootProps = Omit<React.ComponentProps<typeof Dialog>, \"children\">;\n\nexport type DesignDialogProps = {\n trigger?: React.ReactElement,\n size?: DesignDialogSize,\n variant?: DesignDialogVariant,\n icon?: DesignDialogIcon | null,\n title?: React.ReactNode,\n description?: React.ReactNode,\n headerContent?: React.ReactNode,\n customHeader?: React.ReactNode,\n footer?: React.ReactNode,\n noBodyPadding?: boolean,\n hideTopCloseButton?: boolean,\n className?: string,\n overlayClassName?: string,\n headerClassName?: string,\n bodyClassName?: string,\n footerClassName?: string,\n children?: React.ReactNode,\n} & DesignDialogRootProps;\n\n/**\n * Canonical dashboard modal surface. This wraps the base dialog primitives with\n * a reusable glassmorphic shell and consistent header/body/footer regions.\n */\nexport function DesignDialog({\n trigger,\n size = \"lg\",\n variant = \"glassmorphic\",\n icon: Icon = null,\n title,\n description,\n headerContent,\n customHeader,\n footer,\n noBodyPadding = false,\n hideTopCloseButton = false,\n className,\n overlayClassName,\n headerClassName,\n bodyClassName,\n footerClassName,\n children,\n ...dialogRootProps\n}: DesignDialogProps) {\n const resolvedSizeClass = dialogSizeClasses.get(size) ?? \"max-w-lg\";\n const resolvedSurfaceClass = dialogSurfaceClasses.get(variant) ?? dialogSurfaceClasses.get(\"glassmorphic\");\n const resolvedOverlayClass = cn(dialogOverlayClasses.get(variant), overlayClassName);\n const shouldRenderTopHeaderRow = Icon != null || title != null || description != null;\n const shouldRenderHeader = customHeader != null || shouldRenderTopHeaderRow || headerContent != null;\n // Use toArray + filter(Boolean) instead of Children.count so that\n // expressions like `{condition && <Foo/>}` resolving to `false` don't\n // produce an empty DialogBody (which would still render padding/borders).\n const shouldRenderBody = React.Children.toArray(children).filter(Boolean).length > 0;\n const hasStandardTitle = title != null;\n const needsAccessibleTitleFallback = !hasStandardTitle && customHeader == null;\n\n if (process.env.NODE_ENV !== \"production\" && needsAccessibleTitleFallback) {\n console.warn(\n \"[DesignDialog] Rendered without a `title` or `customHeader`. Every dialog needs an accessible name — pass `title`, or render a `DialogTitle` inside `customHeader`.\",\n );\n }\n\n return (\n <Dialog {...dialogRootProps}>\n {trigger != null && (\n <DialogTrigger asChild>\n {trigger}\n </DialogTrigger>\n )}\n\n <DialogContent\n className={cn(\n \"gap-0 p-0 overflow-hidden\",\n resolvedSizeClass,\n resolvedSurfaceClass,\n className\n )}\n overlayProps={resolvedOverlayClass ? { className: resolvedOverlayClass } : undefined}\n noCloseButton={hideTopCloseButton}\n >\n {needsAccessibleTitleFallback && (\n <DialogTitle className=\"sr-only\">Dialog</DialogTitle>\n )}\n {shouldRenderHeader && (\n <DialogHeader className={cn(\"px-6 pt-6 pb-4 border-b border-foreground/[0.06]\", headerClassName)}>\n {customHeader ?? (\n <>\n {shouldRenderTopHeaderRow && (\n <div className={cn(\"flex items-start gap-3\", Icon == null && \"gap-0\")}>\n {Icon != null && (\n <div className=\"h-9 w-9 rounded-xl bg-primary/10 ring-1 ring-primary/15 flex items-center justify-center shrink-0\">\n <Icon className=\"h-4 w-4 text-primary\" />\n </div>\n )}\n {(title != null || description != null) && (\n <div className=\"flex-1 min-w-0 space-y-1\">\n {title != null ? (\n <DialogTitle className=\"text-base\">\n {title}\n </DialogTitle>\n ) : null}\n {description != null ? (\n <DialogDescription className=\"text-xs\">\n {description}\n </DialogDescription>\n ) : null}\n </div>\n )}\n </div>\n )}\n\n {headerContent != null ? (\n <div className={cn(shouldRenderTopHeaderRow && \"mt-4\")}>\n {headerContent}\n </div>\n ) : null}\n </>\n )}\n </DialogHeader>\n )}\n\n {shouldRenderBody && (\n <DialogBody\n className={cn(\n \"mx-0 my-0 w-auto\",\n noBodyPadding ? \"px-0 py-0\" : \"px-6 py-4\",\n bodyClassName\n )}\n >\n {children}\n </DialogBody>\n )}\n\n {footer != null ? (\n <DialogFooter className={cn(\"px-6 py-3 border-t border-foreground/[0.06] bg-foreground/[0.02]\", footerClassName)}>\n {footer}\n </DialogFooter>\n ) : null}\n </DialogContent>\n </Dialog>\n );\n}\n\nexport const DesignDialogRoot = Dialog;\nexport const DesignDialogTrigger = DialogTrigger;\nexport const DesignDialogClose = DialogClose;\nexport const DesignDialogTitle = DialogTitle;\nexport const DesignDialogDescription = DialogDescription;\n"],"mappings":";;;;;;;;;;AA+BA,MAAM,oBAAoB,IAAI,IAA8B;CAC1D,CAAC,MAAM,WAAW;CAClB,CAAC,MAAM,WAAW;CAClB,CAAC,MAAM,WAAW;CAClB,CAAC,MAAM,WAAW;CAClB,CAAC,OAAO,YAAY;CACpB,CAAC,OAAO,YAAY;CACpB,CAAC,OAAO,YAAY;CACpB,CAAC,OAAO,YAAY;CACpB,CAAC,OAAO,YAAY;CACpB,CAAC,OAAO,YAAY;CACpB,CAAC,QAAQ,2BAA2B;CACrC,CAAC;AAEF,MAAM,uBAAuB,IAAI,IAAiC,CAChE,CAAC,gBAAgB,qNAAqN,EACtO,CAAC,SAAS,+CAA+C,CAC1D,CAAC;AAEF,MAAM,uBAAuB,IAAI,IAA6C,CAC5E,CAAC,gBAAgB,+BAA+B,EAChD,CAAC,SAAS,OAAU,CACrB,CAAC;;;;;AA8BF,SAAgB,aAAa,EAC3B,SACA,OAAO,MACP,UAAU,gBACV,MAAM,OAAO,MACb,OACA,aACA,eACA,cACA,QACA,gBAAgB,OAChB,qBAAqB,OACrB,WACA,kBACA,iBACA,eACA,iBACA,UACA,GAAG,mBACiB;CACpB,MAAM,oBAAoB,kBAAkB,IAAI,KAAK,IAAI;CACzD,MAAM,uBAAuB,qBAAqB,IAAI,QAAQ,IAAI,qBAAqB,IAAI,eAAe;CAC1G,MAAM,oDAA0B,qBAAqB,IAAI,QAAQ,EAAE,iBAAiB;CACpF,MAAM,2BAA2B,QAAQ,QAAQ,SAAS,QAAQ,eAAe;CACjF,MAAM,qBAAqB,gBAAgB,QAAQ,4BAA4B,iBAAiB;CAIhG,MAAM,mBAAmBA,cAAM,SAAS,QAAQ,SAAS,CAAC,OAAO,QAAQ,CAAC,SAAS;CAEnF,MAAM,+BAA+B,EADZ,SAAS,SACwB,gBAAgB;AAE1E,KAAI,QAAQ,IAAI,aAAa,gBAAgB,6BAC3C,SAAQ,KACN,sKACD;AAGH,QACE,4CAACC;EAAO,GAAI;aACT,WAAW,QACV,2CAACC;GAAc;aACZ;IACa,EAGlB,4CAACC;GACC,wCACE,6BACA,mBACA,sBACA,UACD;GACD,cAAc,uBAAuB,EAAE,WAAW,sBAAsB,GAAG;GAC3E,eAAe;;IAEd,gCACC,2CAACC;KAAY,WAAU;eAAU;MAAoB;IAEtD,sBACC,2CAACC;KAAa,wCAAc,oDAAoD,gBAAgB;eAC7F,gBACC,qFACG,4BACC,4CAAC;MAAI,wCAAc,0BAA0B,QAAQ,QAAQ,QAAQ;iBAClE,QAAQ,QACP,2CAAC;OAAI,WAAU;iBACb,2CAAC,QAAK,WAAU,yBAAyB;QACrC,GAEN,SAAS,QAAQ,eAAe,SAChC,4CAAC;OAAI,WAAU;kBACZ,SAAS,OACR,2CAACD;QAAY,WAAU;kBACpB;SACW,GACZ,MACH,eAAe,OACd,2CAACE;QAAkB,WAAU;kBAC1B;SACiB,GAClB;QACA;OAEJ,EAGP,iBAAiB,OAChB,2CAAC;MAAI,wCAAc,4BAA4B,OAAO;gBACnD;OACG,GACJ,QACH;MAEQ;IAGhB,oBACC,2CAACC;KACC,wCACE,oBACA,gBAAgB,cAAc,aAC9B,cACD;KAEA;MACU;IAGd,UAAU,OACT,2CAACC;KAAa,wCAAc,oEAAoE,gBAAgB;eAC7G;MACY,GACb;;IACU;GACT;;AAIb,MAAa,mBAAmBP;AAChC,MAAa,sBAAsBC;AACnC,MAAa,oBAAoBO;AACjC,MAAa,oBAAoBL;AACjC,MAAa,0BAA0BE"}