@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,560 @@
1
+ "use strict";
2
+ var __defProp = Object.defineProperty;
3
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
+ var __getOwnPropNames = Object.getOwnPropertyNames;
5
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
6
+ var __export = (target, all) => {
7
+ for (var name in all)
8
+ __defProp(target, name, { get: all[name], enumerable: true });
9
+ };
10
+ var __copyProps = (to, from, except, desc) => {
11
+ if (from && typeof from === "object" || typeof from === "function") {
12
+ for (let key of __getOwnPropNames(from))
13
+ if (!__hasOwnProp.call(to, key) && key !== except)
14
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
15
+ }
16
+ return to;
17
+ };
18
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
19
+
20
+ // src/editor/highlight-api.ts
21
+ var highlight_api_exports = {};
22
+ __export(highlight_api_exports, {
23
+ NORD_ROLE_STYLES: () => NORD_ROLE_STYLES,
24
+ ROLE_TO_ANSI: () => ROLE_TO_ANSI,
25
+ highlightDgmo: () => highlightDgmo,
26
+ renderAnsi: () => renderAnsi
27
+ });
28
+ module.exports = __toCommonJS(highlight_api_exports);
29
+
30
+ // src/editor/dgmo.grammar.js
31
+ var import_lr = require("@lezer/lr");
32
+
33
+ // src/editor/dgmo.grammar.terms.js
34
+ var ChartType = 1;
35
+ var TagKeyword = 2;
36
+ var DirectiveKeyword = 3;
37
+ var ControlKeyword = 4;
38
+ var ModifierKeyword = 5;
39
+
40
+ // src/editor/keywords.ts
41
+ var CHART_TYPES = /* @__PURE__ */ new Set([
42
+ // Diagram types
43
+ "sequence",
44
+ "flowchart",
45
+ "class",
46
+ "er",
47
+ "org",
48
+ "kanban",
49
+ "c4",
50
+ "initiative-status",
51
+ "state",
52
+ "sitemap",
53
+ "infra",
54
+ "gantt",
55
+ // Data chart types
56
+ "bar",
57
+ "line",
58
+ "pie",
59
+ "doughnut",
60
+ "area",
61
+ "polar-area",
62
+ "radar",
63
+ "bar-stacked",
64
+ "multi-line",
65
+ "scatter",
66
+ "sankey",
67
+ "chord",
68
+ "function",
69
+ "heatmap",
70
+ "funnel",
71
+ // Visualization types
72
+ "slope",
73
+ "wordcloud",
74
+ "arc",
75
+ "timeline",
76
+ "venn",
77
+ "quadrant"
78
+ ]);
79
+ var TAG_KEYWORD = "tag";
80
+ var DIRECTIVE_KEYWORDS = /* @__PURE__ */ new Set([
81
+ // Gantt
82
+ "start",
83
+ "era",
84
+ "marker",
85
+ "holiday",
86
+ "workweek",
87
+ "today-marker",
88
+ "critical-path",
89
+ "no-dependencies",
90
+ "sort",
91
+ // Tags
92
+ "tags",
93
+ "import",
94
+ "active-tag",
95
+ "hide",
96
+ // ER
97
+ "notation",
98
+ // Class
99
+ "extends",
100
+ "implements",
101
+ "abstract",
102
+ "interface",
103
+ "enum",
104
+ // C4
105
+ "containers",
106
+ "components",
107
+ "deployment",
108
+ // Infra directives
109
+ "sub-node-label",
110
+ "show-sub-node-count",
111
+ "no-auto-color",
112
+ // Infra node properties
113
+ "description",
114
+ "instances",
115
+ "max-rps",
116
+ "latency-ms",
117
+ "uptime",
118
+ "firewall-block",
119
+ "ratelimit-rps",
120
+ "cb-error-threshold",
121
+ "cb-latency-threshold-ms",
122
+ "buffer",
123
+ "drain-rate",
124
+ "retention-hours",
125
+ "partitions",
126
+ "split",
127
+ "slo-p90-latency-ms",
128
+ "slo-availability",
129
+ "cache-hit",
130
+ "concurrency",
131
+ "duration-ms",
132
+ "cold-start-ms",
133
+ "rps",
134
+ // Sequence
135
+ "activations",
136
+ "no-activations",
137
+ "collapse-notes",
138
+ "no-collapse-notes",
139
+ // Data charts
140
+ "stacked",
141
+ "no-label-name",
142
+ "no-label-value",
143
+ "no-label-percent",
144
+ // Slope
145
+ "period",
146
+ // Quadrant
147
+ "x-axis",
148
+ "y-axis",
149
+ "top-right",
150
+ "top-left",
151
+ "bottom-right",
152
+ "bottom-left",
153
+ // Layout
154
+ "direction-tb",
155
+ "direction-lr",
156
+ // Initiative-status
157
+ "contains",
158
+ // Data chart metadata
159
+ "title",
160
+ "series",
161
+ "orientation",
162
+ "x-label",
163
+ "y-label",
164
+ "size-label",
165
+ "columns",
166
+ "rows",
167
+ "labels",
168
+ "rotate",
169
+ "scale",
170
+ "values"
171
+ ]);
172
+ var CONTROL_KEYWORDS = /* @__PURE__ */ new Set([
173
+ "if",
174
+ "else",
175
+ "loop",
176
+ "parallel",
177
+ "note"
178
+ ]);
179
+ var STATUS_KEYWORDS = /* @__PURE__ */ new Set([
180
+ "na",
181
+ "todo",
182
+ "wip",
183
+ "done",
184
+ "blocked",
185
+ "in-progress",
186
+ "backlog",
187
+ "ready"
188
+ ]);
189
+ var MODIFIER_KEYWORDS = /* @__PURE__ */ new Set([
190
+ "alias",
191
+ "aka",
192
+ "position",
193
+ "default",
194
+ // Sequence participant types
195
+ "actor",
196
+ "service",
197
+ "database",
198
+ "queue",
199
+ "cache",
200
+ "gateway",
201
+ "external",
202
+ "networking",
203
+ "frontend",
204
+ // C4 element types
205
+ "system",
206
+ "person",
207
+ "container",
208
+ "component",
209
+ // ER column modifiers
210
+ "pk",
211
+ "fk",
212
+ "nullable",
213
+ "unique",
214
+ // ER data types
215
+ "int",
216
+ "varchar",
217
+ "text",
218
+ "boolean",
219
+ "date",
220
+ "timestamp",
221
+ "float",
222
+ "decimal"
223
+ ]);
224
+
225
+ // src/editor/tokens.ts
226
+ function specializeKeyword(value) {
227
+ if (CHART_TYPES.has(value)) return ChartType;
228
+ if (value === TAG_KEYWORD) return TagKeyword;
229
+ if (DIRECTIVE_KEYWORDS.has(value)) return DirectiveKeyword;
230
+ if (CONTROL_KEYWORDS.has(value)) return ControlKeyword;
231
+ if (STATUS_KEYWORDS.has(value)) return ModifierKeyword;
232
+ if (MODIFIER_KEYWORDS.has(value)) return ModifierKeyword;
233
+ return -1;
234
+ }
235
+
236
+ // src/editor/dgmo.grammar.js
237
+ var parser = import_lr.LRParser.deserialize({
238
+ version: 14,
239
+ states: "!WQVQPOOOOQO'#DU'#DUOOQO'#DP'#DPO%]QPO'#CdOOQO'#DO'#DOQVQPOOOOQO-E6}-E6}OOQO,59O,59OOOQO-E6|-E6|",
240
+ stateData: "&Q~OvOS~OPPOQPORPOSPOTPOVSOXPOYPOZPO[PO]PO^PO_PO`POaPObPOcPOdPOePOfPOgPOhPOiPOjPOkPOlPOmPOnPOoPOpPOqPOwSO~OPPOQPORPOSPOTPOXPOYPOZPO[PO]PO^PO_PO`POaPObPOcPOdPOePOfPOgPOhPOiPOjPOkPOlPOmPOnPOoPOpPOqPO~OwVO~P#]OVXYZ[]^_`ghijklmnoabcdefpqk~",
241
+ goto: "!byPPPPPPPPzPPPPPPPPPPPPPPPPPPPPPPPPP!O!UPPPP!]TSOTQTORWTSROTRURVQORT",
242
+ 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",
243
+ maxTerm: 40,
244
+ skippedNodes: [0],
245
+ repeatNodeCount: 2,
246
+ 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",
247
+ tokenizers: [0],
248
+ topRules: { "Document": [0, 6] },
249
+ specialized: [{ term: 32, get: (value, stack) => specializeKeyword(value, stack) << 1, external: specializeKeyword }],
250
+ tokenPrec: 204
251
+ });
252
+
253
+ // src/editor/highlight-api.ts
254
+ var NODE_TO_ROLE = {
255
+ Comment: "comment",
256
+ ChartType: "chartType",
257
+ TagKeyword: "definitionKeyword",
258
+ DirectiveKeyword: "keyword",
259
+ ControlKeyword: "controlKeyword",
260
+ ModifierKeyword: "modifier",
261
+ SyncArrow: "operator",
262
+ AsyncArrow: "operator",
263
+ Dash: "operator",
264
+ Tilde: "operator",
265
+ Star: "operator",
266
+ Question: "operator",
267
+ Duration: "number",
268
+ DateLiteral: "number",
269
+ Number: "number",
270
+ Percentage: "number",
271
+ SectionMarker: "heading",
272
+ Url: "url",
273
+ ColorAnnotation: "colorAnnotation",
274
+ OpenBracket: "bracket",
275
+ CloseBracket: "bracket",
276
+ OpenParen: "bracket",
277
+ CloseParen: "bracket",
278
+ OpenAngle: "bracket",
279
+ CloseAngle: "bracket",
280
+ Pipe: "separator",
281
+ Colon: "separator",
282
+ Plus: "separator",
283
+ Comma: "punctuation",
284
+ Punct: "punctuation",
285
+ Identifier: "default"
286
+ };
287
+ var OVERRIDE_IN_LABEL = /* @__PURE__ */ new Set([
288
+ "ChartType",
289
+ "TagKeyword",
290
+ "DirectiveKeyword",
291
+ "ControlKeyword",
292
+ "ModifierKeyword",
293
+ "Number",
294
+ "Percentage",
295
+ "Duration",
296
+ "DateLiteral"
297
+ ]);
298
+ var KEYWORD_STARTS = /* @__PURE__ */ new Set([
299
+ "TagKeyword",
300
+ "DirectiveKeyword",
301
+ "ControlKeyword",
302
+ "ModifierKeyword",
303
+ "SectionMarker",
304
+ "Comment",
305
+ "Duration",
306
+ "DateLiteral"
307
+ ]);
308
+ function highlightDgmo(source) {
309
+ const tree = parser.parse(source);
310
+ const tokens = [];
311
+ let pos = 0;
312
+ const cursor = tree.cursor();
313
+ function descend() {
314
+ for (; ; ) {
315
+ if (cursor.firstChild()) continue;
316
+ emitLeaf();
317
+ while (!cursor.nextSibling()) {
318
+ if (!cursor.parent()) return;
319
+ }
320
+ }
321
+ }
322
+ function emitLeaf() {
323
+ const from = cursor.from;
324
+ const to = cursor.to;
325
+ if (from > pos) {
326
+ tokens.push({ text: source.slice(pos, from), role: "default" });
327
+ }
328
+ if (to > from) {
329
+ const role = NODE_TO_ROLE[cursor.name] ?? "default";
330
+ tokens.push({ text: source.slice(from, to), role });
331
+ }
332
+ pos = to;
333
+ }
334
+ descend();
335
+ if (pos < source.length) {
336
+ tokens.push({ text: source.slice(pos, source.length), role: "default" });
337
+ }
338
+ applyLabelOverrides(tokens);
339
+ applyNoteContent(tokens);
340
+ return tokens;
341
+ }
342
+ function applyLabelOverrides(tokens) {
343
+ const ROLE_TO_NODES = {};
344
+ for (const [node, role] of Object.entries(NODE_TO_ROLE)) {
345
+ (ROLE_TO_NODES[role] ??= []).push(node);
346
+ }
347
+ const lines = [[]];
348
+ for (let i = 0; i < tokens.length; i++) {
349
+ const t = tokens[i];
350
+ const currentLine = lines[lines.length - 1];
351
+ const role = t.role;
352
+ let nodeName = "";
353
+ for (const [node, r] of Object.entries(NODE_TO_ROLE)) {
354
+ if (r === role) {
355
+ nodeName = node;
356
+ break;
357
+ }
358
+ }
359
+ if (role === "operator") {
360
+ const text = t.text;
361
+ if (text === "->" || text.endsWith("->")) nodeName = "SyncArrow";
362
+ else if (text === "~>" || text.endsWith("~>")) nodeName = "AsyncArrow";
363
+ else if (text === "-") nodeName = "Dash";
364
+ else if (text === "~") nodeName = "Tilde";
365
+ else if (text === "*") nodeName = "Star";
366
+ else if (text === "?") nodeName = "Question";
367
+ } else if (role === "number") {
368
+ const text = t.text;
369
+ if (/^\d+[smhd]$/i.test(text)) nodeName = "Duration";
370
+ else if (/^\d{4}-\d{2}-\d{2}/.test(text)) nodeName = "DateLiteral";
371
+ else if (text.endsWith("%")) nodeName = "Percentage";
372
+ else nodeName = "Number";
373
+ } else if (role === "bracket") {
374
+ const text = t.text;
375
+ if (text === "[") nodeName = "OpenBracket";
376
+ else if (text === "]") nodeName = "CloseBracket";
377
+ else if (text === "(") nodeName = "OpenParen";
378
+ else if (text === ")") nodeName = "CloseParen";
379
+ else if (text === "<") nodeName = "OpenAngle";
380
+ else if (text === ">") nodeName = "CloseAngle";
381
+ } else if (role === "separator") {
382
+ if (t.text === "|") nodeName = "Pipe";
383
+ else if (t.text === ":") nodeName = "Colon";
384
+ else if (t.text === "+") nodeName = "Plus";
385
+ } else if (role === "punctuation") {
386
+ if (t.text === ",") nodeName = "Comma";
387
+ else nodeName = "Punct";
388
+ }
389
+ currentLine.push({ idx: i, nodeName });
390
+ if (t.text.includes("\n")) {
391
+ lines.push([]);
392
+ }
393
+ }
394
+ let seenFirstContent = false;
395
+ for (const line of lines) {
396
+ const nonWs = line.filter(
397
+ (ref) => tokens[ref.idx].text.trim().length > 0
398
+ );
399
+ if (nonWs.length === 0) continue;
400
+ const firstTok = nonWs[0];
401
+ if (KEYWORD_STARTS.has(firstTok.nodeName)) continue;
402
+ if (firstTok.nodeName === "ChartType" && !seenFirstContent) {
403
+ seenFirstContent = true;
404
+ continue;
405
+ }
406
+ seenFirstContent = true;
407
+ let firstDashTildeIdx = -1;
408
+ let lastArrowIdx = -1;
409
+ for (let li = 0; li < nonWs.length; li++) {
410
+ const ref = nonWs[li];
411
+ if ((ref.nodeName === "Dash" || ref.nodeName === "Tilde") && firstDashTildeIdx < 0) {
412
+ firstDashTildeIdx = li;
413
+ }
414
+ if (ref.nodeName === "SyncArrow" || ref.nodeName === "AsyncArrow") {
415
+ lastArrowIdx = li;
416
+ }
417
+ }
418
+ const hasArrow = firstDashTildeIdx >= 0 && lastArrowIdx > firstDashTildeIdx;
419
+ if (!hasArrow) continue;
420
+ for (let li = firstDashTildeIdx + 1; li < lastArrowIdx; li++) {
421
+ const ref = nonWs[li];
422
+ if (OVERRIDE_IN_LABEL.has(ref.nodeName)) {
423
+ tokens[ref.idx].role = "default";
424
+ }
425
+ if (ref.nodeName === "ChartType") {
426
+ tokens[ref.idx].role = "default";
427
+ }
428
+ }
429
+ }
430
+ }
431
+ var NOTE_HEAD_RE = /^note(\s|$)/i;
432
+ function applyNoteContent(tokens) {
433
+ const fullText = tokens.map((t) => t.text).join("");
434
+ const lines = fullText.split("\n");
435
+ let inNote = false;
436
+ let noteIndent = 0;
437
+ let charOffset = 0;
438
+ for (const lineText of lines) {
439
+ const lineStart = charOffset;
440
+ const lineEnd = charOffset + lineText.length;
441
+ const trimmed = lineText.trimStart();
442
+ const indent = lineText.length - trimmed.length;
443
+ if (NOTE_HEAD_RE.test(trimmed)) {
444
+ inNote = true;
445
+ noteIndent = indent;
446
+ } else if (inNote) {
447
+ if (trimmed.length === 0) {
448
+ } else if (indent > noteIndent) {
449
+ markTokensInRange(tokens, lineStart, lineEnd, "noteContent");
450
+ } else {
451
+ inNote = false;
452
+ }
453
+ }
454
+ charOffset = lineEnd + 1;
455
+ }
456
+ }
457
+ function markTokensInRange(tokens, from, to, role) {
458
+ let pos = 0;
459
+ for (const token of tokens) {
460
+ const tokenEnd = pos + token.text.length;
461
+ if (tokenEnd > from && pos < to && token.text.trim().length > 0) {
462
+ token.role = role;
463
+ }
464
+ pos = tokenEnd;
465
+ }
466
+ }
467
+ var NORD_ROLE_STYLES = {
468
+ keyword: { color: "#81A1C1", fontWeight: "bold" },
469
+ // nord9
470
+ controlKeyword: { color: "#B48EAD", fontWeight: "bold" },
471
+ // nord15
472
+ definitionKeyword: { color: "#5E81AC", fontWeight: "bold" },
473
+ // nord10
474
+ modifier: { color: "#B48EAD" },
475
+ // nord15
476
+ chartType: { color: "#D08770", fontWeight: "bold" },
477
+ // nord12
478
+ operator: { color: "#BF616A", fontWeight: "bold" },
479
+ // nord11
480
+ number: { color: "#B48EAD" },
481
+ // nord15
482
+ comment: { color: "#616E88", fontStyle: "italic" },
483
+ heading: { color: "#D08770", fontWeight: "bold" },
484
+ // nord12
485
+ bracket: { color: "#5E81AC" },
486
+ // nord10
487
+ separator: { color: "#88C0D0" },
488
+ // nord8
489
+ url: { color: "#88C0D0", textDecoration: "underline" },
490
+ // nord8
491
+ colorAnnotation: { color: "#D08770", fontStyle: "italic" },
492
+ // nord12
493
+ punctuation: { color: "#616E88" },
494
+ noteContent: { color: "#616E88", fontStyle: "italic" },
495
+ default: {}
496
+ };
497
+ var ROLE_TO_ANSI = {
498
+ comment: "\x1B[3;90m",
499
+ // italic dim
500
+ keyword: "\x1B[1;34m",
501
+ // bold blue
502
+ controlKeyword: "\x1B[1;35m",
503
+ // bold magenta
504
+ definitionKeyword: "\x1B[1;34m",
505
+ // bold blue
506
+ modifier: "\x1B[35m",
507
+ // magenta
508
+ chartType: "\x1B[1;33m",
509
+ // bold yellow
510
+ operator: "\x1B[1;31m",
511
+ // bold red
512
+ number: "\x1B[35m",
513
+ // magenta
514
+ heading: "\x1B[1;33m",
515
+ // bold yellow
516
+ bracket: "\x1B[34m",
517
+ // blue
518
+ separator: "\x1B[36m",
519
+ // cyan
520
+ url: "\x1B[4;36m",
521
+ // underline cyan
522
+ colorAnnotation: "\x1B[3;33m",
523
+ // italic yellow
524
+ punctuation: "\x1B[90m",
525
+ // dim
526
+ noteContent: "\x1B[3;90m"
527
+ // italic dim
528
+ };
529
+ var ANSI_RESET = "\x1B[0m";
530
+ function renderAnsi(tokens, useColor) {
531
+ if (!useColor) {
532
+ return tokens.map((t) => t.text).join("");
533
+ }
534
+ let out = "";
535
+ let inStyled = false;
536
+ for (const token of tokens) {
537
+ const ansi = ROLE_TO_ANSI[token.role];
538
+ if (ansi) {
539
+ if (inStyled) out += ANSI_RESET;
540
+ out += ansi + token.text;
541
+ inStyled = true;
542
+ } else {
543
+ if (inStyled) {
544
+ out += ANSI_RESET;
545
+ inStyled = false;
546
+ }
547
+ out += token.text;
548
+ }
549
+ }
550
+ out += ANSI_RESET;
551
+ return out;
552
+ }
553
+ // Annotate the CommonJS export names for ESM import in node:
554
+ 0 && (module.exports = {
555
+ NORD_ROLE_STYLES,
556
+ ROLE_TO_ANSI,
557
+ highlightDgmo,
558
+ renderAnsi
559
+ });
560
+ //# sourceMappingURL=highlight.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/editor/highlight-api.ts","../src/editor/dgmo.grammar.js","../src/editor/dgmo.grammar.terms.js","../src/editor/keywords.ts","../src/editor/tokens.ts"],"sourcesContent":["/**\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","// 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"],"mappings":"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","names":[]}
@@ -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 };