@cqa-lib/cqa-ui 1.1.493 → 1.1.494

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.
@@ -1,8 +1,12 @@
1
1
  import { ChangeDetectionStrategy, Component, EventEmitter, HostListener, Input, Output, ViewChild } from '@angular/core';
2
2
  import * as i0 from "@angular/core";
3
3
  import * as i1 from "@angular/common";
4
- const VAR_NAME_REGEX = /^[A-Za-z_][A-Za-z0-9_]*$/;
5
- const VAR_TOKEN_REGEX = /\$\{([A-Za-z_][A-Za-z0-9_]*)\}/g;
4
+ const VAR_PATH = '[A-Za-z_][A-Za-z0-9_]*(?:\\.[A-Za-z_][A-Za-z0-9_]*|\\[\\d+\\])*';
5
+ const VAR_NAME_REGEX = new RegExp('^' + VAR_PATH + '$');
6
+ const VAR_TOKEN_REGEX = new RegExp('\\$\\{(' + VAR_PATH + ')\\}', 'g');
7
+ const VAR_PATH_TAIL_REGEX = new RegExp('(' + VAR_PATH + ')$');
8
+ const INVALID_VAR_MESSAGE = 'Invalid variable name.';
9
+ const BLUR_HIDE_DELAY_MS = 150;
6
10
  export class MixedVariableInputComponent {
7
11
  constructor(cdr, hostRef) {
8
12
  this.cdr = cdr;
@@ -11,41 +15,56 @@ export class MixedVariableInputComponent {
11
15
  this.placeholder = 'Text Input';
12
16
  this.disabled = false;
13
17
  this.valueChange = new EventEmitter();
18
+ this.validityChange = new EventEmitter();
14
19
  this.showSuggestion = false;
15
20
  this.pendingWord = '';
21
+ this.canAddAsVariable = false;
22
+ this.errorMessage = null;
23
+ this.selectionRange = null;
24
+ this.selectionMode = false;
16
25
  this.lastEmitted = '';
26
+ this.lastValid = null;
17
27
  }
18
- onDocumentMouseDown(event) {
19
- if (!this.showSuggestion)
28
+ // ---- Lifecycle ------------------------------------------------------------
29
+ ngOnChanges(changes) {
30
+ if (!changes['value'] || !this.editorRef)
20
31
  return;
21
- const target = event.target;
22
- if (this.hostRef.nativeElement.contains(target))
32
+ const incoming = changes['value'].currentValue ?? '';
33
+ if (incoming === this.lastEmitted)
23
34
  return;
24
- this.hideSuggestion();
25
- }
26
- ngOnChanges(changes) {
27
- if (changes['value'] && this.editorRef) {
28
- const incoming = changes['value'].currentValue ?? '';
29
- if (incoming !== this.lastEmitted) {
30
- this.render(incoming);
31
- }
32
- }
35
+ this.render(incoming);
36
+ this.validate(incoming);
33
37
  }
34
38
  ngAfterViewInit() {
35
- this.render(this.value || '');
39
+ const initial = this.value || '';
40
+ this.render(initial);
41
+ this.validate(initial);
36
42
  }
43
+ // ---- Editor event handlers -----------------------------------------------
37
44
  onInput() {
38
- const str = this.serialize();
39
- this.lastEmitted = str;
40
- this.valueChange.emit(str);
45
+ this.flattenStructure();
46
+ this.sanitizeChips();
41
47
  this.autoChipify();
48
+ this.emitValue();
49
+ this.refreshSuggestion();
50
+ }
51
+ onEditorFocus() {
42
52
  this.refreshSuggestion();
43
53
  }
54
+ onEditorBlur() {
55
+ // Delay so clicks on suggestion buttons register before we hide them.
56
+ setTimeout(() => {
57
+ this.autoChipify(true);
58
+ this.hideSuggestion();
59
+ }, BLUR_HIDE_DELAY_MS);
60
+ }
44
61
  onKeyDown(event) {
45
62
  if (event.key === 'Enter') {
46
63
  event.preventDefault();
47
64
  return;
48
65
  }
66
+ if (event.key === 'Tab')
67
+ return;
49
68
  if (event.key === 'Escape') {
50
69
  if (this.showSuggestion) {
51
70
  event.preventDefault();
@@ -62,36 +81,119 @@ export class MixedVariableInputComponent {
62
81
  }
63
82
  }
64
83
  }
65
- onEditorBlur() {
66
- // Small timeout so clicks on suggestion buttons register before hiding.
67
- setTimeout(() => this.hideSuggestion(), 150);
84
+ onPaste(event) {
85
+ event.preventDefault();
86
+ const text = event.clipboardData?.getData('text/plain') ?? '';
87
+ if (!text)
88
+ return;
89
+ this.insertPlainTextAtCaret(text);
90
+ this.onInput();
91
+ }
92
+ onDrop(event) {
93
+ event.preventDefault();
94
+ const text = event.dataTransfer?.getData('text/plain') ?? '';
95
+ if (text) {
96
+ this.editorRef.nativeElement.focus();
97
+ this.insertPlainTextAtCaret(text);
98
+ }
99
+ this.onInput();
100
+ }
101
+ onDragStart(event) {
102
+ event.preventDefault();
103
+ }
104
+ onDocumentMouseDown(event) {
105
+ if (!this.showSuggestion)
106
+ return;
107
+ if (this.hostRef.nativeElement.contains(event.target))
108
+ return;
109
+ this.hideSuggestion();
110
+ }
111
+ onDocumentSelectionChange() {
112
+ if (document.activeElement !== this.editorRef?.nativeElement)
113
+ return;
114
+ this.refreshSuggestion();
68
115
  }
116
+ // ---- Suggestion actions ---------------------------------------------------
69
117
  addAsText() {
70
118
  this.hideSuggestion();
71
119
  this.editorRef.nativeElement.focus();
72
120
  }
73
121
  addAsVariable() {
74
- const name = (this.pendingWord || '').trim();
122
+ const name = this.normalizeCandidate(this.pendingWord || '');
75
123
  if (!name || !VAR_NAME_REGEX.test(name)) {
76
124
  this.hideSuggestion();
77
125
  return;
78
126
  }
79
- this.insertChipReplacingCurrentWord(name);
127
+ if (this.selectionMode && this.selectionRange) {
128
+ if (this.rangeCrossesChip(this.selectionRange)) {
129
+ this.hideSuggestion();
130
+ return;
131
+ }
132
+ this.insertChipReplacingRange(name, this.selectionRange);
133
+ }
134
+ else {
135
+ this.insertChipReplacingCurrentWord(name);
136
+ }
80
137
  this.hideSuggestion();
138
+ this.emitValue();
139
+ }
140
+ // ---- Value emission / validation -----------------------------------------
141
+ emitValue() {
81
142
  const str = this.serialize();
82
- this.lastEmitted = str;
83
- this.valueChange.emit(str);
143
+ if (str !== this.lastEmitted) {
144
+ this.lastEmitted = str;
145
+ this.valueChange.emit(str);
146
+ }
147
+ this.validate(str);
148
+ }
149
+ validate(str) {
150
+ const error = this.findValidationError(str);
151
+ const valid = error === null;
152
+ if (this.lastValid === valid && this.errorMessage === error)
153
+ return;
154
+ this.lastValid = valid;
155
+ this.errorMessage = error;
156
+ this.validityChange.emit({ valid, error });
157
+ this.cdr.markForCheck();
158
+ }
159
+ findValidationError(str) {
160
+ if (!str)
161
+ return null;
162
+ const braceBody = /\$\{([^}]*)\}/g;
163
+ let m;
164
+ braceBody.lastIndex = 0;
165
+ while ((m = braceBody.exec(str)) !== null) {
166
+ const body = m[1];
167
+ if (body !== body.trim() || !VAR_NAME_REGEX.test(body)) {
168
+ return INVALID_VAR_MESSAGE;
169
+ }
170
+ }
171
+ const openCount = (str.match(/\$\{/g) || []).length;
172
+ const closedCount = (str.match(braceBody) || []).length;
173
+ if (openCount > closedCount)
174
+ return INVALID_VAR_MESSAGE;
175
+ return null;
84
176
  }
177
+ // ---- Suggestion state -----------------------------------------------------
85
178
  refreshSuggestion() {
86
- const word = this.getCurrentWord();
87
- if (word && VAR_NAME_REGEX.test(word)) {
88
- this.pendingWord = word;
179
+ const selInfo = this.getSelectedInfo();
180
+ if (selInfo) {
181
+ const normalized = this.normalizeCandidate(selInfo.text);
182
+ this.pendingWord = normalized;
183
+ this.selectionRange = selInfo.range;
184
+ this.selectionMode = true;
89
185
  this.showSuggestion = true;
186
+ this.canAddAsVariable =
187
+ !!normalized && VAR_NAME_REGEX.test(normalized) && !this.rangeCrossesChip(selInfo.range);
188
+ this.cdr.markForCheck();
189
+ return;
90
190
  }
91
- else {
92
- this.pendingWord = '';
93
- this.showSuggestion = false;
94
- }
191
+ this.selectionMode = false;
192
+ this.selectionRange = null;
193
+ const word = this.getCurrentWord();
194
+ this.pendingWord = word;
195
+ this.showSuggestion = !!word;
196
+ this.canAddAsVariable = !!word && VAR_NAME_REGEX.test(word);
95
197
  this.cdr.markForCheck();
96
198
  }
97
199
  hideSuggestion() {
@@ -99,45 +201,126 @@ export class MixedVariableInputComponent {
99
201
  return;
100
202
  this.showSuggestion = false;
101
203
  this.pendingWord = '';
204
+ this.canAddAsVariable = false;
102
205
  this.cdr.markForCheck();
103
206
  }
104
- autoChipify() {
207
+ normalizeCandidate(text) {
208
+ let t = text.trim();
209
+ if (t.startsWith('${') && t.endsWith('}')) {
210
+ t = t.slice(2, -1).trim();
211
+ }
212
+ return t;
213
+ }
214
+ // ---- DOM sanitization -----------------------------------------------------
215
+ flattenStructure() {
105
216
  const editor = this.editorRef.nativeElement;
106
- const textNodes = [];
107
- editor.childNodes.forEach((n) => {
108
- if (n.nodeType === Node.TEXT_NODE)
109
- textNodes.push(n);
110
- });
111
- for (const tn of textNodes) {
217
+ const wrappers = Array.from(editor.querySelectorAll('div, p'));
218
+ const strayInlines = Array.from(editor.querySelectorAll('span')).filter((s) => !s.classList.contains('cqa-var-chip') && !s.closest('.cqa-var-chip'));
219
+ for (const el of [...wrappers, ...strayInlines]) {
220
+ if (el.closest('.cqa-var-chip'))
221
+ continue;
222
+ while (el.firstChild)
223
+ el.parentNode?.insertBefore(el.firstChild, el);
224
+ el.remove();
225
+ }
226
+ const brs = Array.from(editor.querySelectorAll('br'));
227
+ for (const br of brs) {
228
+ if (br.closest('.cqa-var-chip'))
229
+ continue;
230
+ br.parentNode?.replaceChild(document.createTextNode(' '), br);
231
+ }
232
+ }
233
+ sanitizeChips() {
234
+ const editor = this.editorRef.nativeElement;
235
+ const chips = Array.from(editor.querySelectorAll('.cqa-var-chip'));
236
+ for (const chip of chips) {
237
+ const name = (chip.getAttribute('data-var') || '').trim();
238
+ if (!name || !VAR_NAME_REGEX.test(name)) {
239
+ chip.parentNode?.replaceChild(document.createTextNode(chip.textContent || ''), chip);
240
+ continue;
241
+ }
242
+ const structureBroken = chip.getAttribute('contenteditable') !== 'false' ||
243
+ !chip.querySelector('.cqa-var-chip__remove');
244
+ if (structureBroken) {
245
+ chip.parentNode?.replaceChild(this.buildChip(name), chip);
246
+ }
247
+ }
248
+ // Remove stray remove icons left from partial undo/redo states.
249
+ const orphans = Array.from(editor.querySelectorAll('.cqa-var-chip__remove'));
250
+ for (const o of orphans) {
251
+ if (!o.closest('.cqa-var-chip'))
252
+ o.remove();
253
+ }
254
+ }
255
+ // ---- Chipification --------------------------------------------------------
256
+ autoChipify(force = false) {
257
+ const editor = this.editorRef.nativeElement;
258
+ const caretRange = this.getSelectionRange();
259
+ let changed = false;
260
+ let lastChip = null;
261
+ for (const tn of this.collectDescendantTextNodes(editor)) {
262
+ if (!tn.parentNode)
263
+ continue;
112
264
  const text = tn.nodeValue || '';
113
265
  VAR_TOKEN_REGEX.lastIndex = 0;
114
266
  if (!VAR_TOKEN_REGEX.test(text))
115
267
  continue;
116
268
  VAR_TOKEN_REGEX.lastIndex = 0;
117
- const parent = tn.parentNode;
118
- if (!parent)
119
- continue;
120
- const frag = document.createDocumentFragment();
121
- let lastIndex = 0;
269
+ const caretOffset = !force && caretRange && caretRange.collapsed && caretRange.startContainer === tn
270
+ ? caretRange.startOffset
271
+ : -1;
272
+ const matches = [];
122
273
  let m;
123
274
  while ((m = VAR_TOKEN_REGEX.exec(text)) !== null) {
124
- if (m.index > lastIndex) {
125
- frag.appendChild(document.createTextNode(text.slice(lastIndex, m.index)));
275
+ const start = m.index;
276
+ const end = start + m[0].length;
277
+ // Keep the token as plain text while the caret is strictly inside.
278
+ if (caretOffset > start && caretOffset < end)
279
+ continue;
280
+ matches.push({ start, end, name: m[1] });
281
+ }
282
+ if (!matches.length)
283
+ continue;
284
+ const frag = document.createDocumentFragment();
285
+ let cursor = 0;
286
+ for (const mt of matches) {
287
+ if (mt.start > cursor) {
288
+ frag.appendChild(document.createTextNode(text.slice(cursor, mt.start)));
126
289
  }
127
- frag.appendChild(this.buildChip(m[1]));
128
- lastIndex = m.index + m[0].length;
290
+ const chip = this.buildChip(mt.name);
291
+ frag.appendChild(chip);
292
+ lastChip = chip;
293
+ cursor = mt.end;
129
294
  }
130
- if (lastIndex < text.length) {
131
- frag.appendChild(document.createTextNode(text.slice(lastIndex)));
295
+ if (cursor < text.length) {
296
+ frag.appendChild(document.createTextNode(text.slice(cursor)));
132
297
  }
133
- parent.replaceChild(frag, tn);
134
- this.placeCaretAtEnd();
135
- const str = this.serialize();
136
- this.lastEmitted = str;
137
- this.valueChange.emit(str);
138
- return;
298
+ tn.parentNode.replaceChild(frag, tn);
299
+ changed = true;
139
300
  }
301
+ if (changed && lastChip)
302
+ this.placeCaretAfter(lastChip);
303
+ }
304
+ collectDescendantTextNodes(root) {
305
+ const out = [];
306
+ const walker = document.createTreeWalker(root, NodeFilter.SHOW_TEXT, {
307
+ acceptNode: (node) => {
308
+ let p = node.parentNode;
309
+ while (p && p !== root) {
310
+ if (p.classList?.contains?.('cqa-var-chip')) {
311
+ return NodeFilter.FILTER_REJECT;
312
+ }
313
+ p = p.parentNode;
314
+ }
315
+ return NodeFilter.FILTER_ACCEPT;
316
+ },
317
+ });
318
+ let n;
319
+ while ((n = walker.nextNode()))
320
+ out.push(n);
321
+ return out;
140
322
  }
323
+ // ---- Render / serialize ---------------------------------------------------
141
324
  render(value) {
142
325
  const editor = this.editorRef.nativeElement;
143
326
  editor.innerHTML = '';
@@ -160,22 +343,30 @@ export class MixedVariableInputComponent {
160
343
  serialize() {
161
344
  const editor = this.editorRef.nativeElement;
162
345
  let out = '';
163
- editor.childNodes.forEach((n) => {
164
- if (n.nodeType === Node.TEXT_NODE) {
165
- out += (n.nodeValue || '').replace(/\u00a0/g, ' ');
346
+ const walk = (node) => {
347
+ if (node.nodeType === Node.TEXT_NODE) {
348
+ out += (node.nodeValue || '').replace(/\u00a0/g, ' ');
349
+ return;
166
350
  }
167
- else if (n.nodeType === Node.ELEMENT_NODE) {
168
- const el = n;
169
- if (el.classList.contains('cqa-var-chip')) {
170
- const name = el.getAttribute('data-var') || '';
171
- if (name)
172
- out += '${' + name + '}';
173
- }
174
- else {
175
- out += el.textContent || '';
176
- }
351
+ if (node.nodeType !== Node.ELEMENT_NODE)
352
+ return;
353
+ const el = node;
354
+ if (el.classList.contains('cqa-var-chip')) {
355
+ const name = el.getAttribute('data-var') || '';
356
+ if (name)
357
+ out += '${' + name + '}';
358
+ return;
177
359
  }
178
- });
360
+ if (el.tagName === 'BR') {
361
+ out += '\n';
362
+ return;
363
+ }
364
+ const isBlock = el.tagName === 'DIV' || el.tagName === 'P';
365
+ if (isBlock && out.length > 0 && !out.endsWith('\n'))
366
+ out += '\n';
367
+ el.childNodes.forEach(walk);
368
+ };
369
+ editor.childNodes.forEach(walk);
179
370
  return out;
180
371
  }
181
372
  buildChip(name) {
@@ -200,6 +391,7 @@ export class MixedVariableInputComponent {
200
391
  chip.appendChild(remove);
201
392
  return chip;
202
393
  }
394
+ // ---- Selection / caret helpers -------------------------------------------
203
395
  getSelectionRange() {
204
396
  const sel = window.getSelection();
205
397
  if (!sel || sel.rangeCount === 0)
@@ -209,6 +401,25 @@ export class MixedVariableInputComponent {
209
401
  return null;
210
402
  return range;
211
403
  }
404
+ getSelectedInfo() {
405
+ const sel = window.getSelection();
406
+ if (!sel || sel.rangeCount === 0)
407
+ return null;
408
+ const range = sel.getRangeAt(0);
409
+ if (range.collapsed)
410
+ return null;
411
+ const editor = this.editorRef.nativeElement;
412
+ if (!editor.contains(range.startContainer) || !editor.contains(range.endContainer))
413
+ return null;
414
+ const text = range.toString();
415
+ if (!text)
416
+ return null;
417
+ return { text, range: range.cloneRange() };
418
+ }
419
+ rangeCrossesChip(range) {
420
+ const contents = range.cloneContents();
421
+ return !!contents.querySelector?.('.cqa-var-chip');
422
+ }
212
423
  getNodeBeforeCaret() {
213
424
  const range = this.getSelectionRange();
214
425
  if (!range || !range.collapsed)
@@ -223,16 +434,39 @@ export class MixedVariableInputComponent {
223
434
  }
224
435
  getCurrentWord() {
225
436
  const range = this.getSelectionRange();
226
- if (!range)
227
- return '';
228
- const node = range.startContainer;
229
- if (node.nodeType !== Node.TEXT_NODE)
437
+ if (!range || !range.collapsed)
230
438
  return '';
439
+ let node = range.startContainer;
440
+ let offset = range.startOffset;
441
+ if (node.nodeType !== Node.TEXT_NODE) {
442
+ const prev = node.childNodes[offset - 1];
443
+ if (!prev || prev.nodeType !== Node.TEXT_NODE)
444
+ return '';
445
+ node = prev;
446
+ offset = (prev.nodeValue || '').length;
447
+ }
231
448
  const text = node.nodeValue || '';
232
- const before = text.slice(0, range.startOffset);
233
- const match = before.match(/([A-Za-z_][A-Za-z0-9_]*)$/);
449
+ const braceOpen = text.lastIndexOf('${', offset - 1);
450
+ if (braceOpen !== -1) {
451
+ const closeIdx = text.indexOf('}', braceOpen + 2);
452
+ const tokenClosedBeforeCaret = closeIdx !== -1 && closeIdx < offset;
453
+ if (!tokenClosedBeforeCaret) {
454
+ const start = braceOpen + 2;
455
+ const end = closeIdx === -1 ? text.length : closeIdx;
456
+ return text.slice(start, end);
457
+ }
458
+ }
459
+ const match = text.slice(0, offset).match(VAR_PATH_TAIL_REGEX);
234
460
  return match ? match[1] : '';
235
461
  }
462
+ // ---- Chip insertion -------------------------------------------------------
463
+ insertChipReplacingRange(name, range) {
464
+ const chip = this.buildChip(name);
465
+ range.deleteContents();
466
+ range.insertNode(chip);
467
+ this.stripBracesAround(chip);
468
+ this.placeCaretAfter(chip);
469
+ }
236
470
  insertChipReplacingCurrentWord(name) {
237
471
  const range = this.getSelectionRange();
238
472
  const chip = this.buildChip(name);
@@ -242,26 +476,59 @@ export class MixedVariableInputComponent {
242
476
  return;
243
477
  }
244
478
  const node = range.startContainer;
245
- if (node.nodeType === Node.TEXT_NODE) {
246
- const text = node.nodeValue || '';
247
- const before = text.slice(0, range.startOffset);
248
- const after = text.slice(range.startOffset);
249
- const wordMatch = before.match(/([A-Za-z_][A-Za-z0-9_]*)$/);
250
- const beforeKept = wordMatch ? before.slice(0, before.length - wordMatch[1].length) : before;
251
- const parent = node.parentNode;
252
- const frag = document.createDocumentFragment();
253
- if (beforeKept)
254
- frag.appendChild(document.createTextNode(beforeKept));
255
- frag.appendChild(chip);
256
- if (after)
257
- frag.appendChild(document.createTextNode(after));
258
- parent.replaceChild(frag, node);
259
- }
260
- else {
479
+ if (node.nodeType !== Node.TEXT_NODE) {
261
480
  range.insertNode(chip);
481
+ this.placeCaretAfter(chip);
482
+ return;
262
483
  }
484
+ const text = node.nodeValue || '';
485
+ const before = text.slice(0, range.startOffset);
486
+ const after = text.slice(range.startOffset);
487
+ const wordMatch = before.match(VAR_PATH_TAIL_REGEX);
488
+ let beforeKept = wordMatch ? before.slice(0, before.length - wordMatch[1].length) : before;
489
+ let afterKept = after;
490
+ if (beforeKept.endsWith('${') && afterKept.startsWith('}')) {
491
+ beforeKept = beforeKept.slice(0, -2);
492
+ afterKept = afterKept.slice(1);
493
+ }
494
+ else if (beforeKept.endsWith('${')) {
495
+ beforeKept = beforeKept.slice(0, -2);
496
+ }
497
+ const frag = document.createDocumentFragment();
498
+ if (beforeKept)
499
+ frag.appendChild(document.createTextNode(beforeKept));
500
+ frag.appendChild(chip);
501
+ if (afterKept)
502
+ frag.appendChild(document.createTextNode(afterKept));
503
+ node.parentNode.replaceChild(frag, node);
263
504
  this.placeCaretAfter(chip);
264
505
  }
506
+ stripBracesAround(chip) {
507
+ const prev = chip.previousSibling;
508
+ const next = chip.nextSibling;
509
+ if (prev && prev.nodeType === Node.TEXT_NODE) {
510
+ const t = prev.nodeValue || '';
511
+ if (t.endsWith('${'))
512
+ prev.nodeValue = t.slice(0, -2);
513
+ }
514
+ if (next && next.nodeType === Node.TEXT_NODE) {
515
+ const t = next.nodeValue || '';
516
+ if (t.startsWith('}'))
517
+ next.nodeValue = t.slice(1);
518
+ }
519
+ }
520
+ insertPlainTextAtCaret(text) {
521
+ const range = this.getSelectionRange();
522
+ if (!range) {
523
+ this.editorRef.nativeElement.appendChild(document.createTextNode(text));
524
+ this.placeCaretAtEnd();
525
+ return;
526
+ }
527
+ range.deleteContents();
528
+ const node = document.createTextNode(text);
529
+ range.insertNode(node);
530
+ this.placeCaretAfter(node);
531
+ }
265
532
  placeCaretAfter(node) {
266
533
  const sel = window.getSelection();
267
534
  if (!sel)
@@ -273,22 +540,21 @@ export class MixedVariableInputComponent {
273
540
  sel.addRange(range);
274
541
  }
275
542
  placeCaretAtEnd() {
276
- const editor = this.editorRef.nativeElement;
277
543
  const sel = window.getSelection();
278
544
  if (!sel)
279
545
  return;
280
546
  const range = document.createRange();
281
- range.selectNodeContents(editor);
547
+ range.selectNodeContents(this.editorRef.nativeElement);
282
548
  range.collapse(false);
283
549
  sel.removeAllRanges();
284
550
  sel.addRange(range);
285
551
  }
286
552
  }
287
553
  MixedVariableInputComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: MixedVariableInputComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
288
- MixedVariableInputComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: MixedVariableInputComponent, selector: "cqa-mixed-variable-input", inputs: { value: "value", placeholder: "placeholder", disabled: "disabled" }, outputs: { valueChange: "valueChange" }, host: { listeners: { "document:mousedown": "onDocumentMouseDown($event)" }, classAttribute: "cqa-ui-root" }, viewQueries: [{ propertyName: "editorRef", first: true, predicate: ["editor"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-mixed-var-wrapper\">\n <div\n #editor\n class=\"cqa-mixed-var-editor\"\n role=\"textbox\"\n spellcheck=\"false\"\n [attr.contenteditable]=\"!disabled\"\n [attr.data-placeholder]=\"placeholder\"\n (input)=\"onInput()\"\n (keydown)=\"onKeyDown($event)\"\n (blur)=\"onEditorBlur()\">\n </div>\n\n <div\n *ngIf=\"showSuggestion\"\n class=\"cqa-mixed-var-suggest\"\n (mousedown)=\"$event.preventDefault()\">\n <button type=\"button\" class=\"cqa-mixed-var-suggest__item\" (click)=\"addAsText()\">\n Add as Text <span class=\"cqa-mixed-var-suggest__hint\">\"{{ pendingWord }}\"</span>\n </button>\n <button type=\"button\" class=\"cqa-mixed-var-suggest__item cqa-mixed-var-suggest__item--primary\" (click)=\"addAsVariable()\">\n Add as local variable <span class=\"cqa-mixed-var-suggest__hint\">${{ '{' }}{{ pendingWord }}{{ '}' }}</span>\n </button>\n </div>\n</div>\n", directives: [{ type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
554
+ MixedVariableInputComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: MixedVariableInputComponent, selector: "cqa-mixed-variable-input", inputs: { value: "value", placeholder: "placeholder", disabled: "disabled" }, outputs: { valueChange: "valueChange", validityChange: "validityChange" }, host: { listeners: { "dragstart": "onDragStart($event)", "document:mousedown": "onDocumentMouseDown($event)", "document:selectionchange": "onDocumentSelectionChange()" }, classAttribute: "cqa-ui-root" }, viewQueries: [{ propertyName: "editorRef", first: true, predicate: ["editor"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-mixed-var-wrapper\">\n <div\n #editor\n class=\"cqa-mixed-var-editor\"\n role=\"textbox\"\n spellcheck=\"false\"\n [attr.contenteditable]=\"!disabled\"\n [attr.data-placeholder]=\"placeholder\"\n [attr.aria-invalid]=\"errorMessage ? 'true' : null\"\n [class.cqa-mixed-var-editor--invalid]=\"errorMessage\"\n (input)=\"onInput()\"\n (keydown)=\"onKeyDown($event)\"\n (focus)=\"onEditorFocus()\"\n (paste)=\"onPaste($event)\"\n (drop)=\"onDrop($event)\"\n (blur)=\"onEditorBlur()\">\n </div>\n <p *ngIf=\"errorMessage\" class=\"cqa-mixed-var-error\" role=\"alert\">{{ errorMessage }}</p>\n\n <div\n *ngIf=\"showSuggestion\"\n class=\"cqa-mixed-var-suggest\"\n (mousedown)=\"$event.preventDefault()\">\n <button type=\"button\" class=\"cqa-mixed-var-suggest__item\" (click)=\"addAsText()\">\n Add as Text <span *ngIf=\"pendingWord\" class=\"cqa-mixed-var-suggest__hint\">\"{{ pendingWord }}\"</span>\n </button>\n <button *ngIf=\"canAddAsVariable\" type=\"button\"\n class=\"cqa-mixed-var-suggest__item cqa-mixed-var-suggest__item--primary\"\n (click)=\"addAsVariable()\">\n Add as local variable <span class=\"cqa-mixed-var-suggest__hint\">${{ '{' }}{{ pendingWord }}{{ '}' }}</span>\n </button>\n </div>\n</div>\n", directives: [{ type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
289
555
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: MixedVariableInputComponent, decorators: [{
290
556
  type: Component,
291
- args: [{ selector: 'cqa-mixed-variable-input', changeDetection: ChangeDetectionStrategy.OnPush, host: { class: 'cqa-ui-root' }, template: "<div class=\"cqa-mixed-var-wrapper\">\n <div\n #editor\n class=\"cqa-mixed-var-editor\"\n role=\"textbox\"\n spellcheck=\"false\"\n [attr.contenteditable]=\"!disabled\"\n [attr.data-placeholder]=\"placeholder\"\n (input)=\"onInput()\"\n (keydown)=\"onKeyDown($event)\"\n (blur)=\"onEditorBlur()\">\n </div>\n\n <div\n *ngIf=\"showSuggestion\"\n class=\"cqa-mixed-var-suggest\"\n (mousedown)=\"$event.preventDefault()\">\n <button type=\"button\" class=\"cqa-mixed-var-suggest__item\" (click)=\"addAsText()\">\n Add as Text <span class=\"cqa-mixed-var-suggest__hint\">\"{{ pendingWord }}\"</span>\n </button>\n <button type=\"button\" class=\"cqa-mixed-var-suggest__item cqa-mixed-var-suggest__item--primary\" (click)=\"addAsVariable()\">\n Add as local variable <span class=\"cqa-mixed-var-suggest__hint\">${{ '{' }}{{ pendingWord }}{{ '}' }}</span>\n </button>\n </div>\n</div>\n", styles: [] }]
557
+ args: [{ selector: 'cqa-mixed-variable-input', changeDetection: ChangeDetectionStrategy.OnPush, host: { class: 'cqa-ui-root' }, template: "<div class=\"cqa-mixed-var-wrapper\">\n <div\n #editor\n class=\"cqa-mixed-var-editor\"\n role=\"textbox\"\n spellcheck=\"false\"\n [attr.contenteditable]=\"!disabled\"\n [attr.data-placeholder]=\"placeholder\"\n [attr.aria-invalid]=\"errorMessage ? 'true' : null\"\n [class.cqa-mixed-var-editor--invalid]=\"errorMessage\"\n (input)=\"onInput()\"\n (keydown)=\"onKeyDown($event)\"\n (focus)=\"onEditorFocus()\"\n (paste)=\"onPaste($event)\"\n (drop)=\"onDrop($event)\"\n (blur)=\"onEditorBlur()\">\n </div>\n <p *ngIf=\"errorMessage\" class=\"cqa-mixed-var-error\" role=\"alert\">{{ errorMessage }}</p>\n\n <div\n *ngIf=\"showSuggestion\"\n class=\"cqa-mixed-var-suggest\"\n (mousedown)=\"$event.preventDefault()\">\n <button type=\"button\" class=\"cqa-mixed-var-suggest__item\" (click)=\"addAsText()\">\n Add as Text <span *ngIf=\"pendingWord\" class=\"cqa-mixed-var-suggest__hint\">\"{{ pendingWord }}\"</span>\n </button>\n <button *ngIf=\"canAddAsVariable\" type=\"button\"\n class=\"cqa-mixed-var-suggest__item cqa-mixed-var-suggest__item--primary\"\n (click)=\"addAsVariable()\">\n Add as local variable <span class=\"cqa-mixed-var-suggest__hint\">${{ '{' }}{{ pendingWord }}{{ '}' }}</span>\n </button>\n </div>\n</div>\n", styles: [] }]
292
558
  }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i0.ElementRef }]; }, propDecorators: { value: [{
293
559
  type: Input
294
560
  }], placeholder: [{
@@ -297,11 +563,19 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
297
563
  type: Input
298
564
  }], valueChange: [{
299
565
  type: Output
566
+ }], validityChange: [{
567
+ type: Output
300
568
  }], editorRef: [{
301
569
  type: ViewChild,
302
570
  args: ['editor', { static: true }]
571
+ }], onDragStart: [{
572
+ type: HostListener,
573
+ args: ['dragstart', ['$event']]
303
574
  }], onDocumentMouseDown: [{
304
575
  type: HostListener,
305
576
  args: ['document:mousedown', ['$event']]
577
+ }], onDocumentSelectionChange: [{
578
+ type: HostListener,
579
+ args: ['document:selectionchange']
306
580
  }] } });
307
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mixed-variable-input.component.js","sourceRoot":"","sources":["../../../../../src/lib/mixed-variable-input/mixed-variable-input.component.ts","../../../../../src/lib/mixed-variable-input/mixed-variable-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, EventEmitter, HostListener, Input, OnChanges, Output, SimpleChanges, ViewChild } from '@angular/core';\n\nexport interface MixedSegment {\n  type: 'text' | 'variable';\n  value: string;\n}\n\nconst VAR_NAME_REGEX = /^[A-Za-z_][A-Za-z0-9_]*$/;\nconst VAR_TOKEN_REGEX = /\\$\\{([A-Za-z_][A-Za-z0-9_]*)\\}/g;\n\n@Component({\n  selector: 'cqa-mixed-variable-input',\n  templateUrl: './mixed-variable-input.component.html',\n  styleUrls: [],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root' },\n})\nexport class MixedVariableInputComponent implements OnChanges, AfterViewInit {\n  @Input() value = '';\n  @Input() placeholder = 'Text Input';\n  @Input() disabled = false;\n\n  @Output() valueChange = new EventEmitter<string>();\n\n  @ViewChild('editor', { static: true }) editorRef!: ElementRef<HTMLDivElement>;\n\n  showSuggestion = false;\n  pendingWord = '';\n\n  private lastEmitted = '';\n\n  constructor(\n    private cdr: ChangeDetectorRef, \n    private hostRef: ElementRef<HTMLElement>\n  ) {}\n\n  @HostListener('document:mousedown', ['$event'])\n  onDocumentMouseDown(event: MouseEvent): void {\n    if (!this.showSuggestion) return;\n    const target = event.target as Node;\n    if (this.hostRef.nativeElement.contains(target)) return;\n    this.hideSuggestion();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['value'] && this.editorRef) {\n      const incoming = changes['value'].currentValue ?? '';\n      if (incoming !== this.lastEmitted) {\n        this.render(incoming);\n      }\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.render(this.value || '');\n  }\n\n  onInput(): void {\n    const str = this.serialize();\n    this.lastEmitted = str;\n    this.valueChange.emit(str);\n    this.autoChipify();\n    this.refreshSuggestion();\n  }\n\n  onKeyDown(event: KeyboardEvent): void {\n    if (event.key === 'Enter') {\n      event.preventDefault();\n      return;\n    }\n    if (event.key === 'Escape') {\n      if (this.showSuggestion) {\n        event.preventDefault();\n        this.hideSuggestion();\n      }\n      return;\n    }\n    if (event.key === 'Backspace') {\n      const node = this.getNodeBeforeCaret();\n      if (node && (node as HTMLElement).classList?.contains('cqa-var-chip')) {\n        event.preventDefault();\n        (node as HTMLElement).remove();\n        this.onInput();\n      }\n    }\n  }\n\n  onEditorBlur(): void {\n    // Small timeout so clicks on suggestion buttons register before hiding.\n    setTimeout(() => this.hideSuggestion(), 150);\n  }\n\n  addAsText(): void {\n    this.hideSuggestion();\n    this.editorRef.nativeElement.focus();\n  }\n\n  addAsVariable(): void {\n    const name = (this.pendingWord || '').trim();\n    if (!name || !VAR_NAME_REGEX.test(name)) {\n      this.hideSuggestion();\n      return;\n    }\n    this.insertChipReplacingCurrentWord(name);\n    this.hideSuggestion();\n    const str = this.serialize();\n    this.lastEmitted = str;\n    this.valueChange.emit(str);\n  }\n\n  private refreshSuggestion(): void {\n    const word = this.getCurrentWord();\n    if (word && VAR_NAME_REGEX.test(word)) {\n      this.pendingWord = word;\n      this.showSuggestion = true;\n    } else {\n      this.pendingWord = '';\n      this.showSuggestion = false;\n    }\n    this.cdr.markForCheck();\n  }\n\n  private hideSuggestion(): void {\n    if (!this.showSuggestion && !this.pendingWord) return;\n    this.showSuggestion = false;\n    this.pendingWord = '';\n    this.cdr.markForCheck();\n  }\n\n  private autoChipify(): void {\n    const editor = this.editorRef.nativeElement;\n    const textNodes: Text[] = [];\n    editor.childNodes.forEach((n) => {\n      if (n.nodeType === Node.TEXT_NODE) textNodes.push(n as Text);\n    });\n    for (const tn of textNodes) {\n      const text = tn.nodeValue || '';\n      VAR_TOKEN_REGEX.lastIndex = 0;\n      if (!VAR_TOKEN_REGEX.test(text)) continue;\n      VAR_TOKEN_REGEX.lastIndex = 0;\n      const parent = tn.parentNode;\n      if (!parent) continue;\n      const frag = document.createDocumentFragment();\n      let lastIndex = 0;\n      let m: RegExpExecArray | null;\n      while ((m = VAR_TOKEN_REGEX.exec(text)) !== null) {\n        if (m.index > lastIndex) {\n          frag.appendChild(document.createTextNode(text.slice(lastIndex, m.index)));\n        }\n        frag.appendChild(this.buildChip(m[1]));\n        lastIndex = m.index + m[0].length;\n      }\n      if (lastIndex < text.length) {\n        frag.appendChild(document.createTextNode(text.slice(lastIndex)));\n      }\n      parent.replaceChild(frag, tn);\n      this.placeCaretAtEnd();\n      const str = this.serialize();\n      this.lastEmitted = str;\n      this.valueChange.emit(str);\n      return;\n    }\n  }\n\n  private render(value: string): void {\n    const editor = this.editorRef.nativeElement;\n    editor.innerHTML = '';\n    if (!value) return;\n    VAR_TOKEN_REGEX.lastIndex = 0;\n    let lastIndex = 0;\n    let m: RegExpExecArray | null;\n    while ((m = VAR_TOKEN_REGEX.exec(value)) !== null) {\n      if (m.index > lastIndex) {\n        editor.appendChild(document.createTextNode(value.slice(lastIndex, m.index)));\n      }\n      editor.appendChild(this.buildChip(m[1]));\n      lastIndex = m.index + m[0].length;\n    }\n    if (lastIndex < value.length) {\n      editor.appendChild(document.createTextNode(value.slice(lastIndex)));\n    }\n  }\n\n  private serialize(): string {\n    const editor = this.editorRef.nativeElement;\n    let out = '';\n    editor.childNodes.forEach((n) => {\n      if (n.nodeType === Node.TEXT_NODE) {\n        out += (n.nodeValue || '').replace(/\\u00a0/g, ' ');\n      } else if (n.nodeType === Node.ELEMENT_NODE) {\n        const el = n as HTMLElement;\n        if (el.classList.contains('cqa-var-chip')) {\n          const name = el.getAttribute('data-var') || '';\n          if (name) out += '${' + name + '}';\n        } else {\n          out += el.textContent || '';\n        }\n      }\n    });\n    return out;\n  }\n\n  private buildChip(name: string): HTMLElement {\n    const chip = document.createElement('span');\n    chip.className = 'cqa-var-chip';\n    chip.setAttribute('contenteditable', 'false');\n    chip.setAttribute('data-var', name);\n    const label = document.createElement('span');\n    label.className = 'cqa-var-chip__label';\n    label.textContent = '${' + name + '}';\n    const remove = document.createElement('span');\n    remove.className = 'cqa-var-chip__remove';\n    remove.textContent = '×';\n    remove.addEventListener('mousedown', (e) => e.preventDefault());\n    remove.addEventListener('click', (e) => {\n      e.preventDefault();\n      e.stopPropagation();\n      chip.remove();\n      this.onInput();\n    });\n    chip.appendChild(label);\n    chip.appendChild(remove);\n    return chip;\n  }\n\n  private getSelectionRange(): Range | null {\n    const sel = window.getSelection();\n    if (!sel || sel.rangeCount === 0) return null;\n    const range = sel.getRangeAt(0);\n    if (!this.editorRef.nativeElement.contains(range.startContainer)) return null;\n    return range;\n  }\n\n  private getNodeBeforeCaret(): Node | null {\n    const range = this.getSelectionRange();\n    if (!range || !range.collapsed) return null;\n    const { startContainer, startOffset } = range;\n    if (startContainer.nodeType === Node.TEXT_NODE) {\n      if (startOffset > 0) return null;\n      return startContainer.previousSibling;\n    }\n    return startContainer.childNodes[startOffset - 1] || null;\n  }\n\n  private getCurrentWord(): string {\n    const range = this.getSelectionRange();\n    if (!range) return '';\n    const node = range.startContainer;\n    if (node.nodeType !== Node.TEXT_NODE) return '';\n    const text = node.nodeValue || '';\n    const before = text.slice(0, range.startOffset);\n    const match = before.match(/([A-Za-z_][A-Za-z0-9_]*)$/);\n    return match ? match[1] : '';\n  }\n\n  private insertChipReplacingCurrentWord(name: string): void {\n    const range = this.getSelectionRange();\n    const chip = this.buildChip(name);\n    if (!range) {\n      this.editorRef.nativeElement.appendChild(chip);\n      this.placeCaretAfter(chip);\n      return;\n    }\n    const node = range.startContainer;\n    if (node.nodeType === Node.TEXT_NODE) {\n      const text = node.nodeValue || '';\n      const before = text.slice(0, range.startOffset);\n      const after = text.slice(range.startOffset);\n      const wordMatch = before.match(/([A-Za-z_][A-Za-z0-9_]*)$/);\n      const beforeKept = wordMatch ? before.slice(0, before.length - wordMatch[1].length) : before;\n      const parent = node.parentNode!;\n      const frag = document.createDocumentFragment();\n      if (beforeKept) frag.appendChild(document.createTextNode(beforeKept));\n      frag.appendChild(chip);\n      if (after) frag.appendChild(document.createTextNode(after));\n      parent.replaceChild(frag, node);\n    } else {\n      range.insertNode(chip);\n    }\n    this.placeCaretAfter(chip);\n  }\n\n  private placeCaretAfter(node: Node): void {\n    const sel = window.getSelection();\n    if (!sel) return;\n    const range = document.createRange();\n    range.setStartAfter(node);\n    range.collapse(true);\n    sel.removeAllRanges();\n    sel.addRange(range);\n  }\n\n  private placeCaretAtEnd(): void {\n    const editor = this.editorRef.nativeElement;\n    const sel = window.getSelection();\n    if (!sel) return;\n    const range = document.createRange();\n    range.selectNodeContents(editor);\n    range.collapse(false);\n    sel.removeAllRanges();\n    sel.addRange(range);\n  }\n}\n","<div class=\"cqa-mixed-var-wrapper\">\n  <div\n    #editor\n    class=\"cqa-mixed-var-editor\"\n    role=\"textbox\"\n    spellcheck=\"false\"\n    [attr.contenteditable]=\"!disabled\"\n    [attr.data-placeholder]=\"placeholder\"\n    (input)=\"onInput()\"\n    (keydown)=\"onKeyDown($event)\"\n    (blur)=\"onEditorBlur()\">\n  </div>\n\n  <div\n    *ngIf=\"showSuggestion\"\n    class=\"cqa-mixed-var-suggest\"\n    (mousedown)=\"$event.preventDefault()\">\n    <button type=\"button\" class=\"cqa-mixed-var-suggest__item\" (click)=\"addAsText()\">\n      Add as Text <span class=\"cqa-mixed-var-suggest__hint\">\"{{ pendingWord }}\"</span>\n    </button>\n    <button type=\"button\" class=\"cqa-mixed-var-suggest__item cqa-mixed-var-suggest__item--primary\" (click)=\"addAsVariable()\">\n      Add as local variable <span class=\"cqa-mixed-var-suggest__hint\">${{ '{' }}{{ pendingWord }}{{ '}' }}</span>\n    </button>\n  </div>\n</div>\n"]}
581
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mixed-variable-input.component.js","sourceRoot":"","sources":["../../../../../src/lib/mixed-variable-input/mixed-variable-input.component.ts","../../../../../src/lib/mixed-variable-input/mixed-variable-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, EventEmitter, HostListener, Input, OnChanges, Output, SimpleChanges, ViewChild } from '@angular/core';\n\nexport interface MixedSegment {\n  type: 'text' | 'variable';\n  value: string;\n}\n\nexport interface MixedVariableValidity {\n  valid: boolean;\n  error: string | null;\n}\n\nconst VAR_PATH = '[A-Za-z_][A-Za-z0-9_]*(?:\\\\.[A-Za-z_][A-Za-z0-9_]*|\\\\[\\\\d+\\\\])*';\nconst VAR_NAME_REGEX = new RegExp('^' + VAR_PATH + '$');\nconst VAR_TOKEN_REGEX = new RegExp('\\\\$\\\\{(' + VAR_PATH + ')\\\\}', 'g');\nconst VAR_PATH_TAIL_REGEX = new RegExp('(' + VAR_PATH + ')$');\nconst INVALID_VAR_MESSAGE = 'Invalid variable name.';\nconst BLUR_HIDE_DELAY_MS = 150;\n\n@Component({\n  selector: 'cqa-mixed-variable-input',\n  templateUrl: './mixed-variable-input.component.html',\n  styleUrls: [],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root' },\n})\nexport class MixedVariableInputComponent implements OnChanges, AfterViewInit {\n  @Input() value = '';\n  @Input() placeholder = 'Text Input';\n  @Input() disabled = false;\n\n  @Output() valueChange = new EventEmitter<string>();\n  @Output() validityChange = new EventEmitter<MixedVariableValidity>();\n\n  @ViewChild('editor', { static: true }) editorRef!: ElementRef<HTMLDivElement>;\n\n  showSuggestion = false;\n  pendingWord = '';\n  canAddAsVariable = false;\n  errorMessage: string | null = null;\n\n  private selectionRange: Range | null = null;\n  private selectionMode = false;\n  private lastEmitted = '';\n  private lastValid: boolean | null = null;\n\n  constructor(\n    private cdr: ChangeDetectorRef,\n    private hostRef: ElementRef<HTMLElement>,\n  ) {}\n\n  // ---- Lifecycle ------------------------------------------------------------\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (!changes['value'] || !this.editorRef) return;\n    const incoming = changes['value'].currentValue ?? '';\n    if (incoming === this.lastEmitted) return;\n    this.render(incoming);\n    this.validate(incoming);\n  }\n\n  ngAfterViewInit(): void {\n    const initial = this.value || '';\n    this.render(initial);\n    this.validate(initial);\n  }\n\n  // ---- Editor event handlers -----------------------------------------------\n\n  onInput(): void {\n    this.flattenStructure();\n    this.sanitizeChips();\n    this.autoChipify();\n    this.emitValue();\n    this.refreshSuggestion();\n  }\n\n  onEditorFocus(): void {\n    this.refreshSuggestion();\n  }\n\n  onEditorBlur(): void {\n    // Delay so clicks on suggestion buttons register before we hide them.\n    setTimeout(() => {\n      this.autoChipify(true);\n      this.hideSuggestion();\n    }, BLUR_HIDE_DELAY_MS);\n  }\n\n  onKeyDown(event: KeyboardEvent): void {\n    if (event.key === 'Enter') {\n      event.preventDefault();\n      return;\n    }\n    if (event.key === 'Tab') return;\n    if (event.key === 'Escape') {\n      if (this.showSuggestion) {\n        event.preventDefault();\n        this.hideSuggestion();\n      }\n      return;\n    }\n    if (event.key === 'Backspace') {\n      const node = this.getNodeBeforeCaret();\n      if (node && (node as HTMLElement).classList?.contains('cqa-var-chip')) {\n        event.preventDefault();\n        (node as HTMLElement).remove();\n        this.onInput();\n      }\n    }\n  }\n\n  onPaste(event: ClipboardEvent): void {\n    event.preventDefault();\n    const text = event.clipboardData?.getData('text/plain') ?? '';\n    if (!text) return;\n    this.insertPlainTextAtCaret(text);\n    this.onInput();\n  }\n\n  onDrop(event: DragEvent): void {\n    event.preventDefault();\n    const text = event.dataTransfer?.getData('text/plain') ?? '';\n    if (text) {\n      this.editorRef.nativeElement.focus();\n      this.insertPlainTextAtCaret(text);\n    }\n    this.onInput();\n  }\n\n  @HostListener('dragstart', ['$event'])\n  onDragStart(event: DragEvent): void {\n    event.preventDefault();\n  }\n\n  @HostListener('document:mousedown', ['$event'])\n  onDocumentMouseDown(event: MouseEvent): void {\n    if (!this.showSuggestion) return;\n    if (this.hostRef.nativeElement.contains(event.target as Node)) return;\n    this.hideSuggestion();\n  }\n\n  @HostListener('document:selectionchange')\n  onDocumentSelectionChange(): void {\n    if (document.activeElement !== this.editorRef?.nativeElement) return;\n    this.refreshSuggestion();\n  }\n\n  // ---- Suggestion actions ---------------------------------------------------\n\n  addAsText(): void {\n    this.hideSuggestion();\n    this.editorRef.nativeElement.focus();\n  }\n\n  addAsVariable(): void {\n    const name = this.normalizeCandidate(this.pendingWord || '');\n    if (!name || !VAR_NAME_REGEX.test(name)) {\n      this.hideSuggestion();\n      return;\n    }\n    if (this.selectionMode && this.selectionRange) {\n      if (this.rangeCrossesChip(this.selectionRange)) {\n        this.hideSuggestion();\n        return;\n      }\n      this.insertChipReplacingRange(name, this.selectionRange);\n    } else {\n      this.insertChipReplacingCurrentWord(name);\n    }\n    this.hideSuggestion();\n    this.emitValue();\n  }\n\n  // ---- Value emission / validation -----------------------------------------\n\n  private emitValue(): void {\n    const str = this.serialize();\n    if (str !== this.lastEmitted) {\n      this.lastEmitted = str;\n      this.valueChange.emit(str);\n    }\n    this.validate(str);\n  }\n\n  private validate(str: string): void {\n    const error = this.findValidationError(str);\n    const valid = error === null;\n    if (this.lastValid === valid && this.errorMessage === error) return;\n    this.lastValid = valid;\n    this.errorMessage = error;\n    this.validityChange.emit({ valid, error });\n    this.cdr.markForCheck();\n  }\n\n  private findValidationError(str: string): string | null {\n    if (!str) return null;\n    const braceBody = /\\$\\{([^}]*)\\}/g;\n    let m: RegExpExecArray | null;\n    braceBody.lastIndex = 0;\n    while ((m = braceBody.exec(str)) !== null) {\n      const body = m[1];\n      if (body !== body.trim() || !VAR_NAME_REGEX.test(body)) {\n        return INVALID_VAR_MESSAGE;\n      }\n    }\n    const openCount = (str.match(/\\$\\{/g) || []).length;\n    const closedCount = (str.match(braceBody) || []).length;\n    if (openCount > closedCount) return INVALID_VAR_MESSAGE;\n    return null;\n  }\n\n  // ---- Suggestion state -----------------------------------------------------\n\n  private refreshSuggestion(): void {\n    const selInfo = this.getSelectedInfo();\n    if (selInfo) {\n      const normalized = this.normalizeCandidate(selInfo.text);\n      this.pendingWord = normalized;\n      this.selectionRange = selInfo.range;\n      this.selectionMode = true;\n      this.showSuggestion = true;\n      this.canAddAsVariable =\n        !!normalized && VAR_NAME_REGEX.test(normalized) && !this.rangeCrossesChip(selInfo.range);\n      this.cdr.markForCheck();\n      return;\n    }\n    this.selectionMode = false;\n    this.selectionRange = null;\n    const word = this.getCurrentWord();\n    this.pendingWord = word;\n    this.showSuggestion = !!word;\n    this.canAddAsVariable = !!word && VAR_NAME_REGEX.test(word);\n    this.cdr.markForCheck();\n  }\n\n  private hideSuggestion(): void {\n    if (!this.showSuggestion && !this.pendingWord) return;\n    this.showSuggestion = false;\n    this.pendingWord = '';\n    this.canAddAsVariable = false;\n    this.cdr.markForCheck();\n  }\n\n  private normalizeCandidate(text: string): string {\n    let t = text.trim();\n    if (t.startsWith('${') && t.endsWith('}')) {\n      t = t.slice(2, -1).trim();\n    }\n    return t;\n  }\n\n  // ---- DOM sanitization -----------------------------------------------------\n\n  private flattenStructure(): void {\n    const editor = this.editorRef.nativeElement;\n    const wrappers = Array.from(editor.querySelectorAll('div, p')) as HTMLElement[];\n    const strayInlines = Array.from(editor.querySelectorAll('span')).filter(\n      (s) => !s.classList.contains('cqa-var-chip') && !s.closest('.cqa-var-chip'),\n    ) as HTMLElement[];\n    for (const el of [...wrappers, ...strayInlines]) {\n      if (el.closest('.cqa-var-chip')) continue;\n      while (el.firstChild) el.parentNode?.insertBefore(el.firstChild, el);\n      el.remove();\n    }\n    const brs = Array.from(editor.querySelectorAll('br')) as HTMLElement[];\n    for (const br of brs) {\n      if (br.closest('.cqa-var-chip')) continue;\n      br.parentNode?.replaceChild(document.createTextNode(' '), br);\n    }\n  }\n\n  private sanitizeChips(): void {\n    const editor = this.editorRef.nativeElement;\n    const chips = Array.from(editor.querySelectorAll('.cqa-var-chip')) as HTMLElement[];\n    for (const chip of chips) {\n      const name = (chip.getAttribute('data-var') || '').trim();\n      if (!name || !VAR_NAME_REGEX.test(name)) {\n        chip.parentNode?.replaceChild(document.createTextNode(chip.textContent || ''), chip);\n        continue;\n      }\n      const structureBroken =\n        chip.getAttribute('contenteditable') !== 'false' ||\n        !chip.querySelector('.cqa-var-chip__remove');\n      if (structureBroken) {\n        chip.parentNode?.replaceChild(this.buildChip(name), chip);\n      }\n    }\n    // Remove stray remove icons left from partial undo/redo states.\n    const orphans = Array.from(editor.querySelectorAll('.cqa-var-chip__remove')) as HTMLElement[];\n    for (const o of orphans) {\n      if (!o.closest('.cqa-var-chip')) o.remove();\n    }\n  }\n\n  // ---- Chipification --------------------------------------------------------\n\n  private autoChipify(force = false): void {\n    const editor = this.editorRef.nativeElement;\n    const caretRange = this.getSelectionRange();\n    let changed = false;\n    let lastChip: Node | null = null;\n\n    for (const tn of this.collectDescendantTextNodes(editor)) {\n      if (!tn.parentNode) continue;\n      const text = tn.nodeValue || '';\n      VAR_TOKEN_REGEX.lastIndex = 0;\n      if (!VAR_TOKEN_REGEX.test(text)) continue;\n      VAR_TOKEN_REGEX.lastIndex = 0;\n\n      const caretOffset =\n        !force && caretRange && caretRange.collapsed && caretRange.startContainer === tn\n          ? caretRange.startOffset\n          : -1;\n\n      const matches: Array<{ start: number; end: number; name: string }> = [];\n      let m: RegExpExecArray | null;\n      while ((m = VAR_TOKEN_REGEX.exec(text)) !== null) {\n        const start = m.index;\n        const end = start + m[0].length;\n        // Keep the token as plain text while the caret is strictly inside.\n        if (caretOffset > start && caretOffset < end) continue;\n        matches.push({ start, end, name: m[1] });\n      }\n      if (!matches.length) continue;\n\n      const frag = document.createDocumentFragment();\n      let cursor = 0;\n      for (const mt of matches) {\n        if (mt.start > cursor) {\n          frag.appendChild(document.createTextNode(text.slice(cursor, mt.start)));\n        }\n        const chip = this.buildChip(mt.name);\n        frag.appendChild(chip);\n        lastChip = chip;\n        cursor = mt.end;\n      }\n      if (cursor < text.length) {\n        frag.appendChild(document.createTextNode(text.slice(cursor)));\n      }\n      tn.parentNode.replaceChild(frag, tn);\n      changed = true;\n    }\n\n    if (changed && lastChip) this.placeCaretAfter(lastChip);\n  }\n\n  private collectDescendantTextNodes(root: Element): Text[] {\n    const out: Text[] = [];\n    const walker = document.createTreeWalker(root, NodeFilter.SHOW_TEXT, {\n      acceptNode: (node) => {\n        let p: Node | null = node.parentNode;\n        while (p && p !== root) {\n          if ((p as Element).classList?.contains?.('cqa-var-chip')) {\n            return NodeFilter.FILTER_REJECT;\n          }\n          p = p.parentNode;\n        }\n        return NodeFilter.FILTER_ACCEPT;\n      },\n    });\n    let n: Node | null;\n    while ((n = walker.nextNode())) out.push(n as Text);\n    return out;\n  }\n\n  // ---- Render / serialize ---------------------------------------------------\n\n  private render(value: string): void {\n    const editor = this.editorRef.nativeElement;\n    editor.innerHTML = '';\n    if (!value) return;\n    VAR_TOKEN_REGEX.lastIndex = 0;\n    let lastIndex = 0;\n    let m: RegExpExecArray | null;\n    while ((m = VAR_TOKEN_REGEX.exec(value)) !== null) {\n      if (m.index > lastIndex) {\n        editor.appendChild(document.createTextNode(value.slice(lastIndex, m.index)));\n      }\n      editor.appendChild(this.buildChip(m[1]));\n      lastIndex = m.index + m[0].length;\n    }\n    if (lastIndex < value.length) {\n      editor.appendChild(document.createTextNode(value.slice(lastIndex)));\n    }\n  }\n\n  private serialize(): string {\n    const editor = this.editorRef.nativeElement;\n    let out = '';\n    const walk = (node: Node): void => {\n      if (node.nodeType === Node.TEXT_NODE) {\n        out += (node.nodeValue || '').replace(/\\u00a0/g, ' ');\n        return;\n      }\n      if (node.nodeType !== Node.ELEMENT_NODE) return;\n      const el = node as HTMLElement;\n      if (el.classList.contains('cqa-var-chip')) {\n        const name = el.getAttribute('data-var') || '';\n        if (name) out += '${' + name + '}';\n        return;\n      }\n      if (el.tagName === 'BR') {\n        out += '\\n';\n        return;\n      }\n      const isBlock = el.tagName === 'DIV' || el.tagName === 'P';\n      if (isBlock && out.length > 0 && !out.endsWith('\\n')) out += '\\n';\n      el.childNodes.forEach(walk);\n    };\n    editor.childNodes.forEach(walk);\n    return out;\n  }\n\n  private buildChip(name: string): HTMLElement {\n    const chip = document.createElement('span');\n    chip.className = 'cqa-var-chip';\n    chip.setAttribute('contenteditable', 'false');\n    chip.setAttribute('data-var', name);\n    const label = document.createElement('span');\n    label.className = 'cqa-var-chip__label';\n    label.textContent = '${' + name + '}';\n    const remove = document.createElement('span');\n    remove.className = 'cqa-var-chip__remove';\n    remove.textContent = '×';\n    remove.addEventListener('mousedown', (e) => e.preventDefault());\n    remove.addEventListener('click', (e) => {\n      e.preventDefault();\n      e.stopPropagation();\n      chip.remove();\n      this.onInput();\n    });\n    chip.appendChild(label);\n    chip.appendChild(remove);\n    return chip;\n  }\n\n  // ---- Selection / caret helpers -------------------------------------------\n\n  private getSelectionRange(): Range | null {\n    const sel = window.getSelection();\n    if (!sel || sel.rangeCount === 0) return null;\n    const range = sel.getRangeAt(0);\n    if (!this.editorRef.nativeElement.contains(range.startContainer)) return null;\n    return range;\n  }\n\n  private getSelectedInfo(): { text: string; range: Range } | null {\n    const sel = window.getSelection();\n    if (!sel || sel.rangeCount === 0) return null;\n    const range = sel.getRangeAt(0);\n    if (range.collapsed) return null;\n    const editor = this.editorRef.nativeElement;\n    if (!editor.contains(range.startContainer) || !editor.contains(range.endContainer)) return null;\n    const text = range.toString();\n    if (!text) return null;\n    return { text, range: range.cloneRange() };\n  }\n\n  private rangeCrossesChip(range: Range): boolean {\n    const contents = range.cloneContents();\n    return !!contents.querySelector?.('.cqa-var-chip');\n  }\n\n  private getNodeBeforeCaret(): Node | null {\n    const range = this.getSelectionRange();\n    if (!range || !range.collapsed) return null;\n    const { startContainer, startOffset } = range;\n    if (startContainer.nodeType === Node.TEXT_NODE) {\n      if (startOffset > 0) return null;\n      return startContainer.previousSibling;\n    }\n    return startContainer.childNodes[startOffset - 1] || null;\n  }\n\n  private getCurrentWord(): string {\n    const range = this.getSelectionRange();\n    if (!range || !range.collapsed) return '';\n    let node: Node = range.startContainer;\n    let offset = range.startOffset;\n    if (node.nodeType !== Node.TEXT_NODE) {\n      const prev = node.childNodes[offset - 1];\n      if (!prev || prev.nodeType !== Node.TEXT_NODE) return '';\n      node = prev;\n      offset = (prev.nodeValue || '').length;\n    }\n    const text = (node as Text).nodeValue || '';\n    const braceOpen = text.lastIndexOf('${', offset - 1);\n    if (braceOpen !== -1) {\n      const closeIdx = text.indexOf('}', braceOpen + 2);\n      const tokenClosedBeforeCaret = closeIdx !== -1 && closeIdx < offset;\n      if (!tokenClosedBeforeCaret) {\n        const start = braceOpen + 2;\n        const end = closeIdx === -1 ? text.length : closeIdx;\n        return text.slice(start, end);\n      }\n    }\n    const match = text.slice(0, offset).match(VAR_PATH_TAIL_REGEX);\n    return match ? match[1] : '';\n  }\n\n  // ---- Chip insertion -------------------------------------------------------\n\n  private insertChipReplacingRange(name: string, range: Range): void {\n    const chip = this.buildChip(name);\n    range.deleteContents();\n    range.insertNode(chip);\n    this.stripBracesAround(chip);\n    this.placeCaretAfter(chip);\n  }\n\n  private insertChipReplacingCurrentWord(name: string): void {\n    const range = this.getSelectionRange();\n    const chip = this.buildChip(name);\n    if (!range) {\n      this.editorRef.nativeElement.appendChild(chip);\n      this.placeCaretAfter(chip);\n      return;\n    }\n    const node = range.startContainer;\n    if (node.nodeType !== Node.TEXT_NODE) {\n      range.insertNode(chip);\n      this.placeCaretAfter(chip);\n      return;\n    }\n    const text = node.nodeValue || '';\n    const before = text.slice(0, range.startOffset);\n    const after = text.slice(range.startOffset);\n    const wordMatch = before.match(VAR_PATH_TAIL_REGEX);\n    let beforeKept = wordMatch ? before.slice(0, before.length - wordMatch[1].length) : before;\n    let afterKept = after;\n    if (beforeKept.endsWith('${') && afterKept.startsWith('}')) {\n      beforeKept = beforeKept.slice(0, -2);\n      afterKept = afterKept.slice(1);\n    } else if (beforeKept.endsWith('${')) {\n      beforeKept = beforeKept.slice(0, -2);\n    }\n    const frag = document.createDocumentFragment();\n    if (beforeKept) frag.appendChild(document.createTextNode(beforeKept));\n    frag.appendChild(chip);\n    if (afterKept) frag.appendChild(document.createTextNode(afterKept));\n    node.parentNode!.replaceChild(frag, node);\n    this.placeCaretAfter(chip);\n  }\n\n  private stripBracesAround(chip: HTMLElement): void {\n    const prev = chip.previousSibling;\n    const next = chip.nextSibling;\n    if (prev && prev.nodeType === Node.TEXT_NODE) {\n      const t = prev.nodeValue || '';\n      if (t.endsWith('${')) prev.nodeValue = t.slice(0, -2);\n    }\n    if (next && next.nodeType === Node.TEXT_NODE) {\n      const t = next.nodeValue || '';\n      if (t.startsWith('}')) next.nodeValue = t.slice(1);\n    }\n  }\n\n  private insertPlainTextAtCaret(text: string): void {\n    const range = this.getSelectionRange();\n    if (!range) {\n      this.editorRef.nativeElement.appendChild(document.createTextNode(text));\n      this.placeCaretAtEnd();\n      return;\n    }\n    range.deleteContents();\n    const node = document.createTextNode(text);\n    range.insertNode(node);\n    this.placeCaretAfter(node);\n  }\n\n  private placeCaretAfter(node: Node): void {\n    const sel = window.getSelection();\n    if (!sel) return;\n    const range = document.createRange();\n    range.setStartAfter(node);\n    range.collapse(true);\n    sel.removeAllRanges();\n    sel.addRange(range);\n  }\n\n  private placeCaretAtEnd(): void {\n    const sel = window.getSelection();\n    if (!sel) return;\n    const range = document.createRange();\n    range.selectNodeContents(this.editorRef.nativeElement);\n    range.collapse(false);\n    sel.removeAllRanges();\n    sel.addRange(range);\n  }\n}\n","<div class=\"cqa-mixed-var-wrapper\">\n  <div\n    #editor\n    class=\"cqa-mixed-var-editor\"\n    role=\"textbox\"\n    spellcheck=\"false\"\n    [attr.contenteditable]=\"!disabled\"\n    [attr.data-placeholder]=\"placeholder\"\n    [attr.aria-invalid]=\"errorMessage ? 'true' : null\"\n    [class.cqa-mixed-var-editor--invalid]=\"errorMessage\"\n    (input)=\"onInput()\"\n    (keydown)=\"onKeyDown($event)\"\n    (focus)=\"onEditorFocus()\"\n    (paste)=\"onPaste($event)\"\n    (drop)=\"onDrop($event)\"\n    (blur)=\"onEditorBlur()\">\n  </div>\n  <p *ngIf=\"errorMessage\" class=\"cqa-mixed-var-error\" role=\"alert\">{{ errorMessage }}</p>\n\n  <div\n    *ngIf=\"showSuggestion\"\n    class=\"cqa-mixed-var-suggest\"\n    (mousedown)=\"$event.preventDefault()\">\n    <button type=\"button\" class=\"cqa-mixed-var-suggest__item\" (click)=\"addAsText()\">\n      Add as Text <span *ngIf=\"pendingWord\" class=\"cqa-mixed-var-suggest__hint\">\"{{ pendingWord }}\"</span>\n    </button>\n    <button *ngIf=\"canAddAsVariable\" type=\"button\"\n      class=\"cqa-mixed-var-suggest__item cqa-mixed-var-suggest__item--primary\"\n      (click)=\"addAsVariable()\">\n      Add as local variable <span class=\"cqa-mixed-var-suggest__hint\">${{ '{' }}{{ pendingWord }}{{ '}' }}</span>\n    </button>\n  </div>\n</div>\n"]}