@beyondwork/docx-react-component 1.0.1 → 1.0.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.
Files changed (172) hide show
  1. package/README.md +44 -104
  2. package/package.json +76 -46
  3. package/src/README.md +85 -0
  4. package/src/api/README.md +22 -0
  5. package/src/api/public-types.ts +525 -0
  6. package/src/compare/diff-engine.ts +530 -0
  7. package/src/compare/export-redlines.ts +162 -0
  8. package/src/compare/snapshot.ts +37 -0
  9. package/src/component-inventory.md +99 -0
  10. package/src/core/README.md +10 -0
  11. package/src/core/commands/README.md +3 -0
  12. package/src/core/commands/formatting-commands.ts +161 -0
  13. package/src/core/commands/image-commands.ts +144 -0
  14. package/src/core/commands/index.ts +1013 -0
  15. package/src/core/commands/list-commands.ts +370 -0
  16. package/src/core/commands/review-commands.ts +108 -0
  17. package/src/core/commands/text-commands.ts +119 -0
  18. package/src/core/schema/README.md +3 -0
  19. package/src/core/schema/text-schema.ts +512 -0
  20. package/src/core/selection/README.md +3 -0
  21. package/src/core/selection/mapping.ts +238 -0
  22. package/src/core/selection/review-anchors.ts +94 -0
  23. package/src/core/state/README.md +3 -0
  24. package/src/core/state/editor-state.ts +580 -0
  25. package/src/core/state/text-transaction.ts +276 -0
  26. package/src/formats/xlsx/io/parse-shared-strings.ts +41 -0
  27. package/src/formats/xlsx/io/parse-sheet.ts +289 -0
  28. package/src/formats/xlsx/io/parse-styles.ts +57 -0
  29. package/src/formats/xlsx/io/parse-workbook.ts +75 -0
  30. package/src/formats/xlsx/io/xlsx-session.ts +306 -0
  31. package/src/formats/xlsx/model/cell.ts +189 -0
  32. package/src/formats/xlsx/model/sheet.ts +244 -0
  33. package/src/formats/xlsx/model/styles.ts +118 -0
  34. package/src/formats/xlsx/model/workbook.ts +449 -0
  35. package/src/index.ts +45 -0
  36. package/src/io/README.md +10 -0
  37. package/src/io/docx-session.ts +1763 -0
  38. package/src/io/export/README.md +3 -0
  39. package/src/io/export/export-session.ts +165 -0
  40. package/src/io/export/minimal-docx.ts +115 -0
  41. package/src/io/export/reattach-preserved-parts.ts +54 -0
  42. package/src/io/export/serialize-comments.ts +876 -0
  43. package/src/io/export/serialize-footnotes.ts +217 -0
  44. package/src/io/export/serialize-headers-footers.ts +200 -0
  45. package/src/io/export/serialize-main-document.ts +982 -0
  46. package/src/io/export/serialize-numbering.ts +97 -0
  47. package/src/io/export/serialize-revisions.ts +389 -0
  48. package/src/io/export/serialize-runtime-revisions.ts +265 -0
  49. package/src/io/export/serialize-tables.ts +147 -0
  50. package/src/io/export/split-review-boundaries.ts +194 -0
  51. package/src/io/normalize/README.md +3 -0
  52. package/src/io/normalize/normalize-text.ts +437 -0
  53. package/src/io/ooxml/README.md +3 -0
  54. package/src/io/ooxml/parse-comments.ts +779 -0
  55. package/src/io/ooxml/parse-complex-content.ts +287 -0
  56. package/src/io/ooxml/parse-fields.ts +438 -0
  57. package/src/io/ooxml/parse-footnotes.ts +403 -0
  58. package/src/io/ooxml/parse-headers-footers.ts +483 -0
  59. package/src/io/ooxml/parse-inline-media.ts +431 -0
  60. package/src/io/ooxml/parse-main-document.ts +1846 -0
  61. package/src/io/ooxml/parse-numbering.ts +425 -0
  62. package/src/io/ooxml/parse-revisions.ts +658 -0
  63. package/src/io/ooxml/parse-shapes.ts +271 -0
  64. package/src/io/ooxml/parse-tables.ts +568 -0
  65. package/src/io/ooxml/parse-theme.ts +314 -0
  66. package/src/io/ooxml/part-manifest.ts +136 -0
  67. package/src/io/ooxml/revision-boundaries.ts +351 -0
  68. package/src/io/opc/README.md +3 -0
  69. package/src/io/opc/corrupt-package.ts +166 -0
  70. package/src/io/opc/docx-package.ts +74 -0
  71. package/src/io/opc/package-reader.ts +320 -0
  72. package/src/io/opc/package-writer.ts +273 -0
  73. package/src/legal/bookmarks.ts +196 -0
  74. package/src/legal/cross-references.ts +356 -0
  75. package/src/legal/defined-terms.ts +203 -0
  76. package/src/model/README.md +3 -0
  77. package/src/model/canonical-document.ts +1911 -0
  78. package/src/model/cds-1.0.0.ts +196 -0
  79. package/src/model/snapshot.ts +393 -0
  80. package/src/preservation/README.md +3 -0
  81. package/src/preservation/markup-compatibility.ts +48 -0
  82. package/src/preservation/opaque-fragment-store.ts +89 -0
  83. package/src/preservation/opaque-region.ts +233 -0
  84. package/src/preservation/package-preservation.ts +120 -0
  85. package/src/preservation/preserved-part-manifest.ts +56 -0
  86. package/src/preservation/relationship-retention.ts +57 -0
  87. package/src/preservation/store.ts +185 -0
  88. package/src/review/README.md +16 -0
  89. package/src/review/store/README.md +3 -0
  90. package/src/review/store/comment-anchors.ts +70 -0
  91. package/src/review/store/comment-remapping.ts +154 -0
  92. package/src/review/store/comment-store.ts +331 -0
  93. package/src/review/store/comment-thread.ts +109 -0
  94. package/src/review/store/revision-actions.ts +394 -0
  95. package/src/review/store/revision-store.ts +303 -0
  96. package/src/review/store/revision-types.ts +168 -0
  97. package/src/review/store/runtime-comment-store.ts +43 -0
  98. package/src/runtime/README.md +3 -0
  99. package/src/runtime/ai-action-policy.ts +764 -0
  100. package/src/runtime/document-runtime.ts +967 -0
  101. package/src/runtime/read-only-diagnostics-runtime.ts +232 -0
  102. package/src/runtime/review-runtime.ts +44 -0
  103. package/src/runtime/revision-runtime.ts +107 -0
  104. package/src/runtime/session-capabilities.ts +138 -0
  105. package/src/runtime/surface-projection.ts +570 -0
  106. package/src/runtime/table-commands.ts +87 -0
  107. package/src/runtime/table-schema.ts +140 -0
  108. package/src/runtime/virtualized-rendering.ts +258 -0
  109. package/src/ui/README.md +30 -0
  110. package/src/ui/WordReviewEditor.tsx +1504 -0
  111. package/src/ui/comments/README.md +3 -0
  112. package/src/ui/compatibility/README.md +3 -0
  113. package/src/ui/editor-surface/README.md +3 -0
  114. package/src/ui/headless/comment-decoration-model.ts +124 -0
  115. package/src/ui/headless/revision-decoration-model.ts +128 -0
  116. package/src/ui/headless/selection-helpers.ts +34 -0
  117. package/src/ui/headless/use-editor-keyboard.ts +98 -0
  118. package/src/ui/review/README.md +3 -0
  119. package/src/ui/shared/revision-filters.ts +31 -0
  120. package/src/ui/status/README.md +3 -0
  121. package/src/ui/theme/README.md +3 -0
  122. package/src/ui/toolbar/README.md +3 -0
  123. package/src/ui-tailwind/chrome/tw-alert-banner.tsx +48 -0
  124. package/src/ui-tailwind/chrome/tw-selection-toolbar.tsx +44 -0
  125. package/src/ui-tailwind/chrome/tw-unsaved-modal.tsx +58 -0
  126. package/src/ui-tailwind/chrome/use-before-unload.ts +20 -0
  127. package/src/ui-tailwind/editor-surface/pm-command-bridge.ts +139 -0
  128. package/src/ui-tailwind/editor-surface/pm-decorations.ts +98 -0
  129. package/src/ui-tailwind/editor-surface/pm-position-map.ts +123 -0
  130. package/src/ui-tailwind/editor-surface/pm-schema.ts +452 -0
  131. package/src/ui-tailwind/editor-surface/pm-state-from-snapshot.ts +327 -0
  132. package/src/ui-tailwind/editor-surface/search-plugin.ts +157 -0
  133. package/src/ui-tailwind/editor-surface/tw-caret.tsx +12 -0
  134. package/src/ui-tailwind/editor-surface/tw-editor-surface.tsx +150 -0
  135. package/src/ui-tailwind/editor-surface/tw-inline-token.tsx +118 -0
  136. package/src/ui-tailwind/editor-surface/tw-opaque-block.tsx +52 -0
  137. package/src/ui-tailwind/editor-surface/tw-paragraph-block.tsx +151 -0
  138. package/src/ui-tailwind/editor-surface/tw-prosemirror-surface.tsx +215 -0
  139. package/src/ui-tailwind/editor-surface/tw-segment-view.tsx +111 -0
  140. package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +122 -0
  141. package/src/ui-tailwind/index.ts +61 -0
  142. package/src/ui-tailwind/review/tw-comment-sidebar.tsx +276 -0
  143. package/src/ui-tailwind/review/tw-health-panel.tsx +120 -0
  144. package/src/ui-tailwind/review/tw-review-rail.tsx +120 -0
  145. package/src/ui-tailwind/review/tw-revision-sidebar.tsx +164 -0
  146. package/src/ui-tailwind/status/tw-status-bar.tsx +58 -0
  147. package/src/ui-tailwind/theme/editor-theme.css +190 -0
  148. package/src/ui-tailwind/toolbar/tw-toolbar-icon-button.tsx +48 -0
  149. package/src/ui-tailwind/toolbar/tw-toolbar.tsx +231 -0
  150. package/src/ui-tailwind/tw-review-workspace.tsx +140 -0
  151. package/src/validation/README.md +3 -0
  152. package/src/validation/compatibility-engine.ts +317 -0
  153. package/src/validation/compatibility-report.ts +160 -0
  154. package/src/validation/diagnostics.ts +203 -0
  155. package/src/validation/import-diagnostics.ts +128 -0
  156. package/src/validation/low-priority-word-surfaces.ts +373 -0
  157. package/dist/chunk-32W6IVQE.js +0 -7725
  158. package/dist/chunk-32W6IVQE.js.map +0 -1
  159. package/dist/index.cjs +0 -23722
  160. package/dist/index.cjs.map +0 -1
  161. package/dist/index.d.cts +0 -7
  162. package/dist/index.d.ts +0 -7
  163. package/dist/index.js +0 -16011
  164. package/dist/index.js.map +0 -1
  165. package/dist/public-types-DqCURAz8.d.cts +0 -1152
  166. package/dist/public-types-DqCURAz8.d.ts +0 -1152
  167. package/dist/tailwind.cjs +0 -8295
  168. package/dist/tailwind.cjs.map +0 -1
  169. package/dist/tailwind.d.cts +0 -323
  170. package/dist/tailwind.d.ts +0 -323
  171. package/dist/tailwind.js +0 -553
  172. package/dist/tailwind.js.map +0 -1
@@ -0,0 +1,75 @@
1
+ import { parseXmlAttributes } from "./parse-styles.ts";
2
+
3
+ /**
4
+ * Represents a single sheet entry from the workbook's sheet registry.
5
+ */
6
+ export interface WorkbookSheetEntry {
7
+ /** Human-readable sheet name. */
8
+ name: string;
9
+ /** Workbook-internal numeric sheet ID string. */
10
+ sheetId: string;
11
+ /** Relationship ID used to look up the sheet part path. */
12
+ relationshipId: string;
13
+ /** Visibility state; defaults to "visible" when absent. */
14
+ state: "visible" | "hidden" | "veryHidden";
15
+ }
16
+
17
+ export interface WorkbookParseResult {
18
+ sheets: WorkbookSheetEntry[];
19
+ /**
20
+ * Whether the workbook uses the 1904 date base.
21
+ * Most xlsx files use 1900 (false). Mac-originated files may use 1904 (true).
22
+ */
23
+ date1904: boolean;
24
+ }
25
+
26
+ /**
27
+ * Parses xl/workbook.xml and returns the ordered sheet registry.
28
+ *
29
+ * The r:id attribute in <sheet> elements uses the XML namespace prefix r.
30
+ * The attribute parser handles namespace-prefixed names (e.g. "r:id") correctly.
31
+ */
32
+ export function parseWorkbookXml(xml: string): WorkbookParseResult {
33
+ const sheets: WorkbookSheetEntry[] = [];
34
+ const sheetPattern = /<sheet\b([^>]*?)(?:\/>|>[\s\S]*?<\/sheet>)/g;
35
+ let match: RegExpExecArray | null;
36
+
37
+ while ((match = sheetPattern.exec(xml)) !== null) {
38
+ const attrs = parseXmlAttributes(match[1] ?? "");
39
+ const name = attrs["name"] ?? "";
40
+ const sheetId = attrs["sheetId"] ?? "";
41
+ const relationshipId = attrs["r:id"] ?? attrs["id"] ?? "";
42
+
43
+ if (!name || !sheetId || !relationshipId) {
44
+ continue;
45
+ }
46
+
47
+ const stateAttr = attrs["state"];
48
+ const state: WorkbookSheetEntry["state"] =
49
+ stateAttr === "hidden"
50
+ ? "hidden"
51
+ : stateAttr === "veryHidden"
52
+ ? "veryHidden"
53
+ : "visible";
54
+
55
+ sheets.push({ name, sheetId, relationshipId, state });
56
+ }
57
+
58
+ const date1904 = parseDate1904(xml);
59
+
60
+ return { sheets, date1904 };
61
+ }
62
+
63
+ /**
64
+ * Extracts the date1904 flag from the <workbookPr> element.
65
+ * Returns false (1900 base) if the attribute is absent or not "1" or "true".
66
+ */
67
+ function parseDate1904(xml: string): boolean {
68
+ const workbookPrMatch = /<workbookPr\b([^>]*?)(?:\/>|>)/i.exec(xml);
69
+ if (!workbookPrMatch) {
70
+ return false;
71
+ }
72
+ const attrs = parseXmlAttributes(workbookPrMatch[1] ?? "");
73
+ const val = attrs["date1904"];
74
+ return val === "1" || val === "true";
75
+ }
@@ -0,0 +1,306 @@
1
+ /**
2
+ * SpreadsheetML import session.
3
+ *
4
+ * Orchestrates the OPC package reader, per-part parsers, and canonical workbook
5
+ * model construction into a single import entry point.
6
+ *
7
+ * Architecture:
8
+ * readOpcPackage → find workbook part via officeDocument relationship
9
+ * → parseWorkbookXml (sheet registry, date1904)
10
+ * → parseSharedStringsXml (shared string table)
11
+ * → parseStylesXml → buildStyleRegistry (cell format + numFmt records)
12
+ * → parseSheetXml per sheet → populate CanonicalSheet cells/merges
13
+ * → return XlsxImportResult
14
+ */
15
+
16
+ import { readOpcPackage } from "../../../io/opc/package-reader.ts";
17
+ import { resolveRelationshipTarget } from "../../../io/ooxml/part-manifest.ts";
18
+ import type { CanonicalWorkbook } from "../model/workbook.ts";
19
+ import { addSheet, createWorkbook } from "../model/workbook.ts";
20
+ import { addMerge, setCell } from "../model/sheet.ts";
21
+ import type { CellValue, CellErrorCode, CachedFormulaValue, StyleRef } from "../model/cell.ts";
22
+ import {
23
+ makeBlankCell,
24
+ makeBooleanCell,
25
+ makeErrorCell,
26
+ makeFormulaCell,
27
+ makeNumberCell,
28
+ makeTextCell,
29
+ } from "../model/cell.ts";
30
+ import type { WorkbookStyleRegistry, NumFmtRecord, XfRecord } from "../model/styles.ts";
31
+ import { createEmptyStyleRegistry } from "../model/styles.ts";
32
+ import { parseWorkbookXml } from "./parse-workbook.ts";
33
+ import { parseSharedStringsXml } from "./parse-shared-strings.ts";
34
+ import { parseStylesXml, parseXmlAttributes } from "./parse-styles.ts";
35
+ import { parseSheetXml } from "./parse-sheet.ts";
36
+ import type { XlsxParsedCellValue } from "./parse-sheet.ts";
37
+
38
+ // ---------------------------------------------------------------------------
39
+ // Relationship type constants (SpreadsheetML / OPC)
40
+ // ---------------------------------------------------------------------------
41
+
42
+ export const XLSX_OFFICE_DOCUMENT_REL_TYPE =
43
+ "http://schemas.openxmlformats.org/officeDocument/2006/relationships/officeDocument";
44
+ const XLSX_WORKSHEET_REL_TYPE =
45
+ "http://schemas.openxmlformats.org/officeDocument/2006/relationships/worksheet";
46
+ const XLSX_SHARED_STRINGS_REL_TYPE =
47
+ "http://schemas.openxmlformats.org/officeDocument/2006/relationships/sharedStrings";
48
+ const XLSX_STYLES_REL_TYPE =
49
+ "http://schemas.openxmlformats.org/officeDocument/2006/relationships/styles";
50
+
51
+ // ---------------------------------------------------------------------------
52
+ // Public API
53
+ // ---------------------------------------------------------------------------
54
+
55
+ export interface XlsxImportResult {
56
+ workbook: CanonicalWorkbook;
57
+ /** Non-fatal issues encountered during import (missing optional parts, etc.). */
58
+ warnings: string[];
59
+ }
60
+
61
+ /**
62
+ * Import a .xlsx package from raw bytes into a canonical workbook model.
63
+ *
64
+ * Throws on fatal structural errors (invalid OPC, missing workbook part).
65
+ * Non-fatal issues (missing optional parts) are collected in `warnings`.
66
+ *
67
+ * @param source Raw xlsx bytes (Uint8Array or ArrayBuffer).
68
+ */
69
+ export function loadXlsxWorkbook(source: Uint8Array | ArrayBuffer): XlsxImportResult {
70
+ const pkg = readOpcPackage(source);
71
+ const warnings: string[] = [];
72
+
73
+ // 1. Locate the workbook part via the officeDocument package relationship.
74
+ const workbookRel = pkg.manifest.packageRelationships.find(
75
+ (r) => r.type === XLSX_OFFICE_DOCUMENT_REL_TYPE,
76
+ );
77
+ if (!workbookRel) {
78
+ throw new Error("XLSX package missing officeDocument relationship.");
79
+ }
80
+
81
+ const workbookPath = resolveRelationshipTarget(null, workbookRel);
82
+ const workbookPart = pkg.parts.get(workbookPath);
83
+ if (!workbookPart) {
84
+ throw new Error(`Workbook part not found at ${workbookPath}.`);
85
+ }
86
+
87
+ const workbookXml = decodePartBytes(workbookPart.bytes);
88
+ const workbookData = parseWorkbookXml(workbookXml);
89
+
90
+ // 2. Parse shared strings (optional part — many xlsx files omit it entirely).
91
+ let sharedStrings: string[] = [];
92
+ const ssRel = workbookPart.relationships.find((r) => r.type === XLSX_SHARED_STRINGS_REL_TYPE);
93
+ if (ssRel) {
94
+ const ssPath = resolveRelationshipTarget(workbookPath, ssRel);
95
+ const ssPart = pkg.parts.get(ssPath);
96
+ if (ssPart) {
97
+ sharedStrings = parseSharedStringsXml(decodePartBytes(ssPart.bytes));
98
+ } else {
99
+ warnings.push(`Shared strings part declared but missing: ${ssPath}`);
100
+ }
101
+ }
102
+
103
+ // 3. Parse styles (optional — absent means no cell format references).
104
+ let styleRegistry: WorkbookStyleRegistry = createEmptyStyleRegistry();
105
+ const stylesRel = workbookPart.relationships.find((r) => r.type === XLSX_STYLES_REL_TYPE);
106
+ if (stylesRel) {
107
+ const stylesPath = resolveRelationshipTarget(workbookPath, stylesRel);
108
+ const stylesPart = pkg.parts.get(stylesPath);
109
+ if (stylesPart) {
110
+ styleRegistry = buildStyleRegistry(decodePartBytes(stylesPart.bytes));
111
+ } else {
112
+ warnings.push(`Styles part declared but missing: ${stylesPath}`);
113
+ }
114
+ }
115
+
116
+ // 4. Build the canonical workbook.
117
+ const workbook = createWorkbook({ date1904: workbookData.date1904 });
118
+ workbook.sharedStrings = { strings: sharedStrings };
119
+ workbook.styles = styleRegistry;
120
+
121
+ // 5. Import each sheet in declared order.
122
+ for (const entry of workbookData.sheets) {
123
+ const sheetRel = workbookPart.relationships.find((r) => r.id === entry.relationshipId);
124
+ if (!sheetRel) {
125
+ warnings.push(
126
+ `No relationship found for sheet "${entry.name}" (r:id=${entry.relationshipId}).`,
127
+ );
128
+ continue;
129
+ }
130
+
131
+ const sheetPath = resolveRelationshipTarget(workbookPath, sheetRel);
132
+ const sheetPart = pkg.parts.get(sheetPath);
133
+ if (!sheetPart) {
134
+ warnings.push(`Sheet part not found at ${sheetPath} for sheet "${entry.name}".`);
135
+ continue;
136
+ }
137
+
138
+ const sheet = addSheet(workbook, entry.sheetId, entry.name);
139
+ if (entry.state !== "visible") {
140
+ sheet.hidden = true;
141
+ }
142
+
143
+ const sheetData = parseSheetXml(decodePartBytes(sheetPart.bytes), sharedStrings);
144
+
145
+ for (const parsedCell of sheetData.cells) {
146
+ const styleRef: StyleRef | undefined =
147
+ parsedCell.styleIndex !== null ? { xfIndex: parsedCell.styleIndex } : undefined;
148
+ const cellValue = convertParsedCellValue(parsedCell.value, styleRef);
149
+ if (cellValue !== null) {
150
+ setCell(sheet, parsedCell.row, parsedCell.col, cellValue);
151
+ }
152
+ }
153
+
154
+ for (const merge of sheetData.merges) {
155
+ addMerge(sheet, {
156
+ startRow: merge.startRow,
157
+ startCol: merge.startCol,
158
+ endRow: merge.endRow,
159
+ endCol: merge.endCol,
160
+ });
161
+ }
162
+ }
163
+
164
+ return { workbook, warnings };
165
+ }
166
+
167
+ // ---------------------------------------------------------------------------
168
+ // Style registry builder
169
+ // ---------------------------------------------------------------------------
170
+
171
+ function buildStyleRegistry(stylesXml: string): WorkbookStyleRegistry {
172
+ const entries = parseStylesXml(stylesXml);
173
+ const cellFormats: XfRecord[] = entries.map(
174
+ (entry, index): XfRecord => ({
175
+ xfIndex: index,
176
+ fontId: entry.fontId,
177
+ fillId: entry.fillId,
178
+ borderId: entry.borderId,
179
+ numFmtId: entry.numFmtId,
180
+ rawAttributes: {},
181
+ }),
182
+ );
183
+
184
+ return {
185
+ cellFormats,
186
+ numFormats: parseNumFmtRecords(stylesXml),
187
+ cellFormatCount: entries.length,
188
+ };
189
+ }
190
+
191
+ /**
192
+ * Extracts custom numFmt records from the <numFmts> section of styles.xml.
193
+ * Built-in Excel numFmt IDs (0–163) are not stored here; they are resolved
194
+ * by the style engine's built-in table when needed.
195
+ */
196
+ function parseNumFmtRecords(xml: string): Map<number, NumFmtRecord> {
197
+ const result = new Map<number, NumFmtRecord>();
198
+ const numFmtsMatch = /<numFmts\b[^>]*>([\s\S]*?)<\/numFmts>/i.exec(xml);
199
+ if (!numFmtsMatch) {
200
+ return result;
201
+ }
202
+
203
+ const numFmtPattern = /<numFmt\b([^>]*?)(?:\/>|>[\s\S]*?<\/numFmt>)/g;
204
+ let match: RegExpExecArray | null;
205
+ while ((match = numFmtPattern.exec(numFmtsMatch[1] ?? "")) !== null) {
206
+ const attrs = parseXmlAttributes(match[1] ?? "");
207
+ const rawId = attrs["numFmtId"];
208
+ const formatCode = attrs["formatCode"] ?? "";
209
+ if (rawId === undefined) {
210
+ continue;
211
+ }
212
+ const numFmtId = parseInt(rawId, 10);
213
+ if (!Number.isNaN(numFmtId)) {
214
+ result.set(numFmtId, { numFmtId, formatCode });
215
+ }
216
+ }
217
+
218
+ return result;
219
+ }
220
+
221
+ // ---------------------------------------------------------------------------
222
+ // Cell value conversion
223
+ // ---------------------------------------------------------------------------
224
+
225
+ /**
226
+ * Convert a parse-layer cell value to a canonical CellValue.
227
+ *
228
+ * Returns null for blank cells without a style reference — these are omitted
229
+ * from sparse storage. Blank cells WITH a style reference are preserved as
230
+ * BlankCell so the style index is not lost.
231
+ */
232
+ function convertParsedCellValue(
233
+ parsed: XlsxParsedCellValue,
234
+ styleRef: StyleRef | undefined,
235
+ ): CellValue | null {
236
+ switch (parsed.type) {
237
+ case "blank":
238
+ // Only store blank if it carries a style (e.g. formatted empty cell).
239
+ return styleRef ? makeBlankCell(styleRef) : null;
240
+
241
+ case "text":
242
+ return makeTextCell(parsed.value, false, styleRef);
243
+
244
+ case "number":
245
+ return makeNumberCell(parsed.value, styleRef);
246
+
247
+ case "boolean":
248
+ return makeBooleanCell(parsed.value, styleRef);
249
+
250
+ case "error":
251
+ return makeErrorCell(normalizeErrorCode(parsed.errorCode), styleRef);
252
+
253
+ case "formula":
254
+ return makeFormulaCell(
255
+ parsed.formula,
256
+ convertCachedFormulaValue(parsed.cachedValue),
257
+ styleRef,
258
+ );
259
+ }
260
+ }
261
+
262
+ /** Normalize a raw error string to the canonical CellErrorCode union. */
263
+ function normalizeErrorCode(raw: string): CellErrorCode {
264
+ const known: CellErrorCode[] = [
265
+ "#NULL!", "#DIV/0!", "#VALUE!", "#REF!", "#NAME?",
266
+ "#NUM!", "#N/A", "#GETTING_DATA", "#SPILL!",
267
+ ];
268
+ const upper = raw.toUpperCase();
269
+ for (const code of known) {
270
+ if (code === upper) return code;
271
+ }
272
+ // Unknown error code: fall back to #VALUE! to avoid crashing.
273
+ return "#VALUE!";
274
+ }
275
+
276
+ /**
277
+ * Convert a raw cached-value string from the parse layer into a typed
278
+ * CachedFormulaValue for the canonical model.
279
+ *
280
+ * Phase-1 heuristic: try to parse as a number first, otherwise treat as text.
281
+ * More precise type inference requires the cell's `t` attribute, which the
282
+ * parse layer does not currently surface for formula cells.
283
+ */
284
+ function convertCachedFormulaValue(
285
+ raw: string | null,
286
+ ): CachedFormulaValue | undefined {
287
+ if (raw === null) {
288
+ return undefined;
289
+ }
290
+ const num = Number(raw);
291
+ if (!Number.isNaN(num)) {
292
+ return { type: "number", value: num };
293
+ }
294
+ if (raw === "TRUE" || raw === "FALSE") {
295
+ return { type: "boolean", value: raw === "TRUE" };
296
+ }
297
+ return { type: "text", value: raw };
298
+ }
299
+
300
+ // ---------------------------------------------------------------------------
301
+ // Helpers
302
+ // ---------------------------------------------------------------------------
303
+
304
+ function decodePartBytes(bytes: Uint8Array): string {
305
+ return Buffer.from(bytes.buffer, bytes.byteOffset, bytes.byteLength).toString("utf8");
306
+ }
@@ -0,0 +1,189 @@
1
+ /**
2
+ * Cell value types for the canonical workbook model.
3
+ *
4
+ * Canonical cell state preserves semantic type without coercing everything
5
+ * into a single JavaScript value shape. Empty cells are omitted from canonical
6
+ * storage — this file defines what IS stored when a cell is present.
7
+ */
8
+
9
+ // ---------------------------------------------------------------------------
10
+ // Cell addresses
11
+ // ---------------------------------------------------------------------------
12
+
13
+ /** Zero-based row index. */
14
+ export type RowIndex = number;
15
+
16
+ /** Zero-based column index. */
17
+ export type ColIndex = number;
18
+
19
+ /**
20
+ * A cell address expressed as a zero-based (row, col) pair.
21
+ * Sparse storage uses this as the logical key.
22
+ */
23
+ export interface CellAddress {
24
+ row: RowIndex;
25
+ col: ColIndex;
26
+ }
27
+
28
+ /** Compact string key for use in Maps/Records: `"<row>:<col>"`. */
29
+ export type CellKey = string;
30
+
31
+ export function cellKey(row: RowIndex, col: ColIndex): CellKey {
32
+ return `${row}:${col}`;
33
+ }
34
+
35
+ export function parseCellKey(key: CellKey): CellAddress {
36
+ const colon = key.indexOf(":");
37
+ if (colon === -1) {
38
+ throw new Error(`Invalid cell key: ${key}`);
39
+ }
40
+ return {
41
+ row: parseInt(key.slice(0, colon), 10),
42
+ col: parseInt(key.slice(colon + 1), 10),
43
+ };
44
+ }
45
+
46
+ // ---------------------------------------------------------------------------
47
+ // Cell value families
48
+ // ---------------------------------------------------------------------------
49
+
50
+ export interface BlankCell {
51
+ kind: "blank";
52
+ styleRef?: StyleRef;
53
+ }
54
+
55
+ export interface TextCell {
56
+ kind: "text";
57
+ /** Raw string value or a shared-string index if sharedStringIndex is set. */
58
+ value: string;
59
+ /** When true, `value` is the resolved string from the shared-string table. */
60
+ fromSharedString: boolean;
61
+ styleRef?: StyleRef;
62
+ }
63
+
64
+ export interface NumberCell {
65
+ kind: "number";
66
+ value: number;
67
+ styleRef?: StyleRef;
68
+ }
69
+
70
+ export interface BooleanCell {
71
+ kind: "boolean";
72
+ value: boolean;
73
+ styleRef?: StyleRef;
74
+ }
75
+
76
+ export interface ErrorCell {
77
+ kind: "error";
78
+ errorCode: CellErrorCode;
79
+ styleRef?: StyleRef;
80
+ }
81
+
82
+ export interface FormulaCell {
83
+ kind: "formula";
84
+ formula: string;
85
+ /**
86
+ * Reference tokens extracted from the formula string for remapping.
87
+ * Examples: ["A1", "B2:C3", "Sheet1!A1"] from `=SUM(A1,Sheet1!A1)+B2:C3`.
88
+ * Populated during import; required for safe row/column/range structural edits.
89
+ * Empty array means the formula has no cell references (e.g. `=TODAY()`).
90
+ */
91
+ referenceTokens: string[];
92
+ /** Cached display value, if available from the parsed file. */
93
+ cachedValue?: CachedFormulaValue;
94
+ styleRef?: StyleRef;
95
+ }
96
+
97
+ /** Standard Excel error codes. */
98
+ export type CellErrorCode =
99
+ | "#NULL!"
100
+ | "#DIV/0!"
101
+ | "#VALUE!"
102
+ | "#REF!"
103
+ | "#NAME?"
104
+ | "#NUM!"
105
+ | "#N/A"
106
+ | "#GETTING_DATA"
107
+ | "#SPILL!";
108
+
109
+ /** The cached value that was stored alongside a formula at last save. */
110
+ export type CachedFormulaValue =
111
+ | { type: "number"; value: number }
112
+ | { type: "text"; value: string }
113
+ | { type: "boolean"; value: boolean }
114
+ | { type: "error"; errorCode: CellErrorCode }
115
+ | { type: "blank" };
116
+
117
+ /**
118
+ * The union of all canonical cell value kinds.
119
+ * Every present cell in sparse storage is one of these.
120
+ */
121
+ export type CellValue =
122
+ | BlankCell
123
+ | TextCell
124
+ | NumberCell
125
+ | BooleanCell
126
+ | ErrorCell
127
+ | FormulaCell;
128
+
129
+ // ---------------------------------------------------------------------------
130
+ // Style reference (intentionally thin — full style engine is later work)
131
+ // ---------------------------------------------------------------------------
132
+
133
+ /**
134
+ * A reference into the workbook's style registry.
135
+ * Index corresponds to a style record in the styles table.
136
+ * The full style definition is preserve-only until a style engine is built.
137
+ */
138
+ export interface StyleRef {
139
+ /** Zero-based index into the workbook style table (xf index from styles.xml). */
140
+ xfIndex: number;
141
+ }
142
+
143
+ // ---------------------------------------------------------------------------
144
+ // Cell helpers
145
+ // ---------------------------------------------------------------------------
146
+
147
+ export function makeBlankCell(styleRef?: StyleRef): BlankCell {
148
+ return styleRef ? { kind: "blank", styleRef } : { kind: "blank" };
149
+ }
150
+
151
+ export function makeTextCell(value: string, fromSharedString: boolean, styleRef?: StyleRef): TextCell {
152
+ const cell: TextCell = { kind: "text", value, fromSharedString };
153
+ if (styleRef) cell.styleRef = styleRef;
154
+ return cell;
155
+ }
156
+
157
+ export function makeNumberCell(value: number, styleRef?: StyleRef): NumberCell {
158
+ const cell: NumberCell = { kind: "number", value };
159
+ if (styleRef) cell.styleRef = styleRef;
160
+ return cell;
161
+ }
162
+
163
+ export function makeBooleanCell(value: boolean, styleRef?: StyleRef): BooleanCell {
164
+ const cell: BooleanCell = { kind: "boolean", value };
165
+ if (styleRef) cell.styleRef = styleRef;
166
+ return cell;
167
+ }
168
+
169
+ export function makeErrorCell(errorCode: CellErrorCode, styleRef?: StyleRef): ErrorCell {
170
+ const cell: ErrorCell = { kind: "error", errorCode };
171
+ if (styleRef) cell.styleRef = styleRef;
172
+ return cell;
173
+ }
174
+
175
+ export function makeFormulaCell(
176
+ formula: string,
177
+ cachedValue?: CachedFormulaValue,
178
+ styleRef?: StyleRef,
179
+ referenceTokens?: string[],
180
+ ): FormulaCell {
181
+ const cell: FormulaCell = {
182
+ kind: "formula",
183
+ formula,
184
+ referenceTokens: referenceTokens ?? [],
185
+ };
186
+ if (cachedValue !== undefined) cell.cachedValue = cachedValue;
187
+ if (styleRef) cell.styleRef = styleRef;
188
+ return cell;
189
+ }