@gridsheet/react-dev 3.0.0-rc.0 → 3.0.0-rc.2

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/README.md ADDED
@@ -0,0 +1,74 @@
1
+ # @gridsheet/react-dev
2
+
3
+ Development tools for [GridSheet](https://github.com/walkframe/gridsheet).
4
+
5
+ ## Overview
6
+
7
+ `@gridsheet/react-dev` provides a `Debugger` component that gives you real-time visibility into the internal state of a GridSheet instance. It is intended for use during development only and should **not** be included in production builds.
8
+
9
+ ## Installation
10
+
11
+ ```bash
12
+ npm install --save-dev @gridsheet/react-dev
13
+ # or
14
+ pnpm add -D @gridsheet/react-dev
15
+ ```
16
+
17
+ ## Usage
18
+
19
+ Pass a `hub` object (created via `useHub` from `@gridsheet/react-core`) to the `Debugger` component.
20
+
21
+ ```tsx
22
+ import React from 'react';
23
+ import { GridSheet, useHub } from '@gridsheet/react-core';
24
+ import { Debugger } from '@gridsheet/react-dev';
25
+
26
+ export default function App() {
27
+ const hub = useHub();
28
+ return (
29
+ <>
30
+ <GridSheet hub={hub} />
31
+ <Debugger hub={hub} />
32
+ </>
33
+ );
34
+ }
35
+ ```
36
+
37
+ ## Props
38
+
39
+ ### `Debugger`
40
+
41
+ | Prop | Type | Default | Description |
42
+ |--------------|-----------|---------|-------------------------------------------------------|
43
+ | `hub` | `HubType` | — | The hub object connected to a `GridSheet` instance. |
44
+ | `intervalMs` | `number` | `500` | Polling interval (ms) for refreshing the state view. |
45
+
46
+ ## Panel Layout
47
+
48
+ The `Debugger` renders a resizable panel divided into two rows:
49
+
50
+ ### Top row (resizable height)
51
+
52
+ | Panel | Description |
53
+ |------------------------|------------------------------------------------------------------------|
54
+ | **Wire State** | Snapshot of the shared wire state (choosing address, history, etc.). |
55
+ | **Cell** | Data of the currently selected cell (value, style, meta, etc.). |
56
+ | **Formula Expressions**| Parsed AST of the formula in the selected cell (if any). |
57
+ | **Formula Tokens** | Tokenized result of the formula lexer for the selected cell (if any). |
58
+
59
+ ### Sheet tabs
60
+
61
+ Tabs to switch which sheet's internal data is shown in the bottom row.
62
+
63
+ ### Bottom row (resizable height)
64
+
65
+ | Panel | Description |
66
+ |----------------|--------------------------------------------------|
67
+ | **Table Data** | Raw table instance data for the selected sheet. |
68
+ | **Store Data** | Internal store state for the selected sheet. |
69
+
70
+ Both rows can be resized by dragging the divider between them. Heights are persisted in `sessionStorage`.
71
+
72
+ ## License
73
+
74
+ Apache-2.0
package/dist/index.js CHANGED
@@ -1,21 +1,66 @@
1
1
  import { jsxs, jsx } from "react/jsx-runtime";
2
2
  import { useState, useEffect } from "react";
3
- import { a2p, x2c, Lexer, FormulaParser } from "@gridsheet/react-core";
4
- const Debugger = ({ hub, intervalMs = 500 }) => {
5
- const { wire } = hub;
3
+ import { a2p, x2c, Lexer, FormulaParser, Sheet } from "@gridsheet/react-core";
4
+ import { Light } from "react-syntax-highlighter";
5
+ import jsonLang from "react-syntax-highlighter/dist/esm/languages/hljs/json";
6
+ import atomOneLight from "react-syntax-highlighter/dist/esm/styles/hljs/atom-one-light";
7
+ import tomorrowNight from "react-syntax-highlighter/dist/esm/styles/hljs/tomorrow-night";
8
+ import solarizedDark from "react-syntax-highlighter/dist/esm/styles/hljs/solarized-dark";
9
+ import solarizedLight from "react-syntax-highlighter/dist/esm/styles/hljs/solarized-light";
10
+ import docco from "react-syntax-highlighter/dist/esm/styles/hljs/docco";
11
+ import atelierSeasideLight from "react-syntax-highlighter/dist/esm/styles/hljs/atelier-seaside-light";
12
+ import atelierHeathLight from "react-syntax-highlighter/dist/esm/styles/hljs/atelier-heath-light";
13
+ Light.registerLanguage("json", jsonLang);
14
+ const ss = {
15
+ get: (key) => typeof window !== "undefined" ? sessionStorage.getItem(key) : null,
16
+ set: (key, value) => {
17
+ if (typeof window !== "undefined") {
18
+ sessionStorage.setItem(key, value);
19
+ }
20
+ }
21
+ };
22
+ const JsonCode = ({
23
+ data,
24
+ replacer,
25
+ theme
26
+ }) => {
27
+ const code = JSON.stringify(data, replacer, 2) ?? "";
28
+ return /* @__PURE__ */ jsx(
29
+ Light,
30
+ {
31
+ language: "json",
32
+ style: theme,
33
+ showLineNumbers: true,
34
+ customStyle: { margin: 0, fontSize: "10px", minHeight: "100%", borderRadius: 0, whiteSpace: "pre" },
35
+ codeTagProps: { style: { display: "block", whiteSpace: "pre", fontSize: "10px" } },
36
+ children: code
37
+ }
38
+ );
39
+ };
40
+ const Debugger = ({ book, intervalMs = 500 }) => {
41
+ const { registry } = book;
6
42
  const [snapshot, setSnapshot] = useState({});
7
43
  const [activeTabId, setActiveTabId] = useState(null);
8
- const [topHeight, setTopHeight] = useState(() => {
9
- const saved = sessionStorage.getItem("debugger_top_height");
10
- return saved ? Number(saved) : 200;
11
- });
12
- const [bottomHeight, setBottomHeight] = useState(() => {
13
- const saved = sessionStorage.getItem("debugger_bottom_height");
14
- return saved ? Number(saved) : 200;
15
- });
44
+ const [topHeight, setTopHeight] = useState(200);
45
+ const [bottomHeight, setBottomHeight] = useState(200);
46
+ const [formulaView, setFormulaView] = useState("expressions");
47
+ useEffect(() => {
48
+ const savedTop = ss.get("gs-debugger-top-height");
49
+ if (savedTop) {
50
+ setTopHeight(Number(savedTop));
51
+ }
52
+ const savedBottom = ss.get("gs-debugger-bottom-height");
53
+ if (savedBottom) {
54
+ setBottomHeight(Number(savedBottom));
55
+ }
56
+ const savedFormula = ss.get("gs-debugger-formula-view");
57
+ if (savedFormula) {
58
+ setFormulaView(savedFormula);
59
+ }
60
+ }, []);
16
61
  useEffect(() => {
17
62
  const updateSnapshot = () => {
18
- if (!wire) {
63
+ if (!registry) {
19
64
  return;
20
65
  }
21
66
  const {
@@ -37,7 +82,7 @@ const Debugger = ({ hub, intervalMs = 500 }) => {
37
82
  histories,
38
83
  asyncPending,
39
84
  asyncInflight
40
- } = wire;
85
+ } = registry;
41
86
  setSnapshot({
42
87
  asyncPending,
43
88
  asyncInflight,
@@ -62,7 +107,7 @@ const Debugger = ({ hub, intervalMs = 500 }) => {
62
107
  updateSnapshot();
63
108
  const intervalId = setInterval(updateSnapshot, intervalMs);
64
109
  return () => clearInterval(intervalId);
65
- }, [wire, intervalMs]);
110
+ }, [registry, intervalMs]);
66
111
  const sheets = snapshot.sheetIdsByName ? Object.entries(snapshot.sheetIdsByName).map(([name, id]) => ({ id, name })).sort((a, b) => a.id - b.id) : [];
67
112
  useEffect(() => {
68
113
  if (activeTabId === null && sheets.length > 0) {
@@ -71,59 +116,61 @@ const Debugger = ({ hub, intervalMs = 500 }) => {
71
116
  }, [sheets.length, activeTabId]);
72
117
  let activeStoreData = null;
73
118
  let activeTableData = null;
74
- let wireCellData = null;
75
- let wireCellAddress = (wire == null ? void 0 : wire.choosingAddress) || "";
76
- let wireCellSheetName = "";
77
- let wireFormulaExpr = null;
78
- let wireFormulaTokens = null;
79
- if (wire && activeTabId !== null) {
80
- const { contextsBySheetId } = wire;
119
+ let registryCellData = null;
120
+ let registrySystemData = null;
121
+ let registryCellAddress = (registry == null ? void 0 : registry.choosingAddress) || "";
122
+ let registryCellSheetName = "";
123
+ let registryFormulaExpr = null;
124
+ let registryFormulaTokens = null;
125
+ if (registry && activeTabId !== null) {
126
+ const { contextsBySheetId } = registry;
81
127
  const context = contextsBySheetId[activeTabId];
82
128
  if (context) {
83
129
  activeStoreData = context.store;
84
- const table = context.store.tableReactive.current;
85
- if (table) {
86
- activeTableData = table;
130
+ const sheet = context.store.sheetReactive.current;
131
+ if (sheet) {
132
+ activeTableData = sheet;
87
133
  }
88
134
  }
89
135
  }
90
- if (wire && wire.choosingSheetId != null) {
91
- const { contextsBySheetId } = wire;
92
- const choosingContext = contextsBySheetId[wire.choosingSheetId];
93
- if (wire.sheetIdsByName) {
94
- const entry = Object.entries(wire.sheetIdsByName).find(([, id]) => id === wire.choosingSheetId);
136
+ if (registry && registry.choosingSheetId != null) {
137
+ const { contextsBySheetId } = registry;
138
+ const choosingContext = contextsBySheetId[registry.choosingSheetId];
139
+ if (registry.sheetIdsByName) {
140
+ const entry = Object.entries(registry.sheetIdsByName).find(([, id]) => id === registry.choosingSheetId);
95
141
  if (entry) {
96
- wireCellSheetName = entry[0];
142
+ registryCellSheetName = entry[0];
97
143
  }
98
144
  }
99
145
  if (choosingContext) {
100
- const table = choosingContext.store.tableReactive.current;
146
+ const sheet = choosingContext.store.sheetReactive.current;
101
147
  const store = choosingContext.store;
102
- if (table) {
103
- const rawTable = table.__raw__ || table;
148
+ if (sheet) {
149
+ const rawTable = sheet.__raw__ || sheet;
104
150
  const idMatrix = rawTable.idMatrix;
105
151
  const isTopHeaderSelecting = store.topHeaderSelecting;
106
152
  const isLeftHeaderSelecting = store.leftHeaderSelecting;
107
- const pos = isTopHeaderSelecting ? { y: 0, x: store.choosing.x } : isLeftHeaderSelecting ? { y: store.choosing.y, x: 0 } : a2p(wire.choosingAddress);
153
+ const pos = isTopHeaderSelecting ? { y: 0, x: store.choosing.x } : isLeftHeaderSelecting ? { y: store.choosing.y, x: 0 } : a2p(registry.choosingAddress);
108
154
  if (isTopHeaderSelecting) {
109
- wireCellAddress = `header:${x2c(store.choosing.x)}`;
155
+ registryCellAddress = `header:${x2c(store.choosing.x)}`;
110
156
  } else if (isLeftHeaderSelecting) {
111
- wireCellAddress = `header:${store.choosing.y}`;
157
+ registryCellAddress = `header:${store.choosing.y}`;
112
158
  }
113
159
  if (pos && idMatrix[pos.y]) {
114
160
  const id = idMatrix[pos.y][pos.x];
115
161
  if (id) {
116
- wireCellData = wire.data[id];
117
- const text = wireCellData == null ? void 0 : wireCellData.value;
162
+ registryCellData = registry.data[id];
163
+ registrySystemData = registry.systems[id] ?? null;
164
+ const text = registryCellData == null ? void 0 : registryCellData.value;
118
165
  if (typeof text === "string" && text.startsWith("=")) {
119
166
  try {
120
167
  const lexer = new Lexer(text.substring(1));
121
168
  lexer.tokenize();
122
- wireFormulaTokens = lexer.tokens;
169
+ registryFormulaTokens = lexer.tokens;
123
170
  const parser = new FormulaParser(lexer.tokens);
124
- wireFormulaExpr = parser.build();
171
+ registryFormulaExpr = parser.build();
125
172
  } catch (e) {
126
- wireFormulaExpr = { error: String(e) };
173
+ registryFormulaExpr = { error: String(e) };
127
174
  }
128
175
  }
129
176
  }
@@ -131,8 +178,8 @@ const Debugger = ({ hub, intervalMs = 500 }) => {
131
178
  }
132
179
  }
133
180
  }
134
- const jsonReplacer = (key, value) => {
135
- if (key === "wire" || key === "__raw__" || value && typeof value === "object" && "current" in value && Object.keys(value).length === 1) {
181
+ const baseReplacer = (key, value) => {
182
+ if (key === "registry" || key === "__raw__" || value && typeof value === "object" && "current" in value && Object.keys(value).length === 1) {
136
183
  return void 0;
137
184
  }
138
185
  if (value instanceof Map) {
@@ -143,6 +190,12 @@ const Debugger = ({ hub, intervalMs = 500 }) => {
143
190
  }
144
191
  return value;
145
192
  };
193
+ const jsonReplacer = (key, value) => {
194
+ if (value instanceof Sheet) {
195
+ return value.toString();
196
+ }
197
+ return baseReplacer(key, value);
198
+ };
146
199
  const startDrag = (e) => {
147
200
  e.preventDefault();
148
201
  const startY = e.clientY;
@@ -150,7 +203,7 @@ const Debugger = ({ hub, intervalMs = 500 }) => {
150
203
  const onMouseMove = (moveEvent) => {
151
204
  const h = Math.max(100, startHeight + moveEvent.clientY - startY);
152
205
  setTopHeight(h);
153
- sessionStorage.setItem("debugger_top_height", String(h));
206
+ ss.set("gs-debugger-top-height", String(h));
154
207
  };
155
208
  const onMouseUp = () => {
156
209
  document.removeEventListener("mousemove", onMouseMove);
@@ -166,7 +219,7 @@ const Debugger = ({ hub, intervalMs = 500 }) => {
166
219
  const onMouseMove = (moveEvent) => {
167
220
  const h = Math.max(100, startHeight + moveEvent.clientY - startY);
168
221
  setBottomHeight(h);
169
- sessionStorage.setItem("debugger_bottom_height", String(h));
222
+ ss.set("gs-debugger-bottom-height", String(h));
170
223
  };
171
224
  const onMouseUp = () => {
172
225
  document.removeEventListener("mousemove", onMouseMove);
@@ -222,10 +275,10 @@ const Debugger = ({ hub, intervalMs = 500 }) => {
222
275
  fontWeight: "bold",
223
276
  borderBottom: "2px solid #ccc"
224
277
  },
225
- children: "Wire State"
278
+ children: "Registry State"
226
279
  }
227
280
  ),
228
- /* @__PURE__ */ jsx("pre", { style: { margin: 0, padding: "12px" }, children: JSON.stringify(snapshot, jsonReplacer, 2) })
281
+ /* @__PURE__ */ jsx(JsonCode, { data: snapshot, replacer: jsonReplacer, theme: atomOneLight })
229
282
  ]
230
283
  }
231
284
  ),
@@ -254,14 +307,12 @@ const Debugger = ({ hub, intervalMs = 500 }) => {
254
307
  },
255
308
  children: [
256
309
  "Cell: ",
257
- wireCellSheetName && `${wireCellSheetName}!`,
258
- wireCellAddress,
259
- " ",
260
- (wire == null ? void 0 : wire.choosingSheetId) != null && `(SheetID: ${wire.choosingSheetId})`
310
+ registryCellSheetName && `'${registryCellSheetName}'!`,
311
+ registryCellAddress
261
312
  ]
262
313
  }
263
314
  ),
264
- wireCellData ? /* @__PURE__ */ jsx("pre", { style: { margin: 0, padding: "12px" }, children: JSON.stringify(wireCellData, null, 2) }) : /* @__PURE__ */ jsx("div", { style: { fontStyle: "italic", color: "#6c757d", padding: "12px" }, children: "No cell data" })
315
+ registryCellData ? /* @__PURE__ */ jsx(JsonCode, { data: registryCellData, replacer: baseReplacer, theme: atelierSeasideLight }) : /* @__PURE__ */ jsx("div", { style: { fontStyle: "italic", color: "#6c757d", padding: "12px" }, children: "No cell data" })
265
316
  ]
266
317
  }
267
318
  ),
@@ -272,47 +323,94 @@ const Debugger = ({ hub, intervalMs = 500 }) => {
272
323
  flex: 1,
273
324
  borderRight: "1px solid #dee2e6",
274
325
  overflow: "auto",
275
- backgroundColor: "#fffbf0",
326
+ backgroundColor: "#fdf0f2",
276
327
  position: "relative"
277
328
  },
278
329
  children: [
279
- /* @__PURE__ */ jsx(
330
+ /* @__PURE__ */ jsxs(
280
331
  "div",
281
332
  {
282
333
  style: {
283
334
  position: "sticky",
284
335
  top: 0,
285
- background: "#fffbf0",
336
+ background: "#fdf0f2",
286
337
  zIndex: 1,
287
338
  padding: "8px 12px",
288
339
  fontWeight: "bold",
289
340
  borderBottom: "2px solid #ccc"
290
341
  },
291
- children: "Formula Expressions"
342
+ children: [
343
+ "System: ",
344
+ registryCellSheetName && `'${registryCellSheetName}'!`,
345
+ registryCellAddress
346
+ ]
292
347
  }
293
348
  ),
294
- wireFormulaExpr ? /* @__PURE__ */ jsx("pre", { style: { margin: 0, padding: "12px" }, children: JSON.stringify(wireFormulaExpr, null, 2) }) : /* @__PURE__ */ jsx("div", { style: { fontStyle: "italic", color: "#6c757d", padding: "12px" }, children: wireCellData ? "Not a formula" : "No cell selected" })
349
+ registrySystemData ? /* @__PURE__ */ jsx(JsonCode, { data: registrySystemData, replacer: baseReplacer, theme: atelierHeathLight }) : /* @__PURE__ */ jsx("div", { style: { fontStyle: "italic", color: "#6c757d", padding: "12px" }, children: "No system data" })
295
350
  ]
296
351
  }
297
352
  ),
298
- /* @__PURE__ */ jsxs("div", { style: { flex: 1, overflow: "auto", backgroundColor: "#f5f0ff", position: "relative" }, children: [
299
- /* @__PURE__ */ jsx(
300
- "div",
301
- {
302
- style: {
303
- position: "sticky",
304
- top: 0,
305
- background: "#f5f0ff",
306
- zIndex: 1,
307
- padding: "8px 12px",
308
- fontWeight: "bold",
309
- borderBottom: "2px solid #ccc"
310
- },
311
- children: "Formula Tokens"
312
- }
313
- ),
314
- wireFormulaTokens ? /* @__PURE__ */ jsx("pre", { style: { margin: 0, padding: "12px" }, children: JSON.stringify(wireFormulaTokens, null, 2) }) : /* @__PURE__ */ jsx("div", { style: { fontStyle: "italic", color: "#6c757d", padding: "12px" }, children: wireCellData ? "Not a formula" : "No cell selected" })
315
- ] })
353
+ /* @__PURE__ */ jsxs(
354
+ "div",
355
+ {
356
+ style: {
357
+ flex: 1,
358
+ overflow: "auto",
359
+ backgroundColor: formulaView === "expressions" ? "#fffbf0" : "#f5f0ff",
360
+ position: "relative"
361
+ },
362
+ children: [
363
+ /* @__PURE__ */ jsxs(
364
+ "div",
365
+ {
366
+ style: {
367
+ position: "sticky",
368
+ top: 0,
369
+ background: formulaView === "expressions" ? "#fffbf0" : "#f5f0ff",
370
+ zIndex: 1,
371
+ padding: "6px 12px",
372
+ fontWeight: "bold",
373
+ borderBottom: "2px solid #ccc",
374
+ display: "flex",
375
+ alignItems: "center",
376
+ gap: "8px"
377
+ },
378
+ children: [
379
+ /* @__PURE__ */ jsx("span", { children: formulaView === "expressions" ? "Formula Expressions" : "Formula Tokens" }),
380
+ /* @__PURE__ */ jsx(
381
+ "button",
382
+ {
383
+ onClick: () => {
384
+ const next = formulaView === "expressions" ? "tokens" : "expressions";
385
+ setFormulaView(next);
386
+ ss.set("gs-debugger-formula-view", next);
387
+ },
388
+ style: {
389
+ marginLeft: "auto",
390
+ fontSize: "11px",
391
+ padding: "2px 8px",
392
+ cursor: "pointer",
393
+ border: "1px solid #aaa",
394
+ borderRadius: "4px",
395
+ background: "#fff"
396
+ },
397
+ children: formulaView === "expressions" ? "→ Tokens" : "→ Expressions"
398
+ }
399
+ )
400
+ ]
401
+ }
402
+ ),
403
+ registryCellData ? /* @__PURE__ */ jsx(
404
+ JsonCode,
405
+ {
406
+ data: formulaView === "expressions" ? registryFormulaExpr : registryFormulaTokens,
407
+ replacer: baseReplacer,
408
+ theme: formulaView === "expressions" ? solarizedLight : docco
409
+ }
410
+ ) : /* @__PURE__ */ jsx("div", { style: { fontStyle: "italic", color: "#6c757d", padding: "12px" }, children: "No cell selected" })
411
+ ]
412
+ }
413
+ )
316
414
  ]
317
415
  }
318
416
  ),
@@ -376,35 +474,48 @@ const Debugger = ({ hub, intervalMs = 500 }) => {
376
474
  display: "flex",
377
475
  flexDirection: "row",
378
476
  height: `${bottomHeight}px`,
379
- backgroundColor: "#fff",
477
+ backgroundColor: "#282c34",
380
478
  overflow: "hidden"
381
479
  },
382
480
  children: [
383
- /* @__PURE__ */ jsxs("div", { style: { flex: 1, borderRight: "1px solid #dee2e6", overflow: "auto", position: "relative" }, children: [
384
- /* @__PURE__ */ jsx(
385
- "div",
386
- {
387
- style: {
388
- position: "sticky",
389
- top: 0,
390
- background: "#fff",
391
- zIndex: 1,
392
- padding: "8px 12px",
393
- fontWeight: "bold",
394
- borderBottom: "2px solid #ccc"
395
- },
396
- children: "Table Data"
397
- }
398
- ),
399
- activeTableData ? /* @__PURE__ */ jsx("pre", { style: { margin: 0, padding: "12px" }, children: JSON.stringify(activeTableData, jsonReplacer, 2) }) : /* @__PURE__ */ jsx("div", { style: { fontStyle: "italic", color: "#6c757d", padding: "12px" }, children: "Table instance not found" })
400
- ] }),
401
481
  /* @__PURE__ */ jsxs(
402
482
  "div",
403
483
  {
404
484
  style: {
405
485
  flex: 1,
486
+ borderRight: "1px solid #3a3a3a",
406
487
  overflow: "auto",
407
- backgroundColor: "#fdfdfe",
488
+ position: "relative",
489
+ backgroundColor: "#282c34"
490
+ },
491
+ children: [
492
+ /* @__PURE__ */ jsx(
493
+ "div",
494
+ {
495
+ style: {
496
+ position: "sticky",
497
+ top: 0,
498
+ background: "#21252b",
499
+ zIndex: 1,
500
+ padding: "8px 12px",
501
+ fontWeight: "bold",
502
+ borderBottom: "2px solid #3a3a3a",
503
+ color: "#abb2bf"
504
+ },
505
+ children: "Sheet Data"
506
+ }
507
+ ),
508
+ activeTableData ? /* @__PURE__ */ jsx(JsonCode, { data: activeTableData, replacer: baseReplacer, theme: tomorrowNight }) : /* @__PURE__ */ jsx("div", { style: { fontStyle: "italic", color: "#6c757d", padding: "12px" }, children: "Sheet instance not found" })
509
+ ]
510
+ }
511
+ ),
512
+ /* @__PURE__ */ jsxs(
513
+ "div",
514
+ {
515
+ style: {
516
+ flex: 1,
517
+ overflow: "auto",
518
+ backgroundColor: "#282c34",
408
519
  position: "relative"
409
520
  },
410
521
  children: [
@@ -414,16 +525,17 @@ const Debugger = ({ hub, intervalMs = 500 }) => {
414
525
  style: {
415
526
  position: "sticky",
416
527
  top: 0,
417
- background: "#fdfdfe",
528
+ background: "#21252b",
418
529
  zIndex: 1,
419
530
  padding: "8px 12px",
420
531
  fontWeight: "bold",
421
- borderBottom: "2px solid #ccc"
532
+ borderBottom: "2px solid #3a3a3a",
533
+ color: "#abb2bf"
422
534
  },
423
535
  children: "Store Data"
424
536
  }
425
537
  ),
426
- activeStoreData ? /* @__PURE__ */ jsx("pre", { style: { margin: 0, padding: "12px" }, children: JSON.stringify(activeStoreData, jsonReplacer, 2) }) : /* @__PURE__ */ jsx("div", { style: { fontStyle: "italic", color: "#6c757d", padding: "12px" }, children: "Store state not found" })
538
+ activeStoreData ? /* @__PURE__ */ jsx(JsonCode, { data: activeStoreData, replacer: jsonReplacer, theme: solarizedDark }) : /* @__PURE__ */ jsx("div", { style: { fontStyle: "italic", color: "#6c757d", padding: "12px" }, children: "Store state not found" })
427
539
  ]
428
540
  }
429
541
  )
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sources":["../src/Debugger.tsx"],"sourcesContent":["import React, { useEffect, useState } from 'react';\nimport type { HubType } from '@gridsheet/react-core';\nimport { a2p, x2c, Lexer, FormulaParser } from '@gridsheet/react-core';\n\nexport type DebuggerProps = {\n hub: HubType;\n intervalMs?: number;\n};\n\nexport const Debugger: React.FC<DebuggerProps> = ({ hub, intervalMs = 500 }) => {\n const { wire } = hub;\n const [snapshot, setSnapshot] = useState<any>({});\n const [activeTabId, setActiveTabId] = useState<number | null>(null);\n const [topHeight, setTopHeight] = useState<number>(() => {\n const saved = sessionStorage.getItem('debugger_top_height');\n return saved ? Number(saved) : 200;\n });\n const [bottomHeight, setBottomHeight] = useState<number>(() => {\n const saved = sessionStorage.getItem('debugger_bottom_height');\n return saved ? Number(saved) : 200;\n });\n\n useEffect(() => {\n const updateSnapshot = () => {\n if (!wire) {\n return;\n }\n\n const {\n choosingAddress,\n choosingSheetId,\n editingAddress,\n editingSheetId,\n ready,\n sheetIdsByName,\n sheetHead,\n cellHead,\n paletteBySheetName,\n solvedCaches,\n copyingSheetId,\n copyingZone,\n cutting,\n historyIndex,\n historyLimit,\n histories,\n asyncPending,\n asyncInflight,\n } = wire;\n\n setSnapshot({\n asyncPending,\n asyncInflight,\n ready,\n sheetIdsByName,\n sheetHead,\n cellHead,\n paletteBySheetName,\n solvedCaches,\n choosingAddress,\n choosingSheetId,\n editingAddress,\n editingSheetId,\n copyingSheetId,\n copyingZone,\n cutting,\n historyIndex,\n historyLimit,\n histories,\n });\n };\n\n updateSnapshot();\n\n const intervalId = setInterval(updateSnapshot, intervalMs);\n return () => clearInterval(intervalId);\n }, [wire, intervalMs]);\n\n const sheets = snapshot.sheetIdsByName\n ? Object.entries(snapshot.sheetIdsByName)\n .map(([name, id]) => ({ id: id as number, name }))\n .sort((a, b) => a.id - b.id)\n : [];\n\n useEffect(() => {\n if (activeTabId === null && sheets.length > 0) {\n setActiveTabId(sheets[0].id);\n }\n }, [sheets.length, activeTabId]);\n\n let activeStoreData = null;\n let activeTableData = null;\n\n // Cell data for wire.choosingSheetId / choosingAddress\n let wireCellData = null;\n let wireCellAddress = wire?.choosingAddress || '';\n let wireCellSheetName = '';\n let wireFormulaExpr = null;\n let wireFormulaTokens = null;\n\n if (wire && activeTabId !== null) {\n const { contextsBySheetId } = wire;\n const context = contextsBySheetId[activeTabId];\n\n if (context) {\n activeStoreData = context.store;\n\n const table = context.store.tableReactive.current;\n if (table) {\n activeTableData = table;\n }\n }\n }\n\n // Resolve cell data for wire.choosingSheetId / choosingAddress\n if (wire && wire.choosingSheetId != null) {\n const { contextsBySheetId } = wire;\n const choosingContext = contextsBySheetId[wire.choosingSheetId];\n\n // Resolve sheet name\n if (wire.sheetIdsByName) {\n const entry = Object.entries(wire.sheetIdsByName).find(([, id]) => id === wire.choosingSheetId);\n if (entry) {\n wireCellSheetName = entry[0];\n }\n }\n\n if (choosingContext) {\n const table = choosingContext.store.tableReactive.current;\n const store = choosingContext.store;\n if (table) {\n const rawTable = (table as any).__raw__ || table;\n const idMatrix = rawTable.idMatrix;\n\n // When a header is selected, refer to y=0 (top) or x=0 (left) header cell\n const isTopHeaderSelecting = store.topHeaderSelecting;\n const isLeftHeaderSelecting = store.leftHeaderSelecting;\n const pos = isTopHeaderSelecting\n ? { y: 0, x: store.choosing.x }\n : isLeftHeaderSelecting\n ? { y: store.choosing.y, x: 0 }\n : a2p(wire.choosingAddress);\n\n if (isTopHeaderSelecting) {\n wireCellAddress = `header:${x2c(store.choosing.x)}`;\n } else if (isLeftHeaderSelecting) {\n wireCellAddress = `header:${store.choosing.y}`;\n }\n\n if (pos && idMatrix[pos.y]) {\n const id = idMatrix[pos.y][pos.x];\n if (id) {\n wireCellData = wire.data[id];\n\n // Parse formula\n const text = wireCellData?.value;\n if (typeof text === 'string' && text.startsWith('=')) {\n try {\n const lexer = new Lexer(text.substring(1));\n lexer.tokenize();\n wireFormulaTokens = lexer.tokens;\n const parser = new FormulaParser(lexer.tokens);\n wireFormulaExpr = parser.build();\n } catch (e) {\n wireFormulaExpr = { error: String(e) };\n }\n }\n }\n }\n }\n }\n }\n\n const jsonReplacer = (key: string, value: any) => {\n // Avoid circular refs, noise, and React RefObjects\n if (\n key === 'wire' ||\n key === '__raw__' ||\n (value && typeof value === 'object' && 'current' in value && Object.keys(value).length === 1) // basic heuristic to skip ref objects\n ) {\n return undefined;\n }\n if (value instanceof Map) {\n return Object.fromEntries(value.entries());\n }\n if (value instanceof Set) {\n return Array.from(value);\n }\n return value;\n };\n\n const startDrag = (e: React.MouseEvent) => {\n e.preventDefault();\n const startY = e.clientY;\n const startHeight = topHeight;\n const onMouseMove = (moveEvent: MouseEvent) => {\n const h = Math.max(100, startHeight + moveEvent.clientY - startY);\n setTopHeight(h);\n sessionStorage.setItem('debugger_top_height', String(h));\n };\n const onMouseUp = () => {\n document.removeEventListener('mousemove', onMouseMove);\n document.removeEventListener('mouseup', onMouseUp);\n };\n document.addEventListener('mousemove', onMouseMove);\n document.addEventListener('mouseup', onMouseUp);\n };\n\n const startDragBottom = (e: React.MouseEvent) => {\n e.preventDefault();\n const startY = e.clientY;\n const startHeight = bottomHeight;\n const onMouseMove = (moveEvent: MouseEvent) => {\n const h = Math.max(100, startHeight + moveEvent.clientY - startY);\n setBottomHeight(h);\n sessionStorage.setItem('debugger_bottom_height', String(h));\n };\n const onMouseUp = () => {\n document.removeEventListener('mousemove', onMouseMove);\n document.removeEventListener('mouseup', onMouseUp);\n };\n document.addEventListener('mousemove', onMouseMove);\n document.addEventListener('mouseup', onMouseUp);\n };\n\n return (\n <div\n style={{\n display: 'flex',\n flexDirection: 'column',\n fontFamily: 'monospace',\n fontSize: '12px',\n color: '#212529',\n border: '1px solid #dee2e6',\n borderRadius: '6px',\n overflow: 'hidden',\n }}\n >\n {/* 上段: Wire State | Wire Cell | Formula Expressions | Formula Tokens */}\n <div\n style={{\n display: 'flex',\n flexDirection: 'row',\n height: `${topHeight}px`,\n overflow: 'hidden',\n }}\n >\n {/* Wire State */}\n <div\n style={{\n flex: 1,\n borderRight: '1px solid #dee2e6',\n backgroundColor: '#fafafa',\n overflow: 'auto',\n position: 'relative',\n }}\n >\n <div\n style={{\n position: 'sticky',\n top: 0,\n background: '#fafafa',\n zIndex: 1,\n padding: '8px 12px',\n fontWeight: 'bold',\n borderBottom: '2px solid #ccc',\n }}\n >\n Wire State\n </div>\n <pre style={{ margin: 0, padding: '12px' }}>{JSON.stringify(snapshot, jsonReplacer, 2)}</pre>\n </div>\n\n {/* Wire Cell Value: wire.choosingSheetId / choosingAddress のセルデータ */}\n <div\n style={{\n flex: 1,\n borderRight: '1px solid #dee2e6',\n overflow: 'auto',\n backgroundColor: '#f0f8f0',\n position: 'relative',\n }}\n >\n <div\n style={{\n position: 'sticky',\n top: 0,\n background: '#f0f8f0',\n zIndex: 1,\n padding: '8px 12px',\n fontWeight: 'bold',\n borderBottom: '2px solid #ccc',\n }}\n >\n Cell: {wireCellSheetName && `${wireCellSheetName}!`}\n {wireCellAddress} {wire?.choosingSheetId != null && `(SheetID: ${wire.choosingSheetId})`}\n </div>\n {wireCellData ? (\n <pre style={{ margin: 0, padding: '12px' }}>{JSON.stringify(wireCellData, null, 2)}</pre>\n ) : (\n <div style={{ fontStyle: 'italic', color: '#6c757d', padding: '12px' }}>No cell data</div>\n )}\n </div>\n\n {/* Formula Expressions */}\n <div\n style={{\n flex: 1,\n borderRight: '1px solid #dee2e6',\n overflow: 'auto',\n backgroundColor: '#fffbf0',\n position: 'relative',\n }}\n >\n <div\n style={{\n position: 'sticky',\n top: 0,\n background: '#fffbf0',\n zIndex: 1,\n padding: '8px 12px',\n fontWeight: 'bold',\n borderBottom: '2px solid #ccc',\n }}\n >\n Formula Expressions\n </div>\n {wireFormulaExpr ? (\n <pre style={{ margin: 0, padding: '12px' }}>{JSON.stringify(wireFormulaExpr, null, 2)}</pre>\n ) : (\n <div style={{ fontStyle: 'italic', color: '#6c757d', padding: '12px' }}>\n {wireCellData ? 'Not a formula' : 'No cell selected'}\n </div>\n )}\n </div>\n\n {/* Formula Tokens: lexer.tokens */}\n <div style={{ flex: 1, overflow: 'auto', backgroundColor: '#f5f0ff', position: 'relative' }}>\n <div\n style={{\n position: 'sticky',\n top: 0,\n background: '#f5f0ff',\n zIndex: 1,\n padding: '8px 12px',\n fontWeight: 'bold',\n borderBottom: '2px solid #ccc',\n }}\n >\n Formula Tokens\n </div>\n {wireFormulaTokens ? (\n <pre style={{ margin: 0, padding: '12px' }}>{JSON.stringify(wireFormulaTokens, null, 2)}</pre>\n ) : (\n <div style={{ fontStyle: 'italic', color: '#6c757d', padding: '12px' }}>\n {wireCellData ? 'Not a formula' : 'No cell selected'}\n </div>\n )}\n </div>\n </div>\n\n {/* Resizer for Top Pane */}\n <div\n onMouseDown={startDrag}\n style={{\n height: '6px',\n backgroundColor: '#dee2e6',\n cursor: 'row-resize',\n transition: 'background-color 0.2s',\n zIndex: 10,\n }}\n onMouseEnter={(e) => {\n e.currentTarget.style.backgroundColor = '#adb5bd';\n }}\n onMouseLeave={(e) => {\n e.currentTarget.style.backgroundColor = '#dee2e6';\n }}\n />\n\n {/* Sheet Tabs */}\n <div\n style={{\n display: 'flex',\n borderBottom: '1px solid #dee2e6',\n borderTop: '1px solid #dee2e6',\n backgroundColor: '#f8f9fa',\n }}\n >\n {sheets.map((sheet) => (\n <div\n key={sheet.id}\n onClick={() => setActiveTabId(sheet.id)}\n style={{\n padding: '8px 16px',\n cursor: 'pointer',\n fontWeight: 'bold',\n borderBottom: activeTabId === sheet.id ? '2px solid #0d6efd' : '2px solid transparent',\n color: activeTabId === sheet.id ? '#0d6efd' : '#495057',\n }}\n >\n {sheet.name} (ID: {sheet.id})\n </div>\n ))}\n {sheets.length === 0 && <div style={{ padding: '8px 16px', color: '#6c757d' }}>No sheets detected</div>}\n </div>\n\n {/* 下段: Table Data | Store Data */}\n <div\n style={{\n display: 'flex',\n flexDirection: 'row',\n height: `${bottomHeight}px`,\n backgroundColor: '#fff',\n overflow: 'hidden',\n }}\n >\n <div style={{ flex: 1, borderRight: '1px solid #dee2e6', overflow: 'auto', position: 'relative' }}>\n <div\n style={{\n position: 'sticky',\n top: 0,\n background: '#fff',\n zIndex: 1,\n padding: '8px 12px',\n fontWeight: 'bold',\n borderBottom: '2px solid #ccc',\n }}\n >\n Table Data\n </div>\n {activeTableData ? (\n <pre style={{ margin: 0, padding: '12px' }}>{JSON.stringify(activeTableData, jsonReplacer, 2)}</pre>\n ) : (\n <div style={{ fontStyle: 'italic', color: '#6c757d', padding: '12px' }}>Table instance not found</div>\n )}\n </div>\n\n <div\n style={{\n flex: 1,\n overflow: 'auto',\n backgroundColor: '#fdfdfe',\n position: 'relative',\n }}\n >\n <div\n style={{\n position: 'sticky',\n top: 0,\n background: '#fdfdfe',\n zIndex: 1,\n padding: '8px 12px',\n fontWeight: 'bold',\n borderBottom: '2px solid #ccc',\n }}\n >\n Store Data\n </div>\n {activeStoreData ? (\n <pre style={{ margin: 0, padding: '12px' }}>{JSON.stringify(activeStoreData, jsonReplacer, 2)}</pre>\n ) : (\n <div style={{ fontStyle: 'italic', color: '#6c757d', padding: '12px' }}>Store state not found</div>\n )}\n </div>\n </div>\n\n {/* Resizer for Bottom Pane */}\n <div\n onMouseDown={startDragBottom}\n style={{\n height: '6px',\n backgroundColor: '#dee2e6',\n cursor: 'row-resize',\n transition: 'background-color 0.2s',\n zIndex: 10,\n }}\n onMouseEnter={(e) => {\n e.currentTarget.style.backgroundColor = '#adb5bd';\n }}\n onMouseLeave={(e) => {\n e.currentTarget.style.backgroundColor = '#dee2e6';\n }}\n />\n </div>\n );\n};\n"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"index.js","sources":["../src/Debugger.tsx"],"sourcesContent":["import React, { useEffect, useState } from 'react';\nimport type { BookType } from '@gridsheet/react-core';\nimport { a2p, x2c, Lexer, FormulaParser, Sheet } from '@gridsheet/react-core';\nimport { Light as SyntaxHighlighter } from 'react-syntax-highlighter';\nimport type { SyntaxHighlighterProps } from 'react-syntax-highlighter';\nimport jsonLang from 'react-syntax-highlighter/dist/esm/languages/hljs/json';\nimport atomOneLight from 'react-syntax-highlighter/dist/esm/styles/hljs/atom-one-light';\nimport tomorrowNight from 'react-syntax-highlighter/dist/esm/styles/hljs/tomorrow-night';\nimport solarizedDark from 'react-syntax-highlighter/dist/esm/styles/hljs/solarized-dark';\nimport solarizedLight from 'react-syntax-highlighter/dist/esm/styles/hljs/solarized-light';\nimport docco from 'react-syntax-highlighter/dist/esm/styles/hljs/docco';\nimport atelierSeasideLight from 'react-syntax-highlighter/dist/esm/styles/hljs/atelier-seaside-light';\nimport atelierHeathLight from 'react-syntax-highlighter/dist/esm/styles/hljs/atelier-heath-light';\n\nSyntaxHighlighter.registerLanguage('json', jsonLang);\n\nconst ss = {\n get: (key: string) => (typeof window !== 'undefined' ? sessionStorage.getItem(key) : null),\n set: (key: string, value: string) => {\n if (typeof window !== 'undefined') {\n sessionStorage.setItem(key, value);\n }\n },\n};\n\ntype HljsStyle = SyntaxHighlighterProps['style'];\n\nconst JsonCode: React.FC<{ data: any; replacer?: (k: string, v: any) => any; theme: HljsStyle }> = ({\n data,\n replacer,\n theme,\n}) => {\n const code = JSON.stringify(data, replacer, 2) ?? '';\n return (\n <SyntaxHighlighter\n language=\"json\"\n style={theme}\n showLineNumbers\n customStyle={{ margin: 0, fontSize: '10px', minHeight: '100%', borderRadius: 0, whiteSpace: 'pre' }}\n codeTagProps={{ style: { display: 'block', whiteSpace: 'pre', fontSize: '10px' } }}\n >\n {code}\n </SyntaxHighlighter>\n );\n};\n\nexport type DebuggerProps = {\n book: BookType;\n intervalMs?: number;\n};\n\nexport const Debugger: React.FC<DebuggerProps> = ({ book, intervalMs = 500 }) => {\n const { registry } = book;\n const [snapshot, setSnapshot] = useState<any>({});\n const [activeTabId, setActiveTabId] = useState<number | null>(null);\n const [topHeight, setTopHeight] = useState<number>(200);\n const [bottomHeight, setBottomHeight] = useState<number>(200);\n const [formulaView, setFormulaView] = useState<'expressions' | 'tokens'>('expressions');\n\n useEffect(() => {\n const savedTop = ss.get('gs-debugger-top-height');\n if (savedTop) {\n setTopHeight(Number(savedTop));\n }\n const savedBottom = ss.get('gs-debugger-bottom-height');\n if (savedBottom) {\n setBottomHeight(Number(savedBottom));\n }\n const savedFormula = ss.get('gs-debugger-formula-view') as 'expressions' | 'tokens';\n if (savedFormula) {\n setFormulaView(savedFormula);\n }\n }, []);\n\n useEffect(() => {\n const updateSnapshot = () => {\n if (!registry) {\n return;\n }\n\n const {\n choosingAddress,\n choosingSheetId,\n editingAddress,\n editingSheetId,\n ready,\n sheetIdsByName,\n sheetHead,\n cellHead,\n paletteBySheetName,\n solvedCaches,\n copyingSheetId,\n copyingZone,\n cutting,\n historyIndex,\n historyLimit,\n histories,\n asyncPending,\n asyncInflight,\n } = registry;\n\n setSnapshot({\n asyncPending,\n asyncInflight,\n ready,\n sheetIdsByName,\n sheetHead,\n cellHead,\n paletteBySheetName,\n solvedCaches,\n choosingAddress,\n choosingSheetId,\n editingAddress,\n editingSheetId,\n copyingSheetId,\n copyingZone,\n cutting,\n historyIndex,\n historyLimit,\n histories,\n });\n };\n\n updateSnapshot();\n\n const intervalId = setInterval(updateSnapshot, intervalMs);\n return () => clearInterval(intervalId);\n }, [registry, intervalMs]);\n\n const sheets = snapshot.sheetIdsByName\n ? Object.entries(snapshot.sheetIdsByName)\n .map(([name, id]) => ({ id: id as number, name }))\n .sort((a, b) => a.id - b.id)\n : [];\n\n useEffect(() => {\n if (activeTabId === null && sheets.length > 0) {\n setActiveTabId(sheets[0].id);\n }\n }, [sheets.length, activeTabId]);\n\n let activeStoreData = null;\n let activeTableData = null;\n\n // Cell data for the currently selected cell (registry.choosingSheetId / choosingAddress)\n let registryCellData = null;\n let registrySystemData = null;\n let registryCellAddress = registry?.choosingAddress || '';\n let registryCellSheetName = '';\n let registryFormulaExpr = null;\n let registryFormulaTokens = null;\n\n if (registry && activeTabId !== null) {\n const { contextsBySheetId } = registry;\n const context = contextsBySheetId[activeTabId];\n\n if (context) {\n activeStoreData = context.store;\n\n const sheet = context.store.sheetReactive.current;\n if (sheet) {\n activeTableData = sheet;\n }\n }\n }\n\n // Resolve cell data for the currently selected cell (registry.choosingSheetId / choosingAddress)\n if (registry && registry.choosingSheetId != null) {\n const { contextsBySheetId } = registry;\n const choosingContext = contextsBySheetId[registry.choosingSheetId];\n\n // Resolve sheet name\n if (registry.sheetIdsByName) {\n const entry = Object.entries(registry.sheetIdsByName).find(([, id]) => id === registry.choosingSheetId);\n if (entry) {\n registryCellSheetName = entry[0];\n }\n }\n\n if (choosingContext) {\n const sheet = choosingContext.store.sheetReactive.current;\n const store = choosingContext.store;\n if (sheet) {\n const rawTable = (sheet as any).__raw__ || sheet;\n const idMatrix = rawTable.idMatrix;\n\n // When a header cell is selected, use y=0 (top header) or x=0 (left header)\n const isTopHeaderSelecting = store.topHeaderSelecting;\n const isLeftHeaderSelecting = store.leftHeaderSelecting;\n const pos = isTopHeaderSelecting\n ? { y: 0, x: store.choosing.x }\n : isLeftHeaderSelecting\n ? { y: store.choosing.y, x: 0 }\n : a2p(registry.choosingAddress);\n\n if (isTopHeaderSelecting) {\n registryCellAddress = `header:${x2c(store.choosing.x)}`;\n } else if (isLeftHeaderSelecting) {\n registryCellAddress = `header:${store.choosing.y}`;\n }\n\n if (pos && idMatrix[pos.y]) {\n const id = idMatrix[pos.y][pos.x];\n if (id) {\n registryCellData = registry.data[id];\n registrySystemData = registry.systems[id] ?? null;\n\n // Parse formula\n const text = registryCellData?.value;\n if (typeof text === 'string' && text.startsWith('=')) {\n try {\n const lexer = new Lexer(text.substring(1));\n lexer.tokenize();\n registryFormulaTokens = lexer.tokens;\n const parser = new FormulaParser(lexer.tokens);\n registryFormulaExpr = parser.build();\n } catch (e) {\n registryFormulaExpr = { error: String(e) };\n }\n }\n }\n }\n }\n }\n }\n\n const baseReplacer = (key: string, value: any) => {\n // Avoid circular refs, noisy internals, and React RefObjects\n if (\n key === 'registry' ||\n key === '__raw__' ||\n (value && typeof value === 'object' && 'current' in value && Object.keys(value).length === 1) // heuristic to skip React ref objects\n ) {\n return undefined;\n }\n if (value instanceof Map) {\n return Object.fromEntries(value.entries());\n }\n if (value instanceof Set) {\n return Array.from(value);\n }\n return value;\n };\n\n // Replace nested Sheet instances with their string representation (for Registry State / Store Data panels)\n const jsonReplacer = (key: string, value: any) => {\n if (value instanceof Sheet) {\n return value.toString();\n }\n return baseReplacer(key, value);\n };\n\n const startDrag = (e: React.MouseEvent) => {\n e.preventDefault();\n const startY = e.clientY;\n const startHeight = topHeight;\n const onMouseMove = (moveEvent: MouseEvent) => {\n const h = Math.max(100, startHeight + moveEvent.clientY - startY);\n setTopHeight(h);\n ss.set('gs-debugger-top-height', String(h));\n };\n const onMouseUp = () => {\n document.removeEventListener('mousemove', onMouseMove);\n document.removeEventListener('mouseup', onMouseUp);\n };\n document.addEventListener('mousemove', onMouseMove);\n document.addEventListener('mouseup', onMouseUp);\n };\n\n const startDragBottom = (e: React.MouseEvent) => {\n e.preventDefault();\n const startY = e.clientY;\n const startHeight = bottomHeight;\n const onMouseMove = (moveEvent: MouseEvent) => {\n const h = Math.max(100, startHeight + moveEvent.clientY - startY);\n setBottomHeight(h);\n ss.set('gs-debugger-bottom-height', String(h));\n };\n const onMouseUp = () => {\n document.removeEventListener('mousemove', onMouseMove);\n document.removeEventListener('mouseup', onMouseUp);\n };\n document.addEventListener('mousemove', onMouseMove);\n document.addEventListener('mouseup', onMouseUp);\n };\n\n return (\n <div\n style={{\n display: 'flex',\n flexDirection: 'column',\n fontFamily: 'monospace',\n fontSize: '12px',\n color: '#212529',\n border: '1px solid #dee2e6',\n borderRadius: '6px',\n overflow: 'hidden',\n }}\n >\n {/* Top pane: Registry State | Cell | System | Formula */}\n <div\n style={{\n display: 'flex',\n flexDirection: 'row',\n height: `${topHeight}px`,\n overflow: 'hidden',\n }}\n >\n {/* Registry State panel */}\n <div\n style={{\n flex: 1,\n borderRight: '1px solid #dee2e6',\n backgroundColor: '#fafafa',\n overflow: 'auto',\n position: 'relative',\n }}\n >\n <div\n style={{\n position: 'sticky',\n top: 0,\n background: '#fafafa',\n zIndex: 1,\n padding: '8px 12px',\n fontWeight: 'bold',\n borderBottom: '2px solid #ccc',\n }}\n >\n Registry State\n </div>\n <JsonCode data={snapshot} replacer={jsonReplacer} theme={atomOneLight} />\n </div>\n\n {/* Registry Cell Value: cell data for registry.choosingSheetId / choosingAddress */}\n <div\n style={{\n flex: 1,\n borderRight: '1px solid #dee2e6',\n overflow: 'auto',\n backgroundColor: '#f0f8f0',\n position: 'relative',\n }}\n >\n <div\n style={{\n position: 'sticky',\n top: 0,\n background: '#f0f8f0',\n zIndex: 1,\n padding: '8px 12px',\n fontWeight: 'bold',\n borderBottom: '2px solid #ccc',\n }}\n >\n Cell: {registryCellSheetName && `'${registryCellSheetName}'!`}\n {registryCellAddress}\n </div>\n {registryCellData ? (\n <JsonCode data={registryCellData} replacer={baseReplacer} theme={atelierSeasideLight} />\n ) : (\n <div style={{ fontStyle: 'italic', color: '#6c757d', padding: '12px' }}>No cell data</div>\n )}\n </div>\n\n {/* System: systems[id] for the choosing cell */}\n <div\n style={{\n flex: 1,\n borderRight: '1px solid #dee2e6',\n overflow: 'auto',\n backgroundColor: '#fdf0f2',\n position: 'relative',\n }}\n >\n <div\n style={{\n position: 'sticky',\n top: 0,\n background: '#fdf0f2',\n zIndex: 1,\n padding: '8px 12px',\n fontWeight: 'bold',\n borderBottom: '2px solid #ccc',\n }}\n >\n System: {registryCellSheetName && `'${registryCellSheetName}'!`}\n {registryCellAddress}\n </div>\n {registrySystemData ? (\n <JsonCode data={registrySystemData} replacer={baseReplacer} theme={atelierHeathLight} />\n ) : (\n <div style={{ fontStyle: 'italic', color: '#6c757d', padding: '12px' }}>No system data</div>\n )}\n </div>\n\n {/* Formula: toggling between Expressions and Tokens */}\n <div\n style={{\n flex: 1,\n overflow: 'auto',\n backgroundColor: formulaView === 'expressions' ? '#fffbf0' : '#f5f0ff',\n position: 'relative',\n }}\n >\n <div\n style={{\n position: 'sticky',\n top: 0,\n background: formulaView === 'expressions' ? '#fffbf0' : '#f5f0ff',\n zIndex: 1,\n padding: '6px 12px',\n fontWeight: 'bold',\n borderBottom: '2px solid #ccc',\n display: 'flex',\n alignItems: 'center',\n gap: '8px',\n }}\n >\n <span>{formulaView === 'expressions' ? 'Formula Expressions' : 'Formula Tokens'}</span>\n <button\n onClick={() => {\n const next = formulaView === 'expressions' ? 'tokens' : 'expressions';\n setFormulaView(next);\n ss.set('gs-debugger-formula-view', next);\n }}\n style={{\n marginLeft: 'auto',\n fontSize: '11px',\n padding: '2px 8px',\n cursor: 'pointer',\n border: '1px solid #aaa',\n borderRadius: '4px',\n background: '#fff',\n }}\n >\n {formulaView === 'expressions' ? '→ Tokens' : '→ Expressions'}\n </button>\n </div>\n {registryCellData ? (\n <JsonCode\n data={formulaView === 'expressions' ? registryFormulaExpr : registryFormulaTokens}\n replacer={baseReplacer}\n theme={formulaView === 'expressions' ? solarizedLight : docco}\n />\n ) : (\n <div style={{ fontStyle: 'italic', color: '#6c757d', padding: '12px' }}>No cell selected</div>\n )}\n </div>\n </div>\n\n {/* Resizer for top pane */}\n <div\n onMouseDown={startDrag}\n style={{\n height: '6px',\n backgroundColor: '#dee2e6',\n cursor: 'row-resize',\n transition: 'background-color 0.2s',\n zIndex: 10,\n }}\n onMouseEnter={(e) => {\n e.currentTarget.style.backgroundColor = '#adb5bd';\n }}\n onMouseLeave={(e) => {\n e.currentTarget.style.backgroundColor = '#dee2e6';\n }}\n />\n\n {/* Sheet tabs */}\n <div\n style={{\n display: 'flex',\n borderBottom: '1px solid #dee2e6',\n borderTop: '1px solid #dee2e6',\n backgroundColor: '#f8f9fa',\n }}\n >\n {sheets.map((sheet) => (\n <div\n key={sheet.id}\n onClick={() => setActiveTabId(sheet.id)}\n style={{\n padding: '8px 16px',\n cursor: 'pointer',\n fontWeight: 'bold',\n borderBottom: activeTabId === sheet.id ? '2px solid #0d6efd' : '2px solid transparent',\n color: activeTabId === sheet.id ? '#0d6efd' : '#495057',\n }}\n >\n {sheet.name} (ID: {sheet.id})\n </div>\n ))}\n {sheets.length === 0 && <div style={{ padding: '8px 16px', color: '#6c757d' }}>No sheets detected</div>}\n </div>\n\n {/* Bottom pane: Sheet Data | Store Data */}\n <div\n style={{\n display: 'flex',\n flexDirection: 'row',\n height: `${bottomHeight}px`,\n backgroundColor: '#282c34',\n overflow: 'hidden',\n }}\n >\n <div\n style={{\n flex: 1,\n borderRight: '1px solid #3a3a3a',\n overflow: 'auto',\n position: 'relative',\n backgroundColor: '#282c34',\n }}\n >\n <div\n style={{\n position: 'sticky',\n top: 0,\n background: '#21252b',\n zIndex: 1,\n padding: '8px 12px',\n fontWeight: 'bold',\n borderBottom: '2px solid #3a3a3a',\n color: '#abb2bf',\n }}\n >\n Sheet Data\n </div>\n {activeTableData ? (\n <JsonCode data={activeTableData} replacer={baseReplacer} theme={tomorrowNight} />\n ) : (\n <div style={{ fontStyle: 'italic', color: '#6c757d', padding: '12px' }}>Sheet instance not found</div>\n )}\n </div>\n\n <div\n style={{\n flex: 1,\n overflow: 'auto',\n backgroundColor: '#282c34',\n position: 'relative',\n }}\n >\n <div\n style={{\n position: 'sticky',\n top: 0,\n background: '#21252b',\n zIndex: 1,\n padding: '8px 12px',\n fontWeight: 'bold',\n borderBottom: '2px solid #3a3a3a',\n color: '#abb2bf',\n }}\n >\n Store Data\n </div>\n {activeStoreData ? (\n <JsonCode data={activeStoreData} replacer={jsonReplacer} theme={solarizedDark} />\n ) : (\n <div style={{ fontStyle: 'italic', color: '#6c757d', padding: '12px' }}>Store state not found</div>\n )}\n </div>\n </div>\n\n {/* Resizer for bottom pane */}\n <div\n onMouseDown={startDragBottom}\n style={{\n height: '6px',\n backgroundColor: '#dee2e6',\n cursor: 'row-resize',\n transition: 'background-color 0.2s',\n zIndex: 10,\n }}\n onMouseEnter={(e) => {\n e.currentTarget.style.backgroundColor = '#adb5bd';\n }}\n onMouseLeave={(e) => {\n e.currentTarget.style.backgroundColor = '#dee2e6';\n }}\n />\n </div>\n );\n};\n"],"names":["SyntaxHighlighter"],"mappings":"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"}
@@ -1,7 +1,7 @@
1
1
  import { default as React } from 'react';
2
- import { HubType } from '../../react-core/src/index.ts';
2
+ import { BookType } from '../../react-core/src/index.ts';
3
3
  export type DebuggerProps = {
4
- hub: HubType;
4
+ book: BookType;
5
5
  intervalMs?: number;
6
6
  };
7
7
  export declare const Debugger: React.FC<DebuggerProps>;
@@ -1 +1 @@
1
- {"version":3,"file":"Debugger.d.ts","sourceRoot":"","sources":["../../../src/Debugger.tsx"],"names":[],"mappings":"AAAA,OAAO,KAA8B,MAAM,OAAO,CAAC;AACnD,OAAO,KAAK,EAAE,OAAO,EAAE,MAAM,uBAAuB,CAAC;AAGrD,MAAM,MAAM,aAAa,GAAG;IAC1B,GAAG,EAAE,OAAO,CAAC;IACb,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB,CAAC;AAEF,eAAO,MAAM,QAAQ,EAAE,KAAK,CAAC,EAAE,CAAC,aAAa,CA2d5C,CAAC"}
1
+ {"version":3,"file":"Debugger.d.ts","sourceRoot":"","sources":["../../../src/Debugger.tsx"],"names":[],"mappings":"AAAA,OAAO,KAA8B,MAAM,OAAO,CAAC;AACnD,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,uBAAuB,CAAC;AA6CtD,MAAM,MAAM,aAAa,GAAG;IAC1B,IAAI,EAAE,QAAQ,CAAC;IACf,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB,CAAC;AAEF,eAAO,MAAM,QAAQ,EAAE,KAAK,CAAC,EAAE,CAAC,aAAa,CAshB5C,CAAC"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@gridsheet/react-dev",
3
- "version": "3.0.0-rc.0",
3
+ "version": "3.0.0-rc.2",
4
4
  "description": "Development tools for GridSheet",
5
5
  "main": "./dist/index.js",
6
6
  "module": "./dist/index.js",
@@ -24,10 +24,17 @@
24
24
  "README.md",
25
25
  "LICENSE"
26
26
  ],
27
- "dependencies": {
28
- "@gridsheet/react-core": "^3.0.0-rc.0"
27
+ "repository": {
28
+ "type": "git",
29
+ "url": "git+https://github.com/walkframe/gridsheet.git"
30
+ },
31
+ "bugs": {
32
+ "url": "https://github.com/walkframe/gridsheet/issues"
29
33
  },
34
+ "homepage": "https://gridsheet.walkframe.com/",
35
+ "packageManager": "pnpm@10.6.5",
30
36
  "peerDependencies": {
37
+ "@gridsheet/react-core": ">=3.0.0-rc.2",
31
38
  "react": ">=16.9.0",
32
39
  "react-dom": ">=16.9.0"
33
40
  },
@@ -38,5 +45,9 @@
38
45
  "typescript": "^5.8.2",
39
46
  "vite": "^6.2.2",
40
47
  "vite-plugin-dts": "^4.5.3"
48
+ },
49
+ "dependencies": {
50
+ "@types/react-syntax-highlighter": "^15.5.13",
51
+ "react-syntax-highlighter": "^16.1.1"
41
52
  }
42
53
  }