@diagrammo/dgmo 0.8.3 → 0.8.5

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 (122) hide show
  1. package/.claude/commands/dgmo-diagram-this.md +60 -0
  2. package/.claude/commands/dgmo-document-project.md +128 -0
  3. package/.claude/commands/dgmo.md +452 -50
  4. package/.cursorrules +32 -37
  5. package/.github/copilot-instructions.md +35 -44
  6. package/.windsurfrules +32 -37
  7. package/README.md +4 -4
  8. package/dist/cli.cjs +188 -185
  9. package/dist/editor.cjs +338 -0
  10. package/dist/editor.cjs.map +1 -0
  11. package/dist/editor.d.cts +27 -0
  12. package/dist/editor.d.ts +27 -0
  13. package/dist/editor.js +307 -0
  14. package/dist/editor.js.map +1 -0
  15. package/dist/highlight.cjs +560 -0
  16. package/dist/highlight.cjs.map +1 -0
  17. package/dist/highlight.d.cts +32 -0
  18. package/dist/highlight.d.ts +32 -0
  19. package/dist/highlight.js +530 -0
  20. package/dist/highlight.js.map +1 -0
  21. package/dist/index.cjs +3467 -1078
  22. package/dist/index.cjs.map +1 -1
  23. package/dist/index.d.cts +22 -1
  24. package/dist/index.d.ts +22 -1
  25. package/dist/index.js +3466 -1078
  26. package/dist/index.js.map +1 -1
  27. package/docs/language-reference.md +46 -37
  28. package/gallery/fixtures/arc.dgmo +18 -0
  29. package/gallery/fixtures/area.dgmo +19 -0
  30. package/gallery/fixtures/bar-stacked.dgmo +10 -0
  31. package/gallery/fixtures/bar.dgmo +10 -0
  32. package/gallery/fixtures/c4-full.dgmo +52 -0
  33. package/gallery/fixtures/c4.dgmo +17 -0
  34. package/gallery/fixtures/chord.dgmo +12 -0
  35. package/gallery/fixtures/class-basic.dgmo +14 -0
  36. package/gallery/fixtures/class-full.dgmo +43 -0
  37. package/gallery/fixtures/doughnut.dgmo +8 -0
  38. package/gallery/fixtures/flowchart-basic.dgmo +3 -0
  39. package/gallery/fixtures/flowchart-colors.dgmo +5 -0
  40. package/gallery/fixtures/flowchart-complex.dgmo +17 -0
  41. package/gallery/fixtures/flowchart-decision.dgmo +5 -0
  42. package/gallery/fixtures/flowchart-full.dgmo +13 -0
  43. package/gallery/fixtures/flowchart-groups.dgmo +10 -0
  44. package/gallery/fixtures/flowchart-loop.dgmo +7 -0
  45. package/gallery/fixtures/flowchart-nested.dgmo +7 -0
  46. package/gallery/fixtures/flowchart-shapes.dgmo +5 -0
  47. package/gallery/fixtures/function.dgmo +8 -0
  48. package/gallery/fixtures/funnel.dgmo +7 -0
  49. package/gallery/fixtures/gantt-full.dgmo +49 -0
  50. package/gallery/fixtures/gantt.dgmo +42 -0
  51. package/gallery/fixtures/heatmap.dgmo +8 -0
  52. package/gallery/fixtures/infra-full.dgmo +78 -0
  53. package/gallery/fixtures/infra-overload.dgmo +25 -0
  54. package/gallery/fixtures/infra.dgmo +47 -0
  55. package/gallery/fixtures/initiative-status-full.dgmo +46 -0
  56. package/gallery/fixtures/initiative-status-phases.dgmo +29 -0
  57. package/gallery/fixtures/initiative-status.dgmo +9 -0
  58. package/gallery/fixtures/line.dgmo +19 -0
  59. package/gallery/fixtures/multi-line.dgmo +11 -0
  60. package/gallery/fixtures/org-basic.dgmo +16 -0
  61. package/gallery/fixtures/org-full.dgmo +69 -0
  62. package/gallery/fixtures/org-teams.dgmo +25 -0
  63. package/gallery/fixtures/pie.dgmo +9 -0
  64. package/gallery/fixtures/polar-area.dgmo +8 -0
  65. package/gallery/fixtures/quadrant.dgmo +18 -0
  66. package/gallery/fixtures/radar.dgmo +8 -0
  67. package/gallery/fixtures/sankey.dgmo +31 -0
  68. package/gallery/fixtures/scatter.dgmo +21 -0
  69. package/gallery/fixtures/sequence-tags-protocols.dgmo +45 -0
  70. package/gallery/fixtures/sequence-tags.dgmo +41 -0
  71. package/gallery/fixtures/sequence.dgmo +35 -0
  72. package/gallery/fixtures/sitemap-basic.dgmo +12 -0
  73. package/gallery/fixtures/sitemap-full.dgmo +156 -0
  74. package/gallery/fixtures/slope.dgmo +9 -0
  75. package/gallery/fixtures/spr-eras.dgmo +62 -0
  76. package/gallery/fixtures/state.dgmo +30 -0
  77. package/gallery/fixtures/timeline-intraday.dgmo +14 -0
  78. package/gallery/fixtures/timeline.dgmo +32 -0
  79. package/gallery/fixtures/venn.dgmo +10 -0
  80. package/gallery/fixtures/wordcloud.dgmo +24 -0
  81. package/package.json +71 -2
  82. package/src/c4/layout.ts +372 -90
  83. package/src/c4/parser.ts +100 -55
  84. package/src/chart.ts +91 -28
  85. package/src/class/parser.ts +41 -12
  86. package/src/cli.ts +211 -62
  87. package/src/completion.ts +378 -183
  88. package/src/d3.ts +1044 -303
  89. package/src/dgmo-mermaid.ts +16 -13
  90. package/src/dgmo-router.ts +69 -23
  91. package/src/echarts.ts +646 -153
  92. package/src/editor/dgmo.grammar +69 -0
  93. package/src/editor/dgmo.grammar.d.ts +2 -0
  94. package/src/editor/dgmo.grammar.js +18 -0
  95. package/src/editor/dgmo.grammar.terms.d.ts +5 -0
  96. package/src/editor/dgmo.grammar.terms.js +35 -0
  97. package/src/editor/highlight-api.ts +444 -0
  98. package/src/editor/highlight.ts +36 -0
  99. package/src/editor/index.ts +28 -0
  100. package/src/editor/keywords.ts +222 -0
  101. package/src/editor/tokens.ts +30 -0
  102. package/src/er/parser.ts +48 -14
  103. package/src/er/renderer.ts +112 -53
  104. package/src/gantt/calculator.ts +91 -29
  105. package/src/gantt/parser.ts +197 -71
  106. package/src/gantt/renderer.ts +1120 -350
  107. package/src/graph/flowchart-parser.ts +46 -25
  108. package/src/graph/state-parser.ts +47 -17
  109. package/src/index.ts +96 -31
  110. package/src/infra/parser.ts +157 -53
  111. package/src/infra/renderer.ts +723 -271
  112. package/src/initiative-status/parser.ts +138 -44
  113. package/src/kanban/parser.ts +25 -14
  114. package/src/org/layout.ts +111 -44
  115. package/src/org/parser.ts +69 -22
  116. package/src/palettes/index.ts +3 -2
  117. package/src/sequence/parser.ts +193 -61
  118. package/src/sitemap/parser.ts +65 -29
  119. package/src/utils/arrows.ts +2 -22
  120. package/src/utils/duration.ts +39 -21
  121. package/src/utils/legend-constants.ts +0 -2
  122. package/src/utils/parsing.ts +75 -31
@@ -0,0 +1,32 @@
1
+ /**
2
+ * Standalone DGMO syntax highlighter — no CodeMirror dependency.
3
+ *
4
+ * Exports:
5
+ * - `highlightDgmo(source)` → `HighlightToken[]` (consumer-agnostic)
6
+ * - `NORD_ROLE_STYLES` — inline style objects keyed by role (for React/Astro)
7
+ * - `ROLE_TO_ANSI` — ANSI escape codes keyed by role (for CLI)
8
+ *
9
+ * Uses the raw Lezer parser directly — keyword specialization is wired into
10
+ * the grammar, so `parser.parse()` runs it automatically.
11
+ *
12
+ * @module @diagrammo/dgmo/highlight
13
+ */
14
+ interface HighlightToken {
15
+ text: string;
16
+ role: string;
17
+ }
18
+ /**
19
+ * Tokenize DGMO source into annotated highlight spans.
20
+ *
21
+ * Guarantees lossless round-trip:
22
+ * `highlightDgmo(src).map(t => t.text).join('') === src`
23
+ */
24
+ declare function highlightDgmo(source: string): HighlightToken[];
25
+ declare const NORD_ROLE_STYLES: Record<string, Record<string, string>>;
26
+ declare const ROLE_TO_ANSI: Record<string, string>;
27
+ /**
28
+ * Render highlighted tokens to an ANSI string for terminal display.
29
+ */
30
+ declare function renderAnsi(tokens: HighlightToken[], useColor: boolean): string;
31
+
32
+ export { type HighlightToken, NORD_ROLE_STYLES, ROLE_TO_ANSI, highlightDgmo, renderAnsi };
@@ -0,0 +1,530 @@
1
+ // src/editor/dgmo.grammar.js
2
+ import { LRParser } from "@lezer/lr";
3
+
4
+ // src/editor/dgmo.grammar.terms.js
5
+ var ChartType = 1;
6
+ var TagKeyword = 2;
7
+ var DirectiveKeyword = 3;
8
+ var ControlKeyword = 4;
9
+ var ModifierKeyword = 5;
10
+
11
+ // src/editor/keywords.ts
12
+ var CHART_TYPES = /* @__PURE__ */ new Set([
13
+ // Diagram types
14
+ "sequence",
15
+ "flowchart",
16
+ "class",
17
+ "er",
18
+ "org",
19
+ "kanban",
20
+ "c4",
21
+ "initiative-status",
22
+ "state",
23
+ "sitemap",
24
+ "infra",
25
+ "gantt",
26
+ // Data chart types
27
+ "bar",
28
+ "line",
29
+ "pie",
30
+ "doughnut",
31
+ "area",
32
+ "polar-area",
33
+ "radar",
34
+ "bar-stacked",
35
+ "multi-line",
36
+ "scatter",
37
+ "sankey",
38
+ "chord",
39
+ "function",
40
+ "heatmap",
41
+ "funnel",
42
+ // Visualization types
43
+ "slope",
44
+ "wordcloud",
45
+ "arc",
46
+ "timeline",
47
+ "venn",
48
+ "quadrant"
49
+ ]);
50
+ var TAG_KEYWORD = "tag";
51
+ var DIRECTIVE_KEYWORDS = /* @__PURE__ */ new Set([
52
+ // Gantt
53
+ "start",
54
+ "era",
55
+ "marker",
56
+ "holiday",
57
+ "workweek",
58
+ "today-marker",
59
+ "critical-path",
60
+ "no-dependencies",
61
+ "sort",
62
+ // Tags
63
+ "tags",
64
+ "import",
65
+ "active-tag",
66
+ "hide",
67
+ // ER
68
+ "notation",
69
+ // Class
70
+ "extends",
71
+ "implements",
72
+ "abstract",
73
+ "interface",
74
+ "enum",
75
+ // C4
76
+ "containers",
77
+ "components",
78
+ "deployment",
79
+ // Infra directives
80
+ "sub-node-label",
81
+ "show-sub-node-count",
82
+ "no-auto-color",
83
+ // Infra node properties
84
+ "description",
85
+ "instances",
86
+ "max-rps",
87
+ "latency-ms",
88
+ "uptime",
89
+ "firewall-block",
90
+ "ratelimit-rps",
91
+ "cb-error-threshold",
92
+ "cb-latency-threshold-ms",
93
+ "buffer",
94
+ "drain-rate",
95
+ "retention-hours",
96
+ "partitions",
97
+ "split",
98
+ "slo-p90-latency-ms",
99
+ "slo-availability",
100
+ "cache-hit",
101
+ "concurrency",
102
+ "duration-ms",
103
+ "cold-start-ms",
104
+ "rps",
105
+ // Sequence
106
+ "activations",
107
+ "no-activations",
108
+ "collapse-notes",
109
+ "no-collapse-notes",
110
+ // Data charts
111
+ "stacked",
112
+ "no-label-name",
113
+ "no-label-value",
114
+ "no-label-percent",
115
+ // Slope
116
+ "period",
117
+ // Quadrant
118
+ "x-axis",
119
+ "y-axis",
120
+ "top-right",
121
+ "top-left",
122
+ "bottom-right",
123
+ "bottom-left",
124
+ // Layout
125
+ "direction-tb",
126
+ "direction-lr",
127
+ // Initiative-status
128
+ "contains",
129
+ // Data chart metadata
130
+ "title",
131
+ "series",
132
+ "orientation",
133
+ "x-label",
134
+ "y-label",
135
+ "size-label",
136
+ "columns",
137
+ "rows",
138
+ "labels",
139
+ "rotate",
140
+ "scale",
141
+ "values"
142
+ ]);
143
+ var CONTROL_KEYWORDS = /* @__PURE__ */ new Set([
144
+ "if",
145
+ "else",
146
+ "loop",
147
+ "parallel",
148
+ "note"
149
+ ]);
150
+ var STATUS_KEYWORDS = /* @__PURE__ */ new Set([
151
+ "na",
152
+ "todo",
153
+ "wip",
154
+ "done",
155
+ "blocked",
156
+ "in-progress",
157
+ "backlog",
158
+ "ready"
159
+ ]);
160
+ var MODIFIER_KEYWORDS = /* @__PURE__ */ new Set([
161
+ "alias",
162
+ "aka",
163
+ "position",
164
+ "default",
165
+ // Sequence participant types
166
+ "actor",
167
+ "service",
168
+ "database",
169
+ "queue",
170
+ "cache",
171
+ "gateway",
172
+ "external",
173
+ "networking",
174
+ "frontend",
175
+ // C4 element types
176
+ "system",
177
+ "person",
178
+ "container",
179
+ "component",
180
+ // ER column modifiers
181
+ "pk",
182
+ "fk",
183
+ "nullable",
184
+ "unique",
185
+ // ER data types
186
+ "int",
187
+ "varchar",
188
+ "text",
189
+ "boolean",
190
+ "date",
191
+ "timestamp",
192
+ "float",
193
+ "decimal"
194
+ ]);
195
+
196
+ // src/editor/tokens.ts
197
+ function specializeKeyword(value) {
198
+ if (CHART_TYPES.has(value)) return ChartType;
199
+ if (value === TAG_KEYWORD) return TagKeyword;
200
+ if (DIRECTIVE_KEYWORDS.has(value)) return DirectiveKeyword;
201
+ if (CONTROL_KEYWORDS.has(value)) return ControlKeyword;
202
+ if (STATUS_KEYWORDS.has(value)) return ModifierKeyword;
203
+ if (MODIFIER_KEYWORDS.has(value)) return ModifierKeyword;
204
+ return -1;
205
+ }
206
+
207
+ // src/editor/dgmo.grammar.js
208
+ var parser = LRParser.deserialize({
209
+ version: 14,
210
+ states: "!WQVQPOOOOQO'#DU'#DUOOQO'#DP'#DPO%]QPO'#CdOOQO'#DO'#DOQVQPOOOOQO-E6}-E6}OOQO,59O,59OOOQO-E6|-E6|",
211
+ stateData: "&Q~OvOS~OPPOQPORPOSPOTPOVSOXPOYPOZPO[PO]PO^PO_PO`POaPObPOcPOdPOePOfPOgPOhPOiPOjPOkPOlPOmPOnPOoPOpPOqPOwSO~OPPOQPORPOSPOTPOXPOYPOZPO[PO]PO^PO_PO`POaPObPOcPOdPOePOfPOgPOhPOiPOjPOkPOlPOmPOnPOoPOpPOqPO~OwVO~P#]OVXYZ[]^_`ghijklmnoabcdefpqk~",
212
+ goto: "!byPPPPPPPPzPPPPPPPPPPPPPPPPPPPPPPPPP!O!UPPPP!]TSOTQTORWTSROTRURVQORT",
213
+ nodeNames: "\u26A0 ChartType TagKeyword DirectiveKeyword ControlKeyword ModifierKeyword Document Comment ContentLine SyncArrow AsyncArrow Duration DateLiteral Percentage Number SectionMarker Url OpenBracket CloseBracket OpenParen CloseParen OpenAngle CloseAngle ColorAnnotation Pipe Colon Comma Plus Dash Tilde Star Question Identifier Punct",
214
+ maxTerm: 40,
215
+ skippedNodes: [0],
216
+ repeatNodeCount: 2,
217
+ tokenData: "9q~RxOX#oXY#tYZ$PZp#opq#tqt#oux#oxy$Uyz$tz{${{|%S|}%Z}!O%b!O!P#o!P!Q%q!Q![&b![!],v!]!^#o!^!_,}!_!`-U!`!a-c!a!b-j!b!c#o!c!}-q!}#O0w#O#P#o#P#Q0|#Q#R#o#R#S-q#S#T#o#T#[-q#[#]1T#]#o-q#o#p#o#p#q9T#q#r#o#r#s9[#s;'S#o;'S;=`9k<%lO#o~#tOq~~#yQv~XY#tpq#t~$UOw~~$]Qc~q~}!O$c#T#o$c~$fRyz$o}!O$c#T#o$c~$tOg~~${Od~q~~%SOn~q~~%ZOk~q~~%bOj~q~~%iPl~q~!`!a%l~%qOX~~%vPq~!P!Q%y~&OSV~OY%yZ;'S%y;'S;=`&[<%lO%y~&_P;=`<%l%y~&iY^~q~uv'X!O!P'^!Q![(z#U#V(U#W#X([#[#]([#a#b(i#e#f([#k#l([#m#n([~'^O]~~'aP!Q!['d~'iX^~uv'X!Q!['d#U#V(U#W#X([#[#]([#a#b(i#e#f([#k#l([#m#n([~(XP#W#X([~(aPZ~!a!b(d~(iOZ~~(nQZ~!a!b(d#]#^(t~(wP#b#c([~)PY^~uv'X!O!P'^!Q![)o#U#V(U#W#X([#[#]([#a#b(i#e#f([#k#l([#m#n([~)tY^~uv'X!O!P'^!Q![*d#U#V(U#W#X([#[#]([#a#b(i#e#f([#k#l([#m#n([~*iZ^~uv'X}!O+[!O!P'^!Q![,R#U#V(U#W#X([#[#]([#a#b(i#e#f([#k#l([#m#n([~+_P!Q![+b~+eP!Q![+h~+mP[~}!O+p~+sP!Q![+v~+yP!Q![+|~,RO[~~,WY^~uv'X!O!P'^!Q![,R#U#V(U#W#X([#[#]([#a#b(i#e#f([#k#l([#m#n([~,}Oi~q~~-UOe~q~~-ZPq~!_!`-^~-cO_~~-jOf~q~~-qOo~q~~-x_p~q~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#o.w~.|_p~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#o.w~0O]qr.wst.wvw.wwx.w{|.w!O!P.w!P!Q.w!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#o.w~0|Oa~~1TOb~q~~1[ap~q~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#h.w#h#i2a#i#o.w~2fap~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#h.w#h#i3k#i#o.w~3pap~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#d.w#d#e4u#e#o.w~4zbp~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w![!]6S!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#g.w#g#h7|#h#o.w~6VP!P!Q6Y~6]P!P!Q6`~6cYOX7RZp7Rqy7Rz|7R}!`7R!a#P7R#Q#p7R#q;'S7R;'S;=`7v<%lO7R~7WY`~OX7RZp7Rqy7Rz|7R}!`7R!a#P7R#Q#p7R#q;'S7R;'S;=`7v<%lO7R~7yP;=`<%l7R~8R`p~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w![!]6S!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#o.w~9[Oh~q~~9cPm~q~!`!a9f~9kOY~~9nP;=`<%l#o",
218
+ tokenizers: [0],
219
+ topRules: { "Document": [0, 6] },
220
+ specialized: [{ term: 32, get: (value, stack) => specializeKeyword(value, stack) << 1, external: specializeKeyword }],
221
+ tokenPrec: 204
222
+ });
223
+
224
+ // src/editor/highlight-api.ts
225
+ var NODE_TO_ROLE = {
226
+ Comment: "comment",
227
+ ChartType: "chartType",
228
+ TagKeyword: "definitionKeyword",
229
+ DirectiveKeyword: "keyword",
230
+ ControlKeyword: "controlKeyword",
231
+ ModifierKeyword: "modifier",
232
+ SyncArrow: "operator",
233
+ AsyncArrow: "operator",
234
+ Dash: "operator",
235
+ Tilde: "operator",
236
+ Star: "operator",
237
+ Question: "operator",
238
+ Duration: "number",
239
+ DateLiteral: "number",
240
+ Number: "number",
241
+ Percentage: "number",
242
+ SectionMarker: "heading",
243
+ Url: "url",
244
+ ColorAnnotation: "colorAnnotation",
245
+ OpenBracket: "bracket",
246
+ CloseBracket: "bracket",
247
+ OpenParen: "bracket",
248
+ CloseParen: "bracket",
249
+ OpenAngle: "bracket",
250
+ CloseAngle: "bracket",
251
+ Pipe: "separator",
252
+ Colon: "separator",
253
+ Plus: "separator",
254
+ Comma: "punctuation",
255
+ Punct: "punctuation",
256
+ Identifier: "default"
257
+ };
258
+ var OVERRIDE_IN_LABEL = /* @__PURE__ */ new Set([
259
+ "ChartType",
260
+ "TagKeyword",
261
+ "DirectiveKeyword",
262
+ "ControlKeyword",
263
+ "ModifierKeyword",
264
+ "Number",
265
+ "Percentage",
266
+ "Duration",
267
+ "DateLiteral"
268
+ ]);
269
+ var KEYWORD_STARTS = /* @__PURE__ */ new Set([
270
+ "TagKeyword",
271
+ "DirectiveKeyword",
272
+ "ControlKeyword",
273
+ "ModifierKeyword",
274
+ "SectionMarker",
275
+ "Comment",
276
+ "Duration",
277
+ "DateLiteral"
278
+ ]);
279
+ function highlightDgmo(source) {
280
+ const tree = parser.parse(source);
281
+ const tokens = [];
282
+ let pos = 0;
283
+ const cursor = tree.cursor();
284
+ function descend() {
285
+ for (; ; ) {
286
+ if (cursor.firstChild()) continue;
287
+ emitLeaf();
288
+ while (!cursor.nextSibling()) {
289
+ if (!cursor.parent()) return;
290
+ }
291
+ }
292
+ }
293
+ function emitLeaf() {
294
+ const from = cursor.from;
295
+ const to = cursor.to;
296
+ if (from > pos) {
297
+ tokens.push({ text: source.slice(pos, from), role: "default" });
298
+ }
299
+ if (to > from) {
300
+ const role = NODE_TO_ROLE[cursor.name] ?? "default";
301
+ tokens.push({ text: source.slice(from, to), role });
302
+ }
303
+ pos = to;
304
+ }
305
+ descend();
306
+ if (pos < source.length) {
307
+ tokens.push({ text: source.slice(pos, source.length), role: "default" });
308
+ }
309
+ applyLabelOverrides(tokens);
310
+ applyNoteContent(tokens);
311
+ return tokens;
312
+ }
313
+ function applyLabelOverrides(tokens) {
314
+ const ROLE_TO_NODES = {};
315
+ for (const [node, role] of Object.entries(NODE_TO_ROLE)) {
316
+ (ROLE_TO_NODES[role] ??= []).push(node);
317
+ }
318
+ const lines = [[]];
319
+ for (let i = 0; i < tokens.length; i++) {
320
+ const t = tokens[i];
321
+ const currentLine = lines[lines.length - 1];
322
+ const role = t.role;
323
+ let nodeName = "";
324
+ for (const [node, r] of Object.entries(NODE_TO_ROLE)) {
325
+ if (r === role) {
326
+ nodeName = node;
327
+ break;
328
+ }
329
+ }
330
+ if (role === "operator") {
331
+ const text = t.text;
332
+ if (text === "->" || text.endsWith("->")) nodeName = "SyncArrow";
333
+ else if (text === "~>" || text.endsWith("~>")) nodeName = "AsyncArrow";
334
+ else if (text === "-") nodeName = "Dash";
335
+ else if (text === "~") nodeName = "Tilde";
336
+ else if (text === "*") nodeName = "Star";
337
+ else if (text === "?") nodeName = "Question";
338
+ } else if (role === "number") {
339
+ const text = t.text;
340
+ if (/^\d+[smhd]$/i.test(text)) nodeName = "Duration";
341
+ else if (/^\d{4}-\d{2}-\d{2}/.test(text)) nodeName = "DateLiteral";
342
+ else if (text.endsWith("%")) nodeName = "Percentage";
343
+ else nodeName = "Number";
344
+ } else if (role === "bracket") {
345
+ const text = t.text;
346
+ if (text === "[") nodeName = "OpenBracket";
347
+ else if (text === "]") nodeName = "CloseBracket";
348
+ else if (text === "(") nodeName = "OpenParen";
349
+ else if (text === ")") nodeName = "CloseParen";
350
+ else if (text === "<") nodeName = "OpenAngle";
351
+ else if (text === ">") nodeName = "CloseAngle";
352
+ } else if (role === "separator") {
353
+ if (t.text === "|") nodeName = "Pipe";
354
+ else if (t.text === ":") nodeName = "Colon";
355
+ else if (t.text === "+") nodeName = "Plus";
356
+ } else if (role === "punctuation") {
357
+ if (t.text === ",") nodeName = "Comma";
358
+ else nodeName = "Punct";
359
+ }
360
+ currentLine.push({ idx: i, nodeName });
361
+ if (t.text.includes("\n")) {
362
+ lines.push([]);
363
+ }
364
+ }
365
+ let seenFirstContent = false;
366
+ for (const line of lines) {
367
+ const nonWs = line.filter(
368
+ (ref) => tokens[ref.idx].text.trim().length > 0
369
+ );
370
+ if (nonWs.length === 0) continue;
371
+ const firstTok = nonWs[0];
372
+ if (KEYWORD_STARTS.has(firstTok.nodeName)) continue;
373
+ if (firstTok.nodeName === "ChartType" && !seenFirstContent) {
374
+ seenFirstContent = true;
375
+ continue;
376
+ }
377
+ seenFirstContent = true;
378
+ let firstDashTildeIdx = -1;
379
+ let lastArrowIdx = -1;
380
+ for (let li = 0; li < nonWs.length; li++) {
381
+ const ref = nonWs[li];
382
+ if ((ref.nodeName === "Dash" || ref.nodeName === "Tilde") && firstDashTildeIdx < 0) {
383
+ firstDashTildeIdx = li;
384
+ }
385
+ if (ref.nodeName === "SyncArrow" || ref.nodeName === "AsyncArrow") {
386
+ lastArrowIdx = li;
387
+ }
388
+ }
389
+ const hasArrow = firstDashTildeIdx >= 0 && lastArrowIdx > firstDashTildeIdx;
390
+ if (!hasArrow) continue;
391
+ for (let li = firstDashTildeIdx + 1; li < lastArrowIdx; li++) {
392
+ const ref = nonWs[li];
393
+ if (OVERRIDE_IN_LABEL.has(ref.nodeName)) {
394
+ tokens[ref.idx].role = "default";
395
+ }
396
+ if (ref.nodeName === "ChartType") {
397
+ tokens[ref.idx].role = "default";
398
+ }
399
+ }
400
+ }
401
+ }
402
+ var NOTE_HEAD_RE = /^note(\s|$)/i;
403
+ function applyNoteContent(tokens) {
404
+ const fullText = tokens.map((t) => t.text).join("");
405
+ const lines = fullText.split("\n");
406
+ let inNote = false;
407
+ let noteIndent = 0;
408
+ let charOffset = 0;
409
+ for (const lineText of lines) {
410
+ const lineStart = charOffset;
411
+ const lineEnd = charOffset + lineText.length;
412
+ const trimmed = lineText.trimStart();
413
+ const indent = lineText.length - trimmed.length;
414
+ if (NOTE_HEAD_RE.test(trimmed)) {
415
+ inNote = true;
416
+ noteIndent = indent;
417
+ } else if (inNote) {
418
+ if (trimmed.length === 0) {
419
+ } else if (indent > noteIndent) {
420
+ markTokensInRange(tokens, lineStart, lineEnd, "noteContent");
421
+ } else {
422
+ inNote = false;
423
+ }
424
+ }
425
+ charOffset = lineEnd + 1;
426
+ }
427
+ }
428
+ function markTokensInRange(tokens, from, to, role) {
429
+ let pos = 0;
430
+ for (const token of tokens) {
431
+ const tokenEnd = pos + token.text.length;
432
+ if (tokenEnd > from && pos < to && token.text.trim().length > 0) {
433
+ token.role = role;
434
+ }
435
+ pos = tokenEnd;
436
+ }
437
+ }
438
+ var NORD_ROLE_STYLES = {
439
+ keyword: { color: "#81A1C1", fontWeight: "bold" },
440
+ // nord9
441
+ controlKeyword: { color: "#B48EAD", fontWeight: "bold" },
442
+ // nord15
443
+ definitionKeyword: { color: "#5E81AC", fontWeight: "bold" },
444
+ // nord10
445
+ modifier: { color: "#B48EAD" },
446
+ // nord15
447
+ chartType: { color: "#D08770", fontWeight: "bold" },
448
+ // nord12
449
+ operator: { color: "#BF616A", fontWeight: "bold" },
450
+ // nord11
451
+ number: { color: "#B48EAD" },
452
+ // nord15
453
+ comment: { color: "#616E88", fontStyle: "italic" },
454
+ heading: { color: "#D08770", fontWeight: "bold" },
455
+ // nord12
456
+ bracket: { color: "#5E81AC" },
457
+ // nord10
458
+ separator: { color: "#88C0D0" },
459
+ // nord8
460
+ url: { color: "#88C0D0", textDecoration: "underline" },
461
+ // nord8
462
+ colorAnnotation: { color: "#D08770", fontStyle: "italic" },
463
+ // nord12
464
+ punctuation: { color: "#616E88" },
465
+ noteContent: { color: "#616E88", fontStyle: "italic" },
466
+ default: {}
467
+ };
468
+ var ROLE_TO_ANSI = {
469
+ comment: "\x1B[3;90m",
470
+ // italic dim
471
+ keyword: "\x1B[1;34m",
472
+ // bold blue
473
+ controlKeyword: "\x1B[1;35m",
474
+ // bold magenta
475
+ definitionKeyword: "\x1B[1;34m",
476
+ // bold blue
477
+ modifier: "\x1B[35m",
478
+ // magenta
479
+ chartType: "\x1B[1;33m",
480
+ // bold yellow
481
+ operator: "\x1B[1;31m",
482
+ // bold red
483
+ number: "\x1B[35m",
484
+ // magenta
485
+ heading: "\x1B[1;33m",
486
+ // bold yellow
487
+ bracket: "\x1B[34m",
488
+ // blue
489
+ separator: "\x1B[36m",
490
+ // cyan
491
+ url: "\x1B[4;36m",
492
+ // underline cyan
493
+ colorAnnotation: "\x1B[3;33m",
494
+ // italic yellow
495
+ punctuation: "\x1B[90m",
496
+ // dim
497
+ noteContent: "\x1B[3;90m"
498
+ // italic dim
499
+ };
500
+ var ANSI_RESET = "\x1B[0m";
501
+ function renderAnsi(tokens, useColor) {
502
+ if (!useColor) {
503
+ return tokens.map((t) => t.text).join("");
504
+ }
505
+ let out = "";
506
+ let inStyled = false;
507
+ for (const token of tokens) {
508
+ const ansi = ROLE_TO_ANSI[token.role];
509
+ if (ansi) {
510
+ if (inStyled) out += ANSI_RESET;
511
+ out += ansi + token.text;
512
+ inStyled = true;
513
+ } else {
514
+ if (inStyled) {
515
+ out += ANSI_RESET;
516
+ inStyled = false;
517
+ }
518
+ out += token.text;
519
+ }
520
+ }
521
+ out += ANSI_RESET;
522
+ return out;
523
+ }
524
+ export {
525
+ NORD_ROLE_STYLES,
526
+ ROLE_TO_ANSI,
527
+ highlightDgmo,
528
+ renderAnsi
529
+ };
530
+ //# sourceMappingURL=highlight.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/editor/dgmo.grammar.js","../src/editor/dgmo.grammar.terms.js","../src/editor/keywords.ts","../src/editor/tokens.ts","../src/editor/highlight-api.ts"],"sourcesContent":["// This file was generated by lezer-generator. You probably shouldn't edit it.\nimport {LRParser} from \"@lezer/lr\"\nimport {specializeKeyword} from \"./tokens\"\nexport const parser = LRParser.deserialize({\n version: 14,\n states: \"!WQVQPOOOOQO'#DU'#DUOOQO'#DP'#DPO%]QPO'#CdOOQO'#DO'#DOQVQPOOOOQO-E6}-E6}OOQO,59O,59OOOQO-E6|-E6|\",\n stateData: \"&Q~OvOS~OPPOQPORPOSPOTPOVSOXPOYPOZPO[PO]PO^PO_PO`POaPObPOcPOdPOePOfPOgPOhPOiPOjPOkPOlPOmPOnPOoPOpPOqPOwSO~OPPOQPORPOSPOTPOXPOYPOZPO[PO]PO^PO_PO`POaPObPOcPOdPOePOfPOgPOhPOiPOjPOkPOlPOmPOnPOoPOpPOqPO~OwVO~P#]OVXYZ[]^_`ghijklmnoabcdefpqk~\",\n goto: \"!byPPPPPPPPzPPPPPPPPPPPPPPPPPPPPPPPPP!O!UPPPP!]TSOTQTORWTSROTRURVQORT\",\n nodeNames: \"⚠ ChartType TagKeyword DirectiveKeyword ControlKeyword ModifierKeyword Document Comment ContentLine SyncArrow AsyncArrow Duration DateLiteral Percentage Number SectionMarker Url OpenBracket CloseBracket OpenParen CloseParen OpenAngle CloseAngle ColorAnnotation Pipe Colon Comma Plus Dash Tilde Star Question Identifier Punct\",\n maxTerm: 40,\n skippedNodes: [0],\n repeatNodeCount: 2,\n tokenData: \"9q~RxOX#oXY#tYZ$PZp#opq#tqt#oux#oxy$Uyz$tz{${{|%S|}%Z}!O%b!O!P#o!P!Q%q!Q![&b![!],v!]!^#o!^!_,}!_!`-U!`!a-c!a!b-j!b!c#o!c!}-q!}#O0w#O#P#o#P#Q0|#Q#R#o#R#S-q#S#T#o#T#[-q#[#]1T#]#o-q#o#p#o#p#q9T#q#r#o#r#s9[#s;'S#o;'S;=`9k<%lO#o~#tOq~~#yQv~XY#tpq#t~$UOw~~$]Qc~q~}!O$c#T#o$c~$fRyz$o}!O$c#T#o$c~$tOg~~${Od~q~~%SOn~q~~%ZOk~q~~%bOj~q~~%iPl~q~!`!a%l~%qOX~~%vPq~!P!Q%y~&OSV~OY%yZ;'S%y;'S;=`&[<%lO%y~&_P;=`<%l%y~&iY^~q~uv'X!O!P'^!Q![(z#U#V(U#W#X([#[#]([#a#b(i#e#f([#k#l([#m#n([~'^O]~~'aP!Q!['d~'iX^~uv'X!Q!['d#U#V(U#W#X([#[#]([#a#b(i#e#f([#k#l([#m#n([~(XP#W#X([~(aPZ~!a!b(d~(iOZ~~(nQZ~!a!b(d#]#^(t~(wP#b#c([~)PY^~uv'X!O!P'^!Q![)o#U#V(U#W#X([#[#]([#a#b(i#e#f([#k#l([#m#n([~)tY^~uv'X!O!P'^!Q![*d#U#V(U#W#X([#[#]([#a#b(i#e#f([#k#l([#m#n([~*iZ^~uv'X}!O+[!O!P'^!Q![,R#U#V(U#W#X([#[#]([#a#b(i#e#f([#k#l([#m#n([~+_P!Q![+b~+eP!Q![+h~+mP[~}!O+p~+sP!Q![+v~+yP!Q![+|~,RO[~~,WY^~uv'X!O!P'^!Q![,R#U#V(U#W#X([#[#]([#a#b(i#e#f([#k#l([#m#n([~,}Oi~q~~-UOe~q~~-ZPq~!_!`-^~-cO_~~-jOf~q~~-qOo~q~~-x_p~q~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#o.w~.|_p~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#o.w~0O]qr.wst.wvw.wwx.w{|.w!O!P.w!P!Q.w!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#o.w~0|Oa~~1TOb~q~~1[ap~q~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#h.w#h#i2a#i#o.w~2fap~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#h.w#h#i3k#i#o.w~3pap~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#d.w#d#e4u#e#o.w~4zbp~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w![!]6S!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#g.w#g#h7|#h#o.w~6VP!P!Q6Y~6]P!P!Q6`~6cYOX7RZp7Rqy7Rz|7R}!`7R!a#P7R#Q#p7R#q;'S7R;'S;=`7v<%lO7R~7WY`~OX7RZp7Rqy7Rz|7R}!`7R!a#P7R#Q#p7R#q;'S7R;'S;=`7v<%lO7R~7yP;=`<%l7R~8R`p~qr.wst.wvw.wwx.w{|.w}!O/{!O!P.w!P!Q.w!Q![.w![!]6S!_!`.w!a!b.w!b!c.w!c!}.w#R#S.w#T#o.w~9[Oh~q~~9cPm~q~!`!a9f~9kOY~~9nP;=`<%l#o\",\n tokenizers: [0],\n topRules: {\"Document\":[0,6]},\n specialized: [{term: 32, get: (value, stack) => (specializeKeyword(value, stack) << 1), external: specializeKeyword}],\n tokenPrec: 204\n})\n","// This file was generated by lezer-generator. You probably shouldn't edit it.\nexport const\n ChartType = 1,\n TagKeyword = 2,\n DirectiveKeyword = 3,\n ControlKeyword = 4,\n ModifierKeyword = 5,\n Document = 6,\n Comment = 7,\n ContentLine = 8,\n SyncArrow = 9,\n AsyncArrow = 10,\n Duration = 11,\n DateLiteral = 12,\n Percentage = 13,\n Number = 14,\n SectionMarker = 15,\n Url = 16,\n OpenBracket = 17,\n CloseBracket = 18,\n OpenParen = 19,\n CloseParen = 20,\n OpenAngle = 21,\n CloseAngle = 22,\n ColorAnnotation = 23,\n Pipe = 24,\n Colon = 25,\n Comma = 26,\n Plus = 27,\n Dash = 28,\n Tilde = 29,\n Star = 30,\n Question = 31,\n Identifier = 32,\n Punct = 33\n","/** All supported DGMO chart types. */\nexport const CHART_TYPES = new Set([\n // Diagram types\n 'sequence',\n 'flowchart',\n 'class',\n 'er',\n 'org',\n 'kanban',\n 'c4',\n 'initiative-status',\n 'state',\n 'sitemap',\n 'infra',\n 'gantt',\n // Data chart types\n 'bar',\n 'line',\n 'pie',\n 'doughnut',\n 'area',\n 'polar-area',\n 'radar',\n 'bar-stacked',\n 'multi-line',\n 'scatter',\n 'sankey',\n 'chord',\n 'function',\n 'heatmap',\n 'funnel',\n // Visualization types\n 'slope',\n 'wordcloud',\n 'arc',\n 'timeline',\n 'venn',\n 'quadrant',\n]);\n\n/** Metadata keys recognized across chart types. */\nexport const METADATA_KEYS = new Set([\n 'title',\n 'series',\n 'orientation',\n 'x-label',\n 'y-label',\n 'size-label',\n 'x',\n 'columns',\n 'rows',\n 'labels',\n 'rotate',\n 'max',\n 'size',\n 'order',\n 'sort',\n 'scale',\n 'values',\n 'notation',\n 'x-axis',\n 'y-axis',\n 'top-right',\n 'top-left',\n 'bottom-right',\n 'bottom-left',\n]);\n\n/** Tag declaration keyword. */\nexport const TAG_KEYWORD = 'tag';\n\n/** Directive keywords — commands that configure chart behavior. */\nexport const DIRECTIVE_KEYWORDS = new Set([\n // Gantt\n 'start',\n 'era',\n 'marker',\n 'holiday',\n 'workweek',\n 'today-marker',\n 'critical-path',\n 'no-dependencies',\n 'sort',\n // Tags\n 'tags',\n 'import',\n 'active-tag',\n 'hide',\n // ER\n 'notation',\n // Class\n 'extends',\n 'implements',\n 'abstract',\n 'interface',\n 'enum',\n // C4\n 'containers',\n 'components',\n 'deployment',\n // Infra directives\n 'sub-node-label',\n 'show-sub-node-count',\n 'no-auto-color',\n // Infra node properties\n 'description',\n 'instances',\n 'max-rps',\n 'latency-ms',\n 'uptime',\n 'firewall-block',\n 'ratelimit-rps',\n 'cb-error-threshold',\n 'cb-latency-threshold-ms',\n 'buffer',\n 'drain-rate',\n 'retention-hours',\n 'partitions',\n 'split',\n 'slo-p90-latency-ms',\n 'slo-availability',\n 'cache-hit',\n 'concurrency',\n 'duration-ms',\n 'cold-start-ms',\n 'rps',\n // Sequence\n 'activations',\n 'no-activations',\n 'collapse-notes',\n 'no-collapse-notes',\n // Data charts\n 'stacked',\n 'no-label-name',\n 'no-label-value',\n 'no-label-percent',\n // Slope\n 'period',\n // Quadrant\n 'x-axis',\n 'y-axis',\n 'top-right',\n 'top-left',\n 'bottom-right',\n 'bottom-left',\n // Layout\n 'direction-tb',\n 'direction-lr',\n // Initiative-status\n 'contains',\n // Data chart metadata\n 'title',\n 'series',\n 'orientation',\n 'x-label',\n 'y-label',\n 'size-label',\n 'columns',\n 'rows',\n 'labels',\n 'rotate',\n 'scale',\n 'values',\n]);\n\n/** Control flow keywords — structural blocks. */\nexport const CONTROL_KEYWORDS = new Set([\n 'if',\n 'else',\n 'loop',\n 'parallel',\n 'note',\n]);\n\n/** Status keywords — initiative-status, kanban. */\nexport const STATUS_KEYWORDS = new Set([\n 'na',\n 'todo',\n 'wip',\n 'done',\n 'blocked',\n 'in-progress',\n 'backlog',\n 'ready',\n]);\n\n/** Modifier keywords — adjust declarations. */\nexport const MODIFIER_KEYWORDS = new Set([\n 'alias',\n 'aka',\n 'position',\n 'default',\n // Sequence participant types\n 'actor',\n 'service',\n 'database',\n 'queue',\n 'cache',\n 'gateway',\n 'external',\n 'networking',\n 'frontend',\n // C4 element types\n 'system',\n 'person',\n 'container',\n 'component',\n // ER column modifiers\n 'pk',\n 'fk',\n 'nullable',\n 'unique',\n // ER data types\n 'int',\n 'varchar',\n 'text',\n 'boolean',\n 'date',\n 'timestamp',\n 'float',\n 'decimal',\n]);\n","import {\n ChartType,\n TagKeyword,\n DirectiveKeyword,\n ControlKeyword,\n ModifierKeyword,\n} from './dgmo.grammar.terms.js';\nimport {\n CHART_TYPES,\n TAG_KEYWORD,\n DIRECTIVE_KEYWORDS,\n CONTROL_KEYWORDS,\n STATUS_KEYWORDS,\n MODIFIER_KEYWORDS,\n} from './keywords';\n\n/**\n * Keyword specializer for the Lezer grammar.\n * Called on every Identifier token — returns a specialized term ID\n * or -1 to keep it as a plain Identifier.\n */\nexport function specializeKeyword(value: string): number {\n if (CHART_TYPES.has(value)) return ChartType;\n if (value === TAG_KEYWORD) return TagKeyword;\n if (DIRECTIVE_KEYWORDS.has(value)) return DirectiveKeyword;\n if (CONTROL_KEYWORDS.has(value)) return ControlKeyword;\n if (STATUS_KEYWORDS.has(value)) return ModifierKeyword;\n if (MODIFIER_KEYWORDS.has(value)) return ModifierKeyword;\n return -1;\n}\n","/**\n * Standalone DGMO syntax highlighter — no CodeMirror dependency.\n *\n * Exports:\n * - `highlightDgmo(source)` → `HighlightToken[]` (consumer-agnostic)\n * - `NORD_ROLE_STYLES` — inline style objects keyed by role (for React/Astro)\n * - `ROLE_TO_ANSI` — ANSI escape codes keyed by role (for CLI)\n *\n * Uses the raw Lezer parser directly — keyword specialization is wired into\n * the grammar, so `parser.parse()` runs it automatically.\n *\n * @module @diagrammo/dgmo/highlight\n */\n\nimport { parser } from './dgmo.grammar.js';\n\n// ============================================================\n// Types\n// ============================================================\n\nexport interface HighlightToken {\n text: string;\n role: string;\n}\n\n// ============================================================\n// NODE_TO_ROLE — keep in sync with highlight.ts\n// ============================================================\n\nconst NODE_TO_ROLE: Record<string, string> = {\n Comment: 'comment',\n ChartType: 'chartType',\n TagKeyword: 'definitionKeyword',\n DirectiveKeyword: 'keyword',\n ControlKeyword: 'controlKeyword',\n ModifierKeyword: 'modifier',\n SyncArrow: 'operator',\n AsyncArrow: 'operator',\n Dash: 'operator',\n Tilde: 'operator',\n Star: 'operator',\n Question: 'operator',\n Duration: 'number',\n DateLiteral: 'number',\n Number: 'number',\n Percentage: 'number',\n SectionMarker: 'heading',\n Url: 'url',\n ColorAnnotation: 'colorAnnotation',\n OpenBracket: 'bracket',\n CloseBracket: 'bracket',\n OpenParen: 'bracket',\n CloseParen: 'bracket',\n OpenAngle: 'bracket',\n CloseAngle: 'bracket',\n Pipe: 'separator',\n Colon: 'separator',\n Plus: 'separator',\n Comma: 'punctuation',\n Punct: 'punctuation',\n Identifier: 'default',\n};\n\n// ============================================================\n// Entity detection — keep in sync with entity-highlight.ts\n// ============================================================\n\n/** Tokens that have grammar-level styling which should be overridden in labels. */\nconst OVERRIDE_IN_LABEL = new Set([\n 'ChartType',\n 'TagKeyword',\n 'DirectiveKeyword',\n 'ControlKeyword',\n 'ModifierKeyword',\n 'Number',\n 'Percentage',\n 'Duration',\n 'DateLiteral',\n]);\n\n/** Lines starting with these are keyword-led — not entity declarations. */\nconst KEYWORD_STARTS = new Set([\n 'TagKeyword',\n 'DirectiveKeyword',\n 'ControlKeyword',\n 'ModifierKeyword',\n 'SectionMarker',\n 'Comment',\n 'Duration',\n 'DateLiteral',\n]);\n\n\n// ============================================================\n// Core: highlightDgmo()\n// ============================================================\n\n/**\n * Tokenize DGMO source into annotated highlight spans.\n *\n * Guarantees lossless round-trip:\n * `highlightDgmo(src).map(t => t.text).join('') === src`\n */\nexport function highlightDgmo(source: string): HighlightToken[] {\n const tree = parser.parse(source);\n const tokens: HighlightToken[] = [];\n\n // Phase 1: Walk tree cursor, collect leaf tokens with gap filling\n let pos = 0;\n const cursor = tree.cursor();\n\n // Descend to leaves, process them, then advance via next() or parent+next()\n function descend(): void {\n for (;;) {\n // Try to go deeper\n if (cursor.firstChild()) continue;\n\n // At a leaf — emit it\n emitLeaf();\n\n // Try to advance to next sibling or ascend\n while (!cursor.nextSibling()) {\n if (!cursor.parent()) return; // back at root — done\n }\n\n // Now at next sibling — loop will try to descend into it\n }\n }\n\n function emitLeaf(): void {\n const from = cursor.from;\n const to = cursor.to;\n\n // Fill gap before this node\n if (from > pos) {\n tokens.push({ text: source.slice(pos, from), role: 'default' });\n }\n\n // Emit this leaf node\n if (to > from) {\n const role = NODE_TO_ROLE[cursor.name] ?? 'default';\n tokens.push({ text: source.slice(from, to), role });\n }\n\n pos = to;\n }\n\n descend();\n\n // Fill trailing gap\n if (pos < source.length) {\n tokens.push({ text: source.slice(pos, source.length), role: 'default' });\n }\n\n // Phase 2: Post-process — entity detection (label override only)\n applyLabelOverrides(tokens);\n\n // Phase 3: Post-process — note content detection\n applyNoteContent(tokens);\n\n return tokens;\n}\n\n// ============================================================\n// Post-processing: label overrides\n// ============================================================\n\ninterface LineTokenRef {\n /** Index into the flat token array. */\n idx: number;\n /** Grammar node name (looked up from role + NODE_TO_ROLE reverse). */\n nodeName: string;\n}\n\n/**\n * Override keyword/number tokens in message-label positions to `default` role.\n *\n * A \"label\" is the span between the first Dash/Tilde and the last arrow on a\n * content line. Tokens in OVERRIDE_IN_LABEL within that zone get their role\n * set to `default` so they render as plain text.\n *\n * Also handles ChartType tokens on non-first content lines — they become\n * `default` in labels.\n */\nfunction applyLabelOverrides(tokens: HighlightToken[]): void {\n // Build reverse map: role → possible node names (for override detection)\n const ROLE_TO_NODES: Record<string, string[]> = {};\n for (const [node, role] of Object.entries(NODE_TO_ROLE)) {\n (ROLE_TO_NODES[role] ??= []).push(node);\n }\n\n // Split tokens into lines\n const lines: LineTokenRef[][] = [[]];\n for (let i = 0; i < tokens.length; i++) {\n const t = tokens[i];\n // If the token contains newlines, it belongs to the current line\n // but signals the start of a new line after it\n const currentLine = lines[lines.length - 1];\n const role = t.role;\n\n // Determine which node name this token likely had\n let nodeName = '';\n for (const [node, r] of Object.entries(NODE_TO_ROLE)) {\n if (r === role) {\n nodeName = node;\n break;\n }\n }\n // For roles mapping to multiple nodes, refine\n if (role === 'operator') {\n const text = t.text;\n if (text === '->' || text.endsWith('->')) nodeName = 'SyncArrow';\n else if (text === '~>' || text.endsWith('~>')) nodeName = 'AsyncArrow';\n else if (text === '-') nodeName = 'Dash';\n else if (text === '~') nodeName = 'Tilde';\n else if (text === '*') nodeName = 'Star';\n else if (text === '?') nodeName = 'Question';\n } else if (role === 'number') {\n const text = t.text;\n if (/^\\d+[smhd]$/i.test(text)) nodeName = 'Duration';\n else if (/^\\d{4}-\\d{2}-\\d{2}/.test(text)) nodeName = 'DateLiteral';\n else if (text.endsWith('%')) nodeName = 'Percentage';\n else nodeName = 'Number';\n } else if (role === 'bracket') {\n const text = t.text;\n if (text === '[') nodeName = 'OpenBracket';\n else if (text === ']') nodeName = 'CloseBracket';\n else if (text === '(') nodeName = 'OpenParen';\n else if (text === ')') nodeName = 'CloseParen';\n else if (text === '<') nodeName = 'OpenAngle';\n else if (text === '>') nodeName = 'CloseAngle';\n } else if (role === 'separator') {\n if (t.text === '|') nodeName = 'Pipe';\n else if (t.text === ':') nodeName = 'Colon';\n else if (t.text === '+') nodeName = 'Plus';\n } else if (role === 'punctuation') {\n if (t.text === ',') nodeName = 'Comma';\n else nodeName = 'Punct';\n }\n\n currentLine.push({ idx: i, nodeName });\n\n // Check if token text ends with newline — start a new line\n if (t.text.includes('\\n')) {\n lines.push([]);\n }\n }\n\n // Track first content line (for ChartType handling)\n let seenFirstContent = false;\n\n for (const line of lines) {\n // Skip empty lines and whitespace-only\n const nonWs = line.filter(\n (ref) => tokens[ref.idx].text.trim().length > 0\n );\n if (nonWs.length === 0) continue;\n\n const firstTok = nonWs[0];\n\n // Skip keyword-led lines\n if (KEYWORD_STARTS.has(firstTok.nodeName)) continue;\n\n // First-line chart type — skip\n if (firstTok.nodeName === 'ChartType' && !seenFirstContent) {\n seenFirstContent = true;\n continue;\n }\n seenFirstContent = true;\n\n // Find structural boundaries within this line\n let firstDashTildeIdx = -1;\n let lastArrowIdx = -1;\n\n for (let li = 0; li < nonWs.length; li++) {\n const ref = nonWs[li];\n if (\n (ref.nodeName === 'Dash' || ref.nodeName === 'Tilde') &&\n firstDashTildeIdx < 0\n ) {\n firstDashTildeIdx = li;\n }\n if (ref.nodeName === 'SyncArrow' || ref.nodeName === 'AsyncArrow') {\n lastArrowIdx = li;\n }\n }\n\n const hasArrow =\n firstDashTildeIdx >= 0 && lastArrowIdx > firstDashTildeIdx;\n\n if (!hasArrow) continue;\n\n // Override tokens in label zone (between first dash/tilde and last arrow)\n for (let li = firstDashTildeIdx + 1; li < lastArrowIdx; li++) {\n const ref = nonWs[li];\n if (OVERRIDE_IN_LABEL.has(ref.nodeName)) {\n tokens[ref.idx].role = 'default';\n }\n // ChartType in label also overridden\n if (ref.nodeName === 'ChartType') {\n tokens[ref.idx].role = 'default';\n }\n }\n }\n}\n\n// ============================================================\n// Post-processing: note content detection\n// ============================================================\n\nconst NOTE_HEAD_RE = /^note(\\s|$)/i;\n\n/**\n * Detect `note` keyword lines and mark indented followers as `noteContent`.\n */\nfunction applyNoteContent(tokens: HighlightToken[]): void {\n // Reconstruct lines from token text\n const fullText = tokens.map((t) => t.text).join('');\n const lines = fullText.split('\\n');\n\n let inNote = false;\n let noteIndent = 0;\n let charOffset = 0;\n\n for (const lineText of lines) {\n const lineStart = charOffset;\n const lineEnd = charOffset + lineText.length;\n const trimmed = lineText.trimStart();\n const indent = lineText.length - trimmed.length;\n\n if (NOTE_HEAD_RE.test(trimmed)) {\n inNote = true;\n noteIndent = indent;\n } else if (inNote) {\n if (trimmed.length === 0) {\n // Blank line — stays in note block\n } else if (indent > noteIndent) {\n // Mark all tokens within this line range as noteContent\n markTokensInRange(tokens, lineStart, lineEnd, 'noteContent');\n } else {\n inNote = false;\n }\n }\n\n charOffset = lineEnd + 1; // +1 for the \\n\n }\n}\n\n/**\n * Set the role of all tokens overlapping [from, to) to the given role.\n */\nfunction markTokensInRange(\n tokens: HighlightToken[],\n from: number,\n to: number,\n role: string\n): void {\n let pos = 0;\n for (const token of tokens) {\n const tokenEnd = pos + token.text.length;\n // Token overlaps range and is not just whitespace\n if (tokenEnd > from && pos < to && token.text.trim().length > 0) {\n token.role = role;\n }\n pos = tokenEnd;\n }\n}\n\n// ============================================================\n// NORD_ROLE_STYLES — hardcoded Nord dark palette for static contexts\n// ============================================================\n\nexport const NORD_ROLE_STYLES: Record<\n string,\n Record<string, string>\n> = {\n keyword: { color: '#81A1C1', fontWeight: 'bold' }, // nord9\n controlKeyword: { color: '#B48EAD', fontWeight: 'bold' }, // nord15\n definitionKeyword: { color: '#5E81AC', fontWeight: 'bold' }, // nord10\n modifier: { color: '#B48EAD' }, // nord15\n chartType: { color: '#D08770', fontWeight: 'bold' }, // nord12\n operator: { color: '#BF616A', fontWeight: 'bold' }, // nord11\n number: { color: '#B48EAD' }, // nord15\n comment: { color: '#616E88', fontStyle: 'italic' },\n heading: { color: '#D08770', fontWeight: 'bold' }, // nord12\n bracket: { color: '#5E81AC' }, // nord10\n separator: { color: '#88C0D0' }, // nord8\n url: { color: '#88C0D0', textDecoration: 'underline' }, // nord8\n colorAnnotation: { color: '#D08770', fontStyle: 'italic' }, // nord12\n punctuation: { color: '#616E88' },\n noteContent: { color: '#616E88', fontStyle: 'italic' },\n default: {},\n};\n\n// ============================================================\n// ROLE_TO_ANSI — ANSI escape codes for CLI output\n// ============================================================\n\nexport const ROLE_TO_ANSI: Record<string, string> = {\n comment: '\\x1b[3;90m', // italic dim\n keyword: '\\x1b[1;34m', // bold blue\n controlKeyword: '\\x1b[1;35m', // bold magenta\n definitionKeyword: '\\x1b[1;34m', // bold blue\n modifier: '\\x1b[35m', // magenta\n chartType: '\\x1b[1;33m', // bold yellow\n operator: '\\x1b[1;31m', // bold red\n number: '\\x1b[35m', // magenta\n heading: '\\x1b[1;33m', // bold yellow\n bracket: '\\x1b[34m', // blue\n separator: '\\x1b[36m', // cyan\n url: '\\x1b[4;36m', // underline cyan\n colorAnnotation: '\\x1b[3;33m', // italic yellow\n punctuation: '\\x1b[90m', // dim\n noteContent: '\\x1b[3;90m', // italic dim\n};\n\nconst ANSI_RESET = '\\x1b[0m';\n\n/**\n * Render highlighted tokens to an ANSI string for terminal display.\n */\nexport function renderAnsi(\n tokens: HighlightToken[],\n useColor: boolean\n): string {\n if (!useColor) {\n return tokens.map((t) => t.text).join('');\n }\n\n let out = '';\n let inStyled = false;\n\n for (const token of tokens) {\n const ansi = ROLE_TO_ANSI[token.role];\n if (ansi) {\n if (inStyled) out += ANSI_RESET;\n out += ansi + token.text;\n inStyled = true;\n } else {\n if (inStyled) {\n out += ANSI_RESET;\n inStyled = false;\n }\n out += token.text;\n }\n }\n\n // Final reset to prevent terminal style leakage\n out += ANSI_RESET;\n return out;\n}\n"],"mappings":"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","names":[]}