@myrmidon/gve-core 0.0.4 → 0.0.6

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 (50) hide show
  1. package/README.md +45 -77
  2. package/esm2022/lib/components/animation-timeline/animation-timeline.component.mjs +50 -19
  3. package/esm2022/lib/components/animation-timeline-set/animation-timeline-set.component.mjs +32 -8
  4. package/esm2022/lib/components/animation-tween/animation-tween.component.mjs +48 -25
  5. package/esm2022/lib/components/base-text-char/base-text-char.component.mjs +13 -4
  6. package/esm2022/lib/components/base-text-editor/base-text-editor.component.mjs +10 -6
  7. package/esm2022/lib/components/base-text-view/base-text-view.component.mjs +39 -5
  8. package/esm2022/lib/components/batch-operation-editor/batch-operation-editor.component.mjs +111 -0
  9. package/esm2022/lib/components/chain-operation-editor/chain-operation-editor.component.mjs +97 -39
  10. package/esm2022/lib/components/chain-result-view/chain-result-view.component.mjs +48 -13
  11. package/esm2022/lib/components/feature-editor/feature-editor.component.mjs +27 -12
  12. package/esm2022/lib/components/feature-set-editor/feature-set-editor.component.mjs +32 -9
  13. package/esm2022/lib/components/feature-set-view/feature-set-view.component.mjs +28 -8
  14. package/esm2022/lib/components/ln-heights-editor/ln-heights-editor.component.mjs +38 -7
  15. package/esm2022/lib/components/operation-source-editor/operation-source-editor.component.mjs +29 -4
  16. package/esm2022/lib/components/simple-tree/simple-tree.component.mjs +20 -4
  17. package/esm2022/lib/components/snapshot-editor/snapshot-editor.component.mjs +471 -159
  18. package/esm2022/lib/components/steps-map/steps-map.component.mjs +37 -9
  19. package/esm2022/lib/models.mjs +1 -1
  20. package/esm2022/lib/services/gve-api.service.mjs +4 -4
  21. package/esm2022/lib/services/settings.service.mjs +6 -5
  22. package/esm2022/lib/validators/svg-validators.mjs +7 -1
  23. package/esm2022/public-api.mjs +2 -2
  24. package/fesm2022/myrmidon-gve-core.mjs +1159 -487
  25. package/fesm2022/myrmidon-gve-core.mjs.map +1 -1
  26. package/lib/components/animation-timeline/animation-timeline.component.d.ts +31 -9
  27. package/lib/components/animation-timeline-set/animation-timeline-set.component.d.ts +28 -4
  28. package/lib/components/animation-tween/animation-tween.component.d.ts +25 -12
  29. package/lib/components/base-text-char/base-text-char.component.d.ts +16 -0
  30. package/lib/components/base-text-editor/base-text-editor.component.d.ts +5 -1
  31. package/lib/components/base-text-view/base-text-view.component.d.ts +37 -0
  32. package/lib/components/batch-operation-editor/batch-operation-editor.component.d.ts +46 -0
  33. package/lib/components/chain-operation-editor/chain-operation-editor.component.d.ts +53 -5
  34. package/lib/components/chain-result-view/chain-result-view.component.d.ts +21 -4
  35. package/lib/components/feature-editor/feature-editor.component.d.ts +20 -6
  36. package/lib/components/feature-set-editor/feature-set-editor.component.d.ts +27 -4
  37. package/lib/components/feature-set-view/feature-set-view.component.d.ts +23 -3
  38. package/lib/components/ln-heights-editor/ln-heights-editor.component.d.ts +22 -0
  39. package/lib/components/operation-source-editor/operation-source-editor.component.d.ts +31 -0
  40. package/lib/components/simple-tree/simple-tree.component.d.ts +19 -0
  41. package/lib/components/snapshot-editor/snapshot-editor.component.d.ts +184 -19
  42. package/lib/components/steps-map/steps-map.component.d.ts +22 -3
  43. package/lib/models.d.ts +8 -0
  44. package/lib/services/gve-api.service.d.ts +33 -0
  45. package/lib/services/settings.service.d.ts +2 -1
  46. package/lib/validators/svg-validators.d.ts +6 -0
  47. package/package.json +10 -10
  48. package/public-api.d.ts +1 -1
  49. package/esm2022/lib/components/animation-vars/animation-vars.component.mjs +0 -141
  50. package/lib/components/animation-vars/animation-vars.component.d.ts +0 -30
@@ -1,4 +1,4 @@
1
- import { CUSTOM_ELEMENTS_SCHEMA, Component, EventEmitter, Input, Output, } from '@angular/core';
1
+ import { CUSTOM_ELEMENTS_SCHEMA, Component, EventEmitter, Input, Output, ViewChild, } from '@angular/core';
2
2
  import { FormControl, FormsModule, ReactiveFormsModule, Validators, } from '@angular/forms';
3
3
  import { CommonModule } from '@angular/common';
4
4
  import { MatButtonModule } from '@angular/material/button';
@@ -21,25 +21,39 @@ import { ChainResultViewComponent } from '../chain-result-view/chain-result-view
21
21
  import { LnHeightsEditorComponent } from '../ln-heights-editor/ln-heights-editor.component';
22
22
  import { BaseTextEditorComponent } from '../base-text-editor/base-text-editor.component';
23
23
  import { AnimationTimelineSetComponent } from '../animation-timeline-set/animation-timeline-set.component';
24
+ import { BatchOperationEditorComponent } from '../batch-operation-editor/batch-operation-editor.component';
24
25
  import * as i0 from "@angular/core";
25
26
  import * as i1 from "@angular/forms";
26
27
  import * as i2 from "../../services/gve-api.service";
27
- import * as i3 from "@myrmidon/ng-mat-tools";
28
- import * as i4 from "@angular/material/snack-bar";
29
- import * as i5 from "@angular/common";
30
- import * as i6 from "@angular/material/button";
31
- import * as i7 from "@angular/material/button-toggle";
32
- import * as i8 from "@angular/material/expansion";
33
- import * as i9 from "@angular/material/form-field";
34
- import * as i10 from "@angular/material/icon";
35
- import * as i11 from "@angular/material/input";
36
- import * as i12 from "@angular/material/progress-bar";
37
- import * as i13 from "@angular/material/tabs";
38
- import * as i14 from "@angular/material/tooltip";
39
- import * as i15 from "@myrmidon/ng-tools";
28
+ import * as i3 from "@angular/material/dialog";
29
+ import * as i4 from "@myrmidon/ng-mat-tools";
30
+ import * as i5 from "@angular/material/snack-bar";
31
+ import * as i6 from "@angular/common";
32
+ import * as i7 from "@angular/material/button";
33
+ import * as i8 from "@angular/material/button-toggle";
34
+ import * as i9 from "@angular/material/expansion";
35
+ import * as i10 from "@angular/material/form-field";
36
+ import * as i11 from "@angular/material/icon";
37
+ import * as i12 from "@angular/material/input";
38
+ import * as i13 from "@angular/material/progress-bar";
39
+ import * as i14 from "@angular/material/tabs";
40
+ import * as i15 from "@angular/material/tooltip";
41
+ import * as i16 from "@myrmidon/ng-tools";
42
+ // default snapshot view title
40
43
  const VIEW_TITLE = 'preview';
44
+ // suffix for IDs of SVG elements to be made transparent at first rendition
45
+ const SVG_TRANSP_SUFFIX = '_t';
41
46
  /**
42
- * A component to edit a text snapshot.
47
+ * 🔑 `gve-snapshot-editor`
48
+ *
49
+ * A component to edit a text snapshot. This is the top level component in the GVE core
50
+ * library.
51
+ *
52
+ * - ▶️ `snapshot` (`Snapshot`): the snapshot to edit.
53
+ * - ▶️ `batchOps` (`string`): the batch operations text to set for the editor.
54
+ * - ▶️ `noSave` (`boolean`): true to disable saving.
55
+ * - 🔥 `snapshotChange` (`Snapshot`): emitted when the user saves the edited snapshot.
56
+ * - 🔥 `snapshotCancel` (`void`): emitted when the user cancels the snapshot editing.
43
57
  */
44
58
  export class SnapshotEditorComponent {
45
59
  /**
@@ -54,10 +68,13 @@ export class SnapshotEditorComponent {
54
68
  }
55
69
  this._snapshot = value || undefined;
56
70
  this.updateForm(this._snapshot);
57
- this.updateViewData();
71
+ setTimeout(() => {
72
+ this.runToLast();
73
+ }, 0);
58
74
  }
59
- constructor(formBuilder, _api, _dialogService, _snackbar) {
75
+ constructor(formBuilder, _api, _dialog, _dialogService, _snackbar) {
60
76
  this._api = _api;
77
+ this._dialog = _dialog;
61
78
  this._dialogService = _dialogService;
62
79
  this._snackbar = _snackbar;
63
80
  this._nanoid = customAlphabet('1234567890abcdef', 10);
@@ -69,9 +86,13 @@ export class SnapshotEditorComponent {
69
86
  * Emitted when the user cancels the snapshot editing.
70
87
  */
71
88
  this.snapshotCancel = new EventEmitter();
72
- this.editedOpIndex = -1;
89
+ // list of operations output tags
73
90
  this.opTags = [];
91
+ // list of operation diplomatic.g element IDs
74
92
  this.opElementIds = [];
93
+ // the lines count for the current base text
94
+ this.lineCount = 0;
95
+ this.editedOpIndex = -1;
75
96
  this.opTypeMap = {
76
97
  0: 'replace',
77
98
  1: 'delete',
@@ -82,9 +103,10 @@ export class SnapshotEditorComponent {
82
103
  6: 'swap',
83
104
  7: 'annotate',
84
105
  };
85
- this.lineCount = 0;
106
+ // snapshot view data
86
107
  this.viewTitle = VIEW_TITLE;
87
108
  this.rulers = true;
109
+ this.initialStepIndex = -1;
88
110
  // general
89
111
  this.width = new FormControl(800, { nonNullable: true });
90
112
  this.height = new FormControl(600, { nonNullable: true });
@@ -105,7 +127,6 @@ export class SnapshotEditorComponent {
105
127
  this.operations = new FormControl([], {
106
128
  nonNullable: true,
107
129
  });
108
- this.inputOps = new FormControl(null);
109
130
  this.opStyle = new FormControl(null);
110
131
  // image
111
132
  this.imageUrl = new FormControl(null);
@@ -119,6 +140,7 @@ export class SnapshotEditorComponent {
119
140
  this.timelines = new FormControl([], {
120
141
  nonNullable: true,
121
142
  });
143
+ // form
122
144
  this.form = formBuilder.group({
123
145
  width: this.width,
124
146
  height: this.height,
@@ -132,7 +154,6 @@ export class SnapshotEditorComponent {
132
154
  spcWidthOffset: this.spcWidthOffset,
133
155
  textStyle: this.textStyle,
134
156
  operations: this.operations,
135
- inputOps: this.inputOps,
136
157
  opStyle: this.opStyle,
137
158
  // image
138
159
  imageUrl: this.imageUrl,
@@ -146,43 +167,38 @@ export class SnapshotEditorComponent {
146
167
  timelines: this.timelines,
147
168
  });
148
169
  }
149
- ngOnInit() {
150
- if (this.batchOps) {
151
- this.inputOps.setValue(this.batchOps);
152
- this.parseOperations();
153
- }
154
- }
155
- updateViewData(newOperation, snapshot, title = VIEW_TITLE) {
156
- console.log('update view data');
170
+ /**
171
+ * Set the view data for the snapshot view.
172
+ *
173
+ * @param snapshot The optional snapshot data to use. If not provided,
174
+ * a snapshot will be created from the form data.
175
+ * @param title The optional title to set for the view.
176
+ */
177
+ setViewData(snapshot, title = VIEW_TITLE) {
178
+ console.log('set view data');
179
+ // get the snapshot to use
157
180
  if (!snapshot) {
158
181
  snapshot = this.getSnapshot();
159
182
  }
160
- // update or add new operation in copy
161
- if (newOperation) {
162
- const index = snapshot.operations.findIndex((op) => op.id === newOperation.id);
163
- if (index > -1) {
164
- snapshot.operations.splice(index, 1, newOperation);
165
- }
166
- else {
167
- snapshot.operations.push(newOperation);
168
- }
169
- }
170
183
  // update view data
171
184
  this.viewTitle = title;
172
185
  this.visualInfo = undefined;
173
186
  this.viewData = {
174
187
  snapshot: snapshot,
175
188
  options: {
176
- // debug: true,
189
+ debug: this.debug,
177
190
  delayedRender: true,
178
191
  showRulers: true,
179
192
  showGrid: true,
180
193
  panZoom: true,
194
+ transparentIds: this._transparentIds,
181
195
  },
182
196
  };
183
- console.log(`view data: ${this.viewData}`);
197
+ console.log('view data: ', this.viewData);
184
198
  }
185
199
  updateForm(snapshot) {
200
+ this._transparentIds = undefined;
201
+ this.initialStepIndex = -1;
186
202
  if (!snapshot) {
187
203
  this.form.reset();
188
204
  }
@@ -206,7 +222,6 @@ export class SnapshotEditorComponent {
206
222
  this.spcWidthOffset.setValue(snapshot.textOptions?.spcWidthOffset);
207
223
  this.textStyle.setValue(snapshot.textStyle || null);
208
224
  this.operations.setValue(snapshot.operations);
209
- this.inputOps.reset();
210
225
  this.opStyle.setValue(snapshot.opStyle || null);
211
226
  // timelines
212
227
  const timelines = [];
@@ -217,9 +232,76 @@ export class SnapshotEditorComponent {
217
232
  }
218
233
  this.timelines.setValue(timelines);
219
234
  }
235
+ this.updateLineCount(this.baseText.value);
236
+ this.updateOpLists();
220
237
  this.form.markAsPristine();
221
238
  }
239
+ // #region base text
240
+ /**
241
+ * Update the line count based on the received text.
242
+ * @param text The text to use for counting lines.
243
+ */
244
+ updateLineCount(text) {
245
+ if (!text.length) {
246
+ this.lineCount = 0;
247
+ }
248
+ else {
249
+ this.lineCount = text.filter((c) => c.data === '\n').length + 1;
250
+ }
251
+ }
252
+ /**
253
+ * Handle the event fired by the base text editor to pick a text range.
254
+ * @param range The picked range.
255
+ */
256
+ onRangePick(range) {
257
+ this.textRange = range;
258
+ }
259
+ /**
260
+ * Handle the event fired by the base text editor to change the base text.
261
+ * @param text The text to set.
262
+ */
263
+ onTextChange(text) {
264
+ console.log('text change', text);
265
+ // update the form control and reset the current text range
266
+ this.baseText.setValue(text);
267
+ this.baseText.updateValueAndValidity();
268
+ this.baseText.markAsDirty();
269
+ this.textRange = undefined;
270
+ // update the line count
271
+ this.updateLineCount(text);
272
+ // remove all operations and update the view data
273
+ this.removeAllOperations();
274
+ // remove all timelines
275
+ this.timelines.reset();
276
+ }
277
+ // #endregion
278
+ // #region operations
279
+ /**
280
+ * Update the lists of operation output tags and element IDs by collecting
281
+ * all the operation tags and the IDs of the elements in their diplomatic.g
282
+ * SVG code if any.
283
+ */
284
+ updateOpLists() {
285
+ const tags = new Set();
286
+ const ids = new Set();
287
+ let n = 0;
288
+ for (const op of this.operations.value) {
289
+ // output tag
290
+ n++;
291
+ tags.add(op.outputTag || `v${n}`);
292
+ // element IDs
293
+ if (op.diplomatics?.g) {
294
+ this.parseSvgIds(op.diplomatics.g)?.forEach((id) => ids.add(id));
295
+ }
296
+ }
297
+ this.opTags = [...tags];
298
+ this.opElementIds = [...ids];
299
+ }
300
+ /**
301
+ * Edit a new operation.
302
+ */
222
303
  editNewOperation() {
304
+ // create a new operation and edit it
223
305
  this.editedOp = {
224
306
  id: this._nanoid(),
225
307
  type: 0,
@@ -228,228 +310,455 @@ export class SnapshotEditorComponent {
228
310
  };
229
311
  this.editedOpIndex = -1;
230
312
  }
313
+ /**
314
+ * Edit (a copy of) the operation at the specified index.
315
+ * @param index The operation index.
316
+ */
231
317
  editOperation(index) {
232
318
  this.editedOpIndex = index;
233
319
  this.editedOp = deepCopy(this.operations.value[index]);
234
320
  }
235
- closeEditedOp() {
321
+ /**
322
+ * Close the currently edited operation.
323
+ */
324
+ closeEditedOperation() {
236
325
  this.editedOpIndex = -1;
237
326
  this.editedOp = undefined;
238
327
  }
328
+ /**
329
+ * Handle the event fired by the operation editor to cancel
330
+ * the current operation edits.
331
+ */
239
332
  onOperationCancel() {
240
- this.closeEditedOp();
333
+ this.closeEditedOperation();
241
334
  }
242
- onOperationChange(op) {
335
+ /**
336
+ * Handle the event fired by the operation editor to change
337
+ * the currently edited operation, or add a new one if that
338
+ * was a new operation.
339
+ * @param op The changed operation.
340
+ */
341
+ async onOperationChange(op) {
243
342
  console.log('operation change');
244
343
  const operations = [...this.operations.value];
344
+ // replace or add the operation
345
+ let i = this.editedOpIndex;
245
346
  if (this.editedOpIndex > -1) {
246
347
  operations.splice(this.editedOpIndex, 1, op);
247
348
  }
248
349
  else {
249
350
  operations.push(op);
351
+ i = operations.length - 1;
250
352
  }
353
+ // update the operations form control
251
354
  this.operations.setValue(operations);
252
355
  this.operations.markAsDirty();
253
356
  this.operations.updateValueAndValidity();
254
- this.closeEditedOp();
357
+ // update the operation lists
358
+ this.updateOpLists();
359
+ // update the view data
360
+ this.result = await this.runTo(i);
361
+ // close the edited operation
362
+ this.closeEditedOperation();
255
363
  }
364
+ /**
365
+ * Delete the operation at the specified index.
366
+ * @param index The index of the operation to delete.
367
+ */
256
368
  deleteOperation(index) {
257
369
  this._dialogService
258
370
  .confirm('Confirm Deletion', `Delete operation "${this.operations.value[index].id}"?`)
259
371
  .subscribe((yes) => {
260
372
  if (yes) {
373
+ // close the edited operation if it is the one being deleted
374
+ if (this.editedOpIndex === index) {
375
+ this.closeEditedOperation();
376
+ }
377
+ // reset the result operation ID if it is the one being deleted
261
378
  if (this.resultOperationId === this.operations.value[index].id) {
262
379
  this.resultOperationId = undefined;
263
380
  }
381
+ // delete the operation and update the form control
264
382
  const operations = [...this.operations.value];
265
383
  operations.splice(index, 1);
266
384
  this.operations.setValue(operations);
267
385
  this.operations.markAsDirty();
268
386
  this.operations.updateValueAndValidity();
387
+ // update the operation lists
388
+ this.updateOpLists();
389
+ // update the view data
390
+ setTimeout(() => {
391
+ this.runToLast();
392
+ }, 0);
269
393
  }
270
394
  });
271
395
  }
272
- onTextChange(text) {
273
- console.log('text change: ' + text);
274
- this.baseText.setValue(text);
275
- this.baseText.updateValueAndValidity();
276
- this.baseText.markAsDirty();
277
- if (!text.length) {
278
- this.lineCount = 0;
279
- }
280
- else {
281
- this.lineCount = text.filter((c) => c.data === '\n').length + 1;
282
- }
283
- this.textRange = undefined;
284
- this.closeEditedOp();
285
- this.operations.reset();
286
- this.updateViewData();
287
- }
288
- onRangePick(range) {
289
- this.textRange = range;
290
- }
396
+ /**
397
+ * Parse the operations from their text and append them to the current
398
+ * snapshot operations.
399
+ */
291
400
  parseOperations() {
292
- if (this.busy || !this.inputOps.value) {
401
+ if (this.busy) {
293
402
  return;
294
403
  }
295
- this._dialogService
296
- .confirm('Confirm', 'Parse operations?')
297
- .subscribe((yes) => {
298
- if (yes) {
299
- this.busy = true;
300
- this.parseError = undefined;
301
- this._api.parseOperations(this.inputOps.value).subscribe({
302
- next: (wrapper) => {
303
- const operations = [...this.operations.value];
304
- operations.push(...wrapper.result);
305
- this.operations.setValue(operations);
306
- this.operations.markAsDirty();
307
- this.operations.updateValueAndValidity();
308
- this.updateViewData();
309
- },
310
- error: (error) => {
311
- this.parseError = error.message;
312
- },
313
- complete: () => {
314
- this.busy = false;
315
- },
316
- });
404
+ const dialogRef = this._dialog.open(BatchOperationEditorComponent, {
405
+ data: { preset: this.batchOps },
406
+ });
407
+ dialogRef.afterClosed().subscribe((batchOps) => {
408
+ if (batchOps) {
409
+ const operations = [...this.operations.value];
410
+ operations.push(...batchOps);
411
+ this.operations.setValue(operations);
412
+ this.operations.markAsDirty();
413
+ this.operations.updateValueAndValidity();
414
+ // update the operation lists
415
+ this.updateOpLists();
416
+ // update the view data
417
+ setTimeout(() => {
418
+ this.runToLast();
419
+ }, 0);
317
420
  }
318
421
  });
319
422
  }
423
+ removeAllOperations() {
424
+ this.resultOperationId = undefined;
425
+ this.closeEditedOperation();
426
+ this.operations.reset();
427
+ this.operations.markAsDirty();
428
+ this.operations.updateValueAndValidity();
429
+ this.opTags = [];
430
+ this.opElementIds = [];
431
+ this.setViewData();
432
+ this.result = undefined;
433
+ }
434
+ /**
435
+ * Remove all the operations.
436
+ */
320
437
  clearOperations() {
321
438
  this._dialogService
322
439
  .confirm('Confirm', 'Remove all operations?')
323
440
  .subscribe((yes) => {
324
441
  if (yes) {
325
- this.resultOperationId = undefined;
326
- this.operations.reset();
327
- this.operations.markAsDirty();
328
- this.operations.updateValueAndValidity();
329
- this.updateViewData();
442
+ this.removeAllOperations();
330
443
  }
331
444
  });
332
445
  }
446
+ /**
447
+ * Parse all the id attributes from the received SVG code and
448
+ * return them as an array.
449
+ *
450
+ * @param svg The SVG content to parse or undefined.
451
+ * @returns Array of IDs found in the SVG content or undefined.
452
+ */
453
+ parseSvgIds(svg) {
454
+ if (!svg) {
455
+ return undefined;
456
+ }
457
+ const regex = /<[^>]+\bid=(["'])(.*?)\1/g;
458
+ const ids = [];
459
+ let match;
460
+ while ((match = regex.exec(svg)) !== null) {
461
+ ids.push(match[2]);
462
+ }
463
+ return ids;
464
+ }
465
+ getTransparentIds(g) {
466
+ if (!g) {
467
+ return undefined;
468
+ }
469
+ return this.parseSvgIds(g)?.filter((id) => id.endsWith(SVG_TRANSP_SUFFIX));
470
+ }
471
+ /**
472
+ * Run the operations up to the specified index (included).
473
+ * This is called when:
474
+ * - a preview is requested by the operation editor.
475
+ * - the currently edited operation is saved.
476
+ * - the user picks a step in the chain result view.
477
+ * - runToLast is called, which happens when:
478
+ * - setting the snapshot.
479
+ * - parsing a batch of operations.
480
+ * - deleting an operation.
481
+ *
482
+ * @param index The index of the operation to run to.
483
+ * @param lastOperation The operation to use in place of the existing
484
+ * operation in the snapshot at index. This is used when previewing
485
+ * the edited operation from within the operation editor.
486
+ */
487
+ runTo(index, lastOperation) {
488
+ return new Promise((resolve, reject) => {
489
+ if (this.busy) {
490
+ return reject('Busy');
491
+ }
492
+ console.log('run to: ' + index);
493
+ // get the snapshot (=text and ops) to run operations on
494
+ const snapshot = this.getSnapshot();
495
+ if (!snapshot.text) {
496
+ return reject('No snapshot text');
497
+ }
498
+ // remove from the snapshot the operations past the specified index,
499
+ // also replacing the last operation when this was received as a parameter
500
+ snapshot.operations = [...this.operations.value];
501
+ if (lastOperation) {
502
+ snapshot.operations = snapshot.operations.slice(0, index);
503
+ snapshot.operations.push(lastOperation);
504
+ }
505
+ else {
506
+ snapshot.operations = snapshot.operations.slice(0, index + 1);
507
+ }
508
+ // run the operations
509
+ this.busy = true;
510
+ this.initialStepIndex = index;
511
+ this._api
512
+ .runOperations(snapshot.text, snapshot.operations)
513
+ .subscribe({
514
+ next: (wrapper) => {
515
+ // handle operation (non-fatal) error or result
516
+ if (wrapper.error) {
517
+ this._snackbar.open(wrapper.error, 'OK');
518
+ reject(wrapper.error);
519
+ return;
520
+ }
521
+ // extract the IDs from the last operation's diplomatics and filter
522
+ // them so that only the ones ending with _t are kept. By convention,
523
+ // all the IDs ending with this suffix are to be made invisible at
524
+ // their first rendition (opacity=0). An animation will then make
525
+ // them visible.
526
+ const lastOp = snapshot.operations[snapshot.operations.length - 1];
527
+ this._transparentIds = this.getTransparentIds(lastOp.diplomatics?.g);
528
+ // update the view data
529
+ this.setViewData(snapshot, lastOp.outputTag);
530
+ // return the result
531
+ resolve(wrapper.result);
532
+ },
533
+ error: (error) => {
534
+ console.error(error);
535
+ this._transparentIds = undefined;
536
+ this._snackbar.open('Error running operations', 'OK');
537
+ reject(error);
538
+ },
539
+ complete: () => {
540
+ this.busy = false;
541
+ },
542
+ });
543
+ });
544
+ }
545
+ /**
546
+ * Run the operations up to the last operation if any, updating the
547
+ * execution result. The execution result is always the result from
548
+ * all the operations, as users must be able to browse across its steps.
549
+ */
550
+ async runToLast() {
551
+ if (this.operations.value.length) {
552
+ this.result = await this.runTo(this.operations.value.length - 1);
553
+ }
554
+ else {
555
+ this.setViewData();
556
+ this.result = undefined;
557
+ }
558
+ }
559
+ /**
560
+ * Update the snapshot view by running the operations up to the
561
+ * currently edited operation if any.
562
+ *
563
+ * @param operation The operation being previewed.
564
+ */
333
565
  onOperationPreview(operation) {
334
- if (this._previewing) {
566
+ // no multiple previews or previewing a new operation
567
+ if (this._previewing || this.editedOpIndex < 0) {
335
568
  return;
336
569
  }
337
570
  this._previewing = true;
338
571
  setTimeout(() => {
339
- this.updateViewData(operation);
572
+ this.runTo(this.editedOpIndex, operation);
340
573
  this._previewing = false;
341
574
  }, 0);
342
575
  }
343
- runTo(index) {
344
- const snapshot = this.getSnapshot();
345
- if (!snapshot.text) {
346
- return;
347
- }
348
- const operations = this.operations.value.slice(0, index + 1);
349
- this.busy = true;
350
- this._api.runOperations(snapshot.text, operations).subscribe({
351
- next: (wrapper) => {
352
- if (wrapper.error) {
353
- this._snackbar.open(wrapper.error, 'OK');
354
- return;
355
- }
356
- this.result = wrapper.result;
357
- // TODO update view
358
- // this.updateViewData(
359
- // undefined,
360
- // snapshot,
361
- // this.operations.value[index].outputTag
362
- // );
363
- },
364
- error: (error) => {
365
- console.error(error);
366
- this._snackbar.open('Error running operations', 'OK');
367
- },
368
- complete: () => {
369
- this.busy = false;
370
- },
371
- });
372
- }
373
- onStepPick(step) {
576
+ // #endregion
577
+ /**
578
+ * Build the snapshot at the specified execution step. This implies adding
579
+ * the nodes introduced by the operations up to the specified step, and
580
+ * updating the features of the nodes introduced by the specified step.
581
+ * Also, the operations after the specified step are removed from the
582
+ * snapshot.
583
+ *
584
+ * @param step The step to build the snapshot at.
585
+ * @returns The snapshot at the specified step.
586
+ */
587
+ buildSnapshotAtStep(step) {
588
+ // no result, nothing to do
374
589
  if (!this.result) {
375
- return;
590
+ return null;
376
591
  }
377
- this.resultOperationId = step.operation.id;
378
- // get snapshot for preview by updating its text and
379
- // removing operations past the last executed
380
- const stepNodes = this.result.taggedNodes[step.outputTag];
381
- // we always display the original base text, just updating
382
- // its nodes features to those of the picked step, and adding
383
- // new nodes if they are not in the original text. It is assumed
384
- // that these new nodes have a manually set position.
592
+ // get the currently edited snapshot
385
593
  const snapshot = this.getSnapshot();
386
- const nodes = deepCopy(snapshot.text);
594
+ // find the max ID in snapshot.text (=original) nodes,
595
+ // so that any ID greater than it belongs to new nodes
596
+ const maxOriginalId = snapshot.text.reduce((max, n) => Math.max(max, n.id), 0);
597
+ // get a copy of the nodes of the snapshot base text
598
+ const snapNodes = deepCopy(snapshot.text);
599
+ // get the nodes of the picked step
600
+ const stepNodes = this.result.taggedNodes[step.outputTag];
601
+ // update the snapshot text nodes copies with the picked step nodes,
602
+ // by updating their features if existing, or adding new nodes if not.
387
603
  for (const sn of stepNodes) {
388
- const i = nodes.findIndex((n) => n.id === sn.id);
604
+ // find the node in the original snapshot text
605
+ const i = snapNodes.findIndex((n) => n.id === sn.id);
606
+ // if found, update its features, else add it
389
607
  if (i > -1) {
390
- nodes[i].features = sn.features;
608
+ snapNodes[i].features = sn.features;
391
609
  }
392
610
  else {
393
- nodes.push(sn);
611
+ // new node: we assume that it has a manually set position (x,y features)
612
+ snapNodes.push(sn);
394
613
  }
395
614
  }
396
- // find max ID in snapshot.text (=original) nodes,
397
- // so that any ID greater than it belongs to new nodes
398
- const maxOriginalId = snapshot.text.reduce((max, n) => Math.max(max, n.id), 0);
399
- // append nodes introduced before the picked step if any
615
+ // append nodes introduced *before* the picked step, if any
400
616
  // (those introduced by the picked step have already been added)
401
617
  const stepIndex = this.result.steps.indexOf(step);
402
618
  for (let i = 0; i < stepIndex; i++) {
403
619
  const stepNodes = this.result.taggedNodes[this.result.steps[i].outputTag];
404
620
  for (const n of stepNodes) {
405
- if (n.id > maxOriginalId && !nodes.find((x) => x.id === n.id)) {
406
- nodes.push(n);
621
+ if (n.id > maxOriginalId && !snapNodes.find((x) => x.id === n.id)) {
622
+ snapNodes.push(n);
407
623
  }
408
624
  }
409
625
  }
410
- snapshot.text = nodes;
626
+ // update the snapshot text nodes
627
+ snapshot.text = snapNodes;
628
+ // remove from the snapshot the operations after the picked step
411
629
  const i = snapshot.operations.findIndex((o) => o.id === step.operation.id);
412
630
  if (i > -1) {
413
631
  snapshot.operations = snapshot.operations.slice(0, i + 1);
414
632
  }
415
- this.updateViewData(undefined, snapshot, step.outputTag);
633
+ return snapshot;
634
+ }
635
+ playTimeline(tl) {
636
+ const shadowRoot = this.snapshotView?.nativeElement?.shadowRoot;
637
+ if (tl && shadowRoot) {
638
+ console.log('play timeline', tl);
639
+ this._renderer?.playTimeline(tl, undefined, shadowRoot);
640
+ }
641
+ else {
642
+ if (!this.snapshotView) {
643
+ console.warn('no snapshotView for timeline');
644
+ }
645
+ else {
646
+ console.warn('no shadowRoot for timeline');
647
+ }
648
+ }
416
649
  }
650
+ /**
651
+ * Handle the event fired by the chain result view to pick a step.
652
+ *
653
+ * @param step The step to pick.
654
+ */
655
+ async onStepPick(step) {
656
+ if (!this.result || this._stepPickFrozen) {
657
+ return;
658
+ }
659
+ // get the currently edited snapshot
660
+ const snapshot = this.buildSnapshotAtStep(step);
661
+ // update result operation ID
662
+ this.resultOperationId = step.operation.id;
663
+ // update transparent IDs
664
+ this._transparentIds = this.getTransparentIds(step.operation.diplomatics?.g);
665
+ // update view data
666
+ this.setViewData(snapshot, step.outputTag);
667
+ // play animation if any
668
+ const tl = this.timelines.value.find((t) => t.tag === step.outputTag);
669
+ if (tl) {
670
+ this._stepPickFrozen = true;
671
+ // find the index of the picked step in the snapshot operations
672
+ const i = snapshot.operations.findIndex((o) => o.id === step.operation.id);
673
+ // run the operations up to the picked step as we need to hide
674
+ // those visuals to be revealed by the animation
675
+ await this.runTo(i);
676
+ this._stepPickFrozen = false;
677
+ // play the timeline
678
+ setTimeout(() => {
679
+ this.playTimeline(tl);
680
+ }, 0);
681
+ }
682
+ }
683
+ /**
684
+ * Handle the event fired by a visual in the snapshot view.
685
+ *
686
+ * @param event The event.
687
+ */
417
688
  onVisualEvent(event) {
418
689
  const d = event.detail;
419
690
  if (d.event.type === 'mouseover') {
420
691
  const visual = d.source;
421
692
  const sb = [];
693
+ // id (type)
422
694
  sb.push('#' + visual.id);
423
695
  sb.push(` (${visual.type})`);
696
+ // : label and features
424
697
  if (visual.data?.label) {
425
698
  sb.push(': ');
426
699
  sb.push(visual.data.label);
427
700
  }
428
701
  if (visual.data?.features?.length) {
702
+ sb.push(' ');
429
703
  sb.push(visual.data.features
430
704
  .map((f) => `${f.name}=${f.value}`)
431
705
  .join('\n'));
432
706
  }
707
+ // (@x,y width×height)
708
+ sb.push(` (@${visual.x.toFixed(1)},${visual.y.toFixed(1)} ` +
709
+ `◻${visual.width.toFixed(1)}×${visual.height.toFixed(1)})`);
433
710
  this.visualInfo = sb.join('');
434
711
  }
435
712
  }
713
+ /**
714
+ * Handle the change of line heights by updating the form control.
715
+ *
716
+ * @param heights The line heights.
717
+ */
436
718
  onHeightsChange(heights) {
437
719
  this.lnHeights.setValue(heights || null);
438
720
  this.lnHeights.markAsDirty();
439
721
  this.lnHeights.updateValueAndValidity();
440
722
  }
723
+ /**
724
+ * Handle the change of timelines by updating the form control.
725
+ *
726
+ * @param timelines The timelines.
727
+ */
728
+ onTimelinesChange(timelines) {
729
+ this.timelines.setValue(timelines);
730
+ this.timelines.markAsDirty();
731
+ this.timelines.updateValueAndValidity();
732
+ }
733
+ /**
734
+ * Emit the cancel event for this snapshot edit.
735
+ */
441
736
  close() {
442
737
  this.snapshotCancel.emit();
443
738
  }
739
+ /**
740
+ * Handle the render event from the snapshot view to get a reference
741
+ * to the renderer.
742
+ *
743
+ * @param event The rendition event.
744
+ */
444
745
  onSnapshotRender(event) {
445
746
  this._renderer = event.detail.renderer;
446
747
  }
748
+ /**
749
+ * Toggle rulers in the snapshot view.
750
+ */
447
751
  toggleRulers() {
448
752
  if (!this._renderer) {
449
753
  return;
450
754
  }
451
755
  this.rulers = this._renderer.toggleRulers();
452
756
  }
757
+ /**
758
+ * Get a snapshot from the form data.
759
+ *
760
+ * @returns New snapshot object.
761
+ */
453
762
  getSnapshot() {
454
763
  const snapshot = {
455
764
  size: {
@@ -472,7 +781,6 @@ export class SnapshotEditorComponent {
472
781
  },
473
782
  operations: [...this.operations.value],
474
783
  opStyle: this.opStyle.value || undefined,
475
- timelines: {}, // TODO
476
784
  };
477
785
  // image
478
786
  if (this.imageUrl.value) {
@@ -504,11 +812,10 @@ export class SnapshotEditorComponent {
504
812
  }
505
813
  return snapshot;
506
814
  }
507
- onTimelinesChange(timelines) {
508
- this.timelines.setValue(timelines);
509
- this.timelines.markAsDirty();
510
- this.timelines.updateValueAndValidity();
511
- }
815
+ /**
816
+ * Get a snapshot from the form data and emit it
817
+ * in the snapshot change event.
818
+ */
512
819
  save() {
513
820
  if (this.form.invalid || this.noSave) {
514
821
  return;
@@ -516,10 +823,10 @@ export class SnapshotEditorComponent {
516
823
  this._snapshot = this.getSnapshot();
517
824
  this.snapshotChange.emit(this._snapshot);
518
825
  }
519
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.6", ngImport: i0, type: SnapshotEditorComponent, deps: [{ token: i1.FormBuilder }, { token: i2.GveApiService }, { token: i3.DialogService }, { token: i4.MatSnackBar }], target: i0.ɵɵFactoryTarget.Component }); }
520
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.6", type: SnapshotEditorComponent, isStandalone: true, selector: "gve-snapshot-editor", inputs: { snapshot: "snapshot", batchOps: "batchOps", noSave: "noSave" }, outputs: { snapshotChange: "snapshotChange", snapshotCancel: "snapshotCancel" }, ngImport: i0, template: "<form [formGroup]=\"form\" (submit)=\"save()\">\r\n <mat-tab-group>\r\n <!-- text -->\r\n <mat-tab label=\"text\">\r\n <mat-expansion-panel [expanded]=\"!baseText.value\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title> base text </mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <gve-base-text-editor\r\n [text]=\"baseText.value\"\r\n (textChange)=\"onTextChange($event)\"\r\n />\r\n <fieldset>\r\n <div class=\"form-row\">\r\n <!-- offsetX -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>X offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"offsetX\"\r\n placeholder=\"X offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- offsetY -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>Y offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"offsetY\"\r\n placeholder=\"Y offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- lineHeightOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>ln h-offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"lineHeightOffset\"\r\n placeholder=\"ln h-offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- charSpacingOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>char spacing</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"charSpacingOffset\"\r\n placeholder=\"char spacing\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- spcWidthOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>spc w-offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"spcWidthOffset\"\r\n placeholder=\"spc w-offset\"\r\n />\r\n </mat-form-field>\r\n <!-- minLineHeights -->\r\n <div>\r\n <gve-ln-heights-editor\r\n [lineCount]=\"lineCount\"\r\n [heights]=\"lnHeights.value\"\r\n (heightsChange)=\"onHeightsChange($event)\"\r\n />\r\n </div>\r\n </div>\r\n <!-- textStyle -->\r\n <div>\r\n <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n <mat-label>text style</mat-label>\r\n <textarea\r\n matInput\r\n [formControl]=\"textStyle\"\r\n placeholder=\"text style\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n </fieldset>\r\n </mat-expansion-panel>\r\n </mat-tab>\r\n\r\n <!-- operations -->\r\n <mat-tab label=\"operations\">\r\n <div id=\"snapshot-container\">\r\n <div id=\"general\">\r\n <div class=\"form-row\">\r\n <!-- width -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>width</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"width\"\r\n placeholder=\"width\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- height -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"height\"\r\n placeholder=\"height\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- add op -->\r\n <button\r\n type=\"button\"\r\n mat-flat-button\r\n color=\"primary\"\r\n class=\"mat-primary right\"\r\n [disabled]=\"!baseText.value\"\r\n (click)=\"editNewOperation()\"\r\n >\r\n <mat-icon>add_circle</mat-icon> operation\r\n </button>\r\n </div>\r\n </div>\r\n\r\n <!-- ops -->\r\n <div id=\"ops\">\r\n <!-- operations list -->\r\n @if (operations.value.length) {\r\n <table id=\"list\">\r\n <thead>\r\n <tr>\r\n <th></th>\r\n <th>ID</th>\r\n <th>type</th>\r\n <th>at</th>\r\n <th>run</th>\r\n <th>value</th>\r\n <th>itag</th>\r\n <th>otag</th>\r\n <th>gid</th>\r\n <th>feats</th>\r\n </tr>\r\n </thead>\r\n <tbody>\r\n @for (operation of operations.value; track operation.id; let\r\n index=$index) {\r\n <tr\r\n [ngClass]=\"{ selected: operation.id === resultOperationId }\"\r\n [ngClass]=\"{ edited: operation.id === editedOp?.id }\"\r\n >\r\n <td class=\"fit-width\">\r\n <!-- edit -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n class=\"mat-primary\"\r\n (click)=\"editOperation(index)\"\r\n matTooltip=\"Edit operation\"\r\n >\r\n <mat-icon>edit</mat-icon>\r\n </button>\r\n <!-- delete -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"warn\"\r\n (click)=\"deleteOperation(index)\"\r\n matTooltip=\"Delete operation\"\r\n >\r\n <mat-icon class=\"mat-warn\">delete</mat-icon>\r\n </button>\r\n <!-- run to -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"accent\"\r\n (click)=\"runTo(index)\"\r\n matTooltip=\"Run to this operation\"\r\n >\r\n <mat-icon class=\"mat-accent\">subscriptions</mat-icon>\r\n </button>\r\n </td>\r\n <td>{{ operation.id }}</td>\r\n <td>{{ operation.type | flatLookup : opTypeMap }}</td>\r\n <td>{{ operation.at }}</td>\r\n <td>{{ operation.run }}</td>\r\n <td>{{ operation.value }}</td>\r\n <td>{{ operation.inputTag }}</td>\r\n <td>{{ operation.outputTag }}</td>\r\n <td>{{ operation.groupId }}</td>\r\n <td>{{ operation.features?.length }}</td>\r\n </tr>\r\n }\r\n </tbody>\r\n </table>\r\n }\r\n\r\n <!-- operation editor -->\r\n <mat-expansion-panel [expanded]=\"editedOp\" [disabled]=\"!editedOp\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title>operation {{ editedOp?.id }}</mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <fieldset>\r\n <gve-chain-operation-editor\r\n [operation]=\"editedOp\"\r\n (operationCancel)=\"onOperationCancel()\"\r\n (operationChange)=\"onOperationChange($event)\"\r\n (operationPreview)=\"onOperationPreview($event)\"\r\n />\r\n </fieldset>\r\n </mat-expansion-panel>\r\n\r\n <div>\r\n <mat-progress-bar mode=\"indeterminate\" *ngIf=\"busy\" />\r\n </div>\r\n\r\n <!-- opStyle -->\r\n <div id=\"opStyle\">\r\n <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n <mat-label>operations style</mat-label>\r\n <textarea\r\n matInput\r\n [formControl]=\"opStyle\"\r\n placeholder=\"operations style\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- batch ops -->\r\n <div>\r\n <mat-expansion-panel>\r\n <mat-expansion-panel-header>\r\n <mat-panel-title>batch operations</mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <fieldset>\r\n <div id=\"batch-container\">\r\n <div id=\"batch-input\">\r\n <div>\r\n <mat-form-field class=\"full-width\">\r\n <mat-label>operations</mat-label>\r\n <textarea\r\n class=\"code\"\r\n matInput\r\n [formControl]=\"inputOps\"\r\n placeholder=\"operations\"\r\n rows=\"8\"\r\n spellcheck=\"false\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n <div class=\"form-row\">\r\n <button\r\n type=\"button\"\r\n color=\"warn\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Remove all the operations\"\r\n [disabled]=\"!operations.value.length || busy\"\r\n (click)=\"clearOperations()\"\r\n >\r\n clear\r\n </button>\r\n <button\r\n type=\"button\"\r\n color=\"primary\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Parse text and add results to the operations\"\r\n [disabled]=\"!inputOps.value || busy\"\r\n (click)=\"parseOperations()\"\r\n >\r\n batch add\r\n </button>\r\n @if (parseError) {\r\n <span class=\"error\">{{ parseError }}</span>\r\n }\r\n </div>\r\n </div>\r\n <div id=\"batch-help\">\r\n <ul>\r\n <li>\r\n <strong>replace</strong>:\r\n <code>ATxRUN<strong>=</strong>\"VALUE\"</code>\r\n </li>\r\n <li>\r\n <strong>delete</strong>:\r\n <code>ATxRUN<strong>-</strong></code>\r\n </li>\r\n <li>\r\n <strong>add-before</strong>:\r\n <code>ATxRUN<strong>+[</strong>\"VALUE\"</code>\r\n </li>\r\n <li>\r\n <strong>add-after</strong>:\r\n <code>ATxRUN<strong>+]</strong>\"VALUE\"</code>\r\n </li>\r\n <li>\r\n <strong>move-before</strong>:\r\n <code>ATxRUN<strong>&gt;[</strong>TO</code>\r\n </li>\r\n <li>\r\n <strong>move-after</strong>:\r\n <code>ATxRUN<strong>&gt;]</strong>TO</code>\r\n </li>\r\n <li>\r\n <strong>swap</strong>:\r\n <code>ATxRUN<strong>&lt;&gt;</strong>TOxRUN</code>\r\n </li>\r\n <li>\r\n <strong>annotate</strong>:\r\n <code>ATxRUN<strong>:</strong></code>\r\n </li>\r\n </ul>\r\n <p>For all the operations:</p>\r\n <ul>\r\n <li>\r\n prefix <code>(ITAG:OTAG)</code> to define input and/or\r\n output tags, separated by colon.\r\n </li>\r\n <li>\r\n prepend <code>&#x40;</code> to AT to use character\r\n indexes (0-N) rather than IDs.\r\n </li>\r\n <li>RUN (where applicable) defaults to 1.</li>\r\n <li>\r\n append features like <code>[NAME OPERATOR VALUE]</code>,\r\n separated by space, where:\r\n </li>\r\n <ol>\r\n <li>\r\n NAME is an arbitrary string. Prefixes:\r\n <code>*</code>=global features, <code>!</code>=remove\r\n feature (no value). Suffixes:\r\n <code>^</code>=short-lived (like\r\n <code>*version^=alpha</code>).\r\n </li>\r\n <li>\r\n OPERATOR is <code>=</code> (multiple),\r\n <code>:=</code> (single),\r\n <code>==</code> (first-single).\r\n </li>\r\n <li>\r\n VALUE is a string. If it includes spaces, wrap it in\r\n <code>\"\"</code>.\r\n </li>\r\n </ol>\r\n </ul>\r\n </div>\r\n </div>\r\n </fieldset>\r\n </mat-expansion-panel>\r\n </div>\r\n </div>\r\n\r\n <!-- result -->\r\n @if (result) {\r\n <fieldset id=\"result\">\r\n <legend>result</legend>\r\n <gve-chain-result-view\r\n [result]=\"result\"\r\n (stepPick)=\"onStepPick($event)\"\r\n />\r\n </fieldset>\r\n }\r\n </div>\r\n </mat-tab>\r\n\r\n <!-- image -->\r\n <mat-tab label=\"image\">\r\n <div id=\"image\">\r\n <div id=\"image-ctl\">\r\n <!-- url -->\r\n <mat-form-field class=\"long-text\">\r\n <mat-label>URL</mat-label>\r\n <input matInput [formControl]=\"imageUrl\" />\r\n </mat-form-field>\r\n <div class=\"form-row\">\r\n <!-- x -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>X</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"imageX\"\r\n placeholder=\"X\"\r\n />\r\n </mat-form-field>\r\n <!-- y -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>Y</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"imageY\"\r\n placeholder=\"Y\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n <div class=\"form-row\">\r\n <!-- width -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>width</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"imageWidth\"\r\n />\r\n </mat-form-field>\r\n <!-- height -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"imageHeight\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n <!-- defs -->\r\n <div>\r\n <mat-form-field class=\"long-text\">\r\n <mat-label>defs</mat-label>\r\n <textarea matInput [formControl]=\"defs\" rows=\"3\"></textarea>\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n <div id=\"image-view\">\r\n @if (imageUrl.value) {\r\n <img alt=\"background\" [src]=\"imageUrl.value\" width=\"600\" />\r\n }\r\n </div>\r\n </div>\r\n </mat-tab>\r\n\r\n <!-- timelines -->\r\n <mat-tab label=\"animation\">\r\n <gve-animation-timeline-set\r\n [tags]=\"opTags\"\r\n [elementIds]=\"opElementIds\"\r\n [timelines]=\"timelines.value\"\r\n (timelinesChange)=\"onTimelinesChange($event)\"\r\n />\r\n </mat-tab>\r\n </mat-tab-group>\r\n\r\n <!-- preview -->\r\n @if (snapshot) {\r\n <fieldset id=\"preview\">\r\n <legend class=\"button-row\">\r\n <span>{{ viewTitle }}</span>\r\n <button\r\n color=\"primary\"\r\n mat-icon-button\r\n type=\"button\"\r\n matTooltip=\"Refresh preview\"\r\n (click)=\"updateViewData()\"\r\n >\r\n <mat-icon class=\"mat-primary\">refresh</mat-icon>\r\n </button>\r\n </legend>\r\n <gve-snapshot-view\r\n [data]=\"viewData\"\r\n (snapshotRender)=\"onSnapshotRender($any($event))\"\r\n (visualEvent)=\"onVisualEvent($any($event))\"\r\n />\r\n <div>\r\n <mat-button-toggle\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n matTooltip=\"Toggle rulers\"\r\n [checked]=\"rulers\"\r\n (change)=\"toggleRulers()\"\r\n >\r\n <mat-icon class=\"mat-primary\">straighten</mat-icon>\r\n </mat-button-toggle>\r\n </div>\r\n @if (visualInfo) {\r\n <div id=\"visual-info\">{{ visualInfo }}</div>\r\n }\r\n </fieldset>\r\n }\r\n\r\n <!--buttons -->\r\n <div class=\"form-row-center\">\r\n <button\r\n type=\"button\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Discard changes\"\r\n (click)=\"close()\"\r\n >\r\n <mat-icon>clear</mat-icon>\r\n close\r\n </button>\r\n @if (!noSave) {\r\n <button\r\n type=\"submit\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Save changes\"\r\n [disabled]=\"form.invalid || form.pristine\"\r\n >\r\n <mat-icon>check_circle</mat-icon>\r\n save\r\n </button>\r\n }\r\n </div>\r\n</form>\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row-center{display:flex;gap:8px;align-items:center;justify-content:center;flex-wrap:wrap}.form-row,.form-row-center *{flex:0 0 auto}.form-row *.right{margin-left:auto}.button-row{display:flex;align-items:center;flex-wrap:wrap}.button-row *{flex:0 0 auto}.long-text{width:100%;max-width:800px}#text-range{margin:8px;border:1px solid silver;border-radius:6px;padding:6px}mat-expansion-panel{margin:8px 0}div#visual-info{font-size:95%;color:#909090;margin:8px}div#batch-container{display:grid;gap:16px;grid-template-rows:auto;grid-template-columns:1fr auto;grid-template-areas:\"batch-input batch-help\"}div#batch-input{grid-area:batch-input}div#batch-help{grid-area:batch-help}div#batch-help strong{color:#ff8c00}#list{margin:8px 0}#opStyle{margin-top:8px}table{width:100%;border-collapse:collapse}th{color:#909090;font-weight:400;text-align:left;background-color:#e1e0e0}th,td{padding:4px;border-bottom:1px solid silver}tbody tr:nth-child(2n){background-color:#e8e8e8}td.fit-width{width:1px;white-space:nowrap}tr.selected{background-color:#c8d9eb}tr.edited{background-color:#f6f6e4}fieldset{border:1px solid silver;border-radius:6px;padding:8px;margin:8px 0}legend{color:#909090}.error{color:red}.input-nr{width:6em}.full-width{width:100%}.code{font-family:Courier New,Courier,monospace}gve-animation-timeline-set{margin:8px 0}div#image{display:grid;gap:8px;grid-template-rows:auto;grid-template-columns:1fr auto;grid-template-areas:\"image-ctl image-view\"}div#image-ctl{grid-area:image-ctl}div#image-view{grid-area:image-view}@media only screen and (max-width: 959px){div#batch-container{grid-template-columns:1fr;grid-template-areas:\"batch-input\" \"batch-help\"}div#image{grid-template-columns:1fr;grid-template-areas:\"image-ctl\" \"image-view\"}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i5.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.MinValidator, selector: "input[type=number][min][formControlName],input[type=number][min][formControl],input[type=number][min][ngModel]", inputs: ["min"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i6.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i6.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatButtonToggleModule }, { kind: "component", type: i7.MatButtonToggle, selector: "mat-button-toggle", inputs: ["aria-label", "aria-labelledby", "id", "name", "value", "tabIndex", "disableRipple", "appearance", "checked", "disabled"], outputs: ["change"], exportAs: ["matButtonToggle"] }, { kind: "ngmodule", type: MatCheckboxModule }, { kind: "ngmodule", type: MatExpansionModule }, { kind: "component", type: i8.MatExpansionPanel, selector: "mat-expansion-panel", inputs: ["hideToggle", "togglePosition"], outputs: ["afterExpand", "afterCollapse"], exportAs: ["matExpansionPanel"] }, { kind: "component", type: i8.MatExpansionPanelHeader, selector: "mat-expansion-panel-header", inputs: ["expandedHeight", "collapsedHeight", "tabIndex"] }, { kind: "directive", type: i8.MatExpansionPanelTitle, selector: "mat-panel-title" }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i9.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i9.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i10.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i11.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "ngmodule", type: MatProgressBarModule }, { kind: "component", type: i12.MatProgressBar, selector: "mat-progress-bar", inputs: ["color", "value", "bufferValue", "mode"], outputs: ["animationEnd"], exportAs: ["matProgressBar"] }, { kind: "ngmodule", type: MatSnackBarModule }, { kind: "ngmodule", type: MatTabsModule }, { kind: "component", type: i13.MatTab, selector: "mat-tab", inputs: ["disabled", "label", "aria-label", "aria-labelledby", "labelClass", "bodyClass"], exportAs: ["matTab"] }, { kind: "component", type: i13.MatTabGroup, selector: "mat-tab-group", inputs: ["color", "fitInkBarToContent", "mat-stretch-tabs", "dynamicHeight", "selectedIndex", "headerPosition", "animationDuration", "contentTabIndex", "disablePagination", "disableRipple", "preserveContent", "backgroundColor"], outputs: ["selectedIndexChange", "focusChange", "animationDone", "selectedTabChange"], exportAs: ["matTabGroup"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i14.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: NgToolsModule }, { kind: "pipe", type: i15.FlatLookupPipe, name: "flatLookup" }, { kind: "component", type: BaseTextEditorComponent, selector: "gve-base-text-editor", inputs: ["text"], outputs: ["textChange"] }, { kind: "component", type: ChainOperationEditorComponent, selector: "gve-chain-operation-editor", inputs: ["operation", "snapshot", "hidePreview"], outputs: ["operationChange", "operationPreview", "operationCancel"] }, { kind: "component", type: ChainResultViewComponent, selector: "gve-chain-result-view", inputs: ["result"], outputs: ["stepPick"] }, { kind: "component", type: LnHeightsEditorComponent, selector: "gve-ln-heights-editor", inputs: ["lineCount", "heights"], outputs: ["heightsChange"] }, { kind: "component", type: AnimationTimelineSetComponent, selector: "gve-animation-timeline-set", inputs: ["timelines", "elementIds", "tags"], outputs: ["timelinesChange", "cancel"] }] }); }
826
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: SnapshotEditorComponent, deps: [{ token: i1.FormBuilder }, { token: i2.GveApiService }, { token: i3.MatDialog }, { token: i4.DialogService }, { token: i5.MatSnackBar }], target: i0.ɵɵFactoryTarget.Component }); }
827
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.12", type: SnapshotEditorComponent, isStandalone: true, selector: "gve-snapshot-editor", inputs: { snapshot: "snapshot", batchOps: "batchOps", noSave: "noSave", debug: "debug" }, outputs: { snapshotChange: "snapshotChange", snapshotCancel: "snapshotCancel" }, viewQueries: [{ propertyName: "snapshotView", first: true, predicate: ["snapshotView"], descendants: true }], ngImport: i0, template: "<form [formGroup]=\"form\" (submit)=\"save()\">\r\n <mat-tab-group>\r\n <!-- text -->\r\n <mat-tab>\r\n <ng-template mat-tab-label>\r\n <mat-icon>article</mat-icon> <span class=\"label\">text</span>\r\n </ng-template>\r\n <mat-expansion-panel [expanded]=\"!baseText.value\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title> base text </mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <gve-base-text-editor\r\n [text]=\"baseText.value\"\r\n (textChange)=\"onTextChange($event)\"\r\n />\r\n <fieldset>\r\n <div class=\"form-row\">\r\n <!-- offsetX -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>X offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"offsetX\"\r\n placeholder=\"X offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- offsetY -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>Y offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"offsetY\"\r\n placeholder=\"Y offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- lineHeightOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>ln h-offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"lineHeightOffset\"\r\n placeholder=\"ln h-offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- charSpacingOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>char spacing</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"charSpacingOffset\"\r\n placeholder=\"char spacing\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- spcWidthOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>spc w-offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"spcWidthOffset\"\r\n placeholder=\"spc w-offset\"\r\n />\r\n </mat-form-field>\r\n <!-- minLineHeights -->\r\n <div class=\"boxed\">\r\n <gve-ln-heights-editor\r\n [lineCount]=\"lineCount\"\r\n [heights]=\"lnHeights.value\"\r\n (heightsChange)=\"onHeightsChange($event)\"\r\n />\r\n </div>\r\n </div>\r\n <!-- textStyle -->\r\n <div>\r\n <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n <mat-label>text style</mat-label>\r\n <textarea\r\n matInput\r\n [formControl]=\"textStyle\"\r\n placeholder=\"text style\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n </fieldset>\r\n </mat-expansion-panel>\r\n </mat-tab>\r\n\r\n <!-- operations -->\r\n <mat-tab>\r\n <ng-template mat-tab-label>\r\n <mat-icon>edit</mat-icon> <span class=\"label\">operations</span>\r\n </ng-template>\r\n\r\n <div id=\"snapshot-container\">\r\n <div id=\"general\">\r\n <div class=\"form-row\">\r\n <!-- width -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>width</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"width\"\r\n placeholder=\"width\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- height -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"height\"\r\n placeholder=\"height\"\r\n />\r\n </mat-form-field>\r\n\r\n <div class=\"form-row right\">\r\n <!-- remove ops -->\r\n <button\r\n type=\"button\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Remove all the operations\"\r\n [disabled]=\"!operations.value.length || busy\"\r\n (click)=\"clearOperations()\"\r\n >\r\n <mat-icon>delete_forever</mat-icon> clear\r\n </button>\r\n\r\n <!-- batch add ops -->\r\n <button\r\n type=\"button\"\r\n mat-flat-button\r\n matTooltip=\"Add a batch of operations\"\r\n class=\"mat-primary\"\r\n [disabled]=\"busy\"\r\n (click)=\"parseOperations()\"\r\n >\r\n <mat-icon>post_add</mat-icon> batch\r\n </button>\r\n\r\n <!-- add op -->\r\n <button\r\n type=\"button\"\r\n mat-flat-button\r\n class=\"mat-primary\"\r\n [disabled]=\"!baseText.value\"\r\n (click)=\"editNewOperation()\"\r\n >\r\n <mat-icon>add_circle</mat-icon> operation\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- ops -->\r\n <div id=\"ops\">\r\n <!-- operations list -->\r\n @if (operations.value.length) {\r\n <table id=\"list\">\r\n <thead>\r\n <tr>\r\n <th></th>\r\n <th>ID</th>\r\n <th>type</th>\r\n <th>at</th>\r\n <th>run</th>\r\n <th>value</th>\r\n <th>itag</th>\r\n <th>otag</th>\r\n <th>gid</th>\r\n <th>feats</th>\r\n </tr>\r\n </thead>\r\n <tbody>\r\n @for (operation of operations.value; track operation.id; let\r\n index=$index) {\r\n <tr\r\n [ngClass]=\"{ selected: operation.id === resultOperationId }\"\r\n [ngClass]=\"{ edited: operation.id === editedOp?.id }\"\r\n >\r\n <td class=\"fit-width\">\r\n <!-- edit -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n class=\"mat-primary\"\r\n (click)=\"editOperation(index)\"\r\n matTooltip=\"Edit operation\"\r\n >\r\n <mat-icon>edit</mat-icon>\r\n </button>\r\n <!-- delete -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n class=\"mat-warn\"\r\n (click)=\"deleteOperation(index)\"\r\n matTooltip=\"Delete operation\"\r\n >\r\n <mat-icon class=\"mat-warn\">delete</mat-icon>\r\n </button>\r\n <!-- run to -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n class=\"mat-accent\"\r\n (click)=\"runTo(index)\"\r\n matTooltip=\"Run to this operation\"\r\n >\r\n <mat-icon class=\"mat-accent\">subscriptions</mat-icon>\r\n </button>\r\n </td>\r\n <td>{{ operation.id }}</td>\r\n <td>{{ operation.type | flatLookup : opTypeMap }}</td>\r\n <td>{{ operation.at }}</td>\r\n <td>{{ operation.run }}</td>\r\n <td>{{ operation.value }}</td>\r\n <td>{{ operation.inputTag }}</td>\r\n <td>{{ operation.outputTag }}</td>\r\n <td>{{ operation.groupId }}</td>\r\n <td>{{ operation.features?.length }}</td>\r\n </tr>\r\n }\r\n </tbody>\r\n </table>\r\n }\r\n\r\n <!-- operation editor -->\r\n <mat-expansion-panel [expanded]=\"editedOp\" [disabled]=\"!editedOp\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title>operation {{ editedOp?.id }}</mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <fieldset>\r\n <gve-chain-operation-editor\r\n [hidePreview]=\"editedOpIndex === -1\"\r\n [operation]=\"editedOp\"\r\n (operationCancel)=\"onOperationCancel()\"\r\n (operationChange)=\"onOperationChange($event)\"\r\n (operationPreview)=\"onOperationPreview($event)\"\r\n />\r\n </fieldset>\r\n </mat-expansion-panel>\r\n\r\n <!-- opStyle -->\r\n <div id=\"opStyle\">\r\n <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n <mat-label>operations style</mat-label>\r\n <textarea\r\n matInput\r\n [formControl]=\"opStyle\"\r\n placeholder=\"operations style\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n </div>\r\n </mat-tab>\r\n\r\n <!-- image -->\r\n <mat-tab>\r\n <ng-template mat-tab-label>\r\n <mat-icon>image</mat-icon> <span class=\"label\">image</span>\r\n </ng-template>\r\n\r\n <div id=\"image\">\r\n <div id=\"image-ctl\">\r\n <!-- url -->\r\n <mat-form-field class=\"long-text\">\r\n <mat-label>URL</mat-label>\r\n <input matInput [formControl]=\"imageUrl\" />\r\n </mat-form-field>\r\n <div class=\"form-row\">\r\n <!-- x -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>X</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"imageX\"\r\n placeholder=\"X\"\r\n />\r\n </mat-form-field>\r\n <!-- y -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>Y</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"imageY\"\r\n placeholder=\"Y\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n <div class=\"form-row\">\r\n <!-- width -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>width</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"imageWidth\"\r\n />\r\n </mat-form-field>\r\n <!-- height -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"imageHeight\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n <!-- defs -->\r\n <div>\r\n <mat-form-field class=\"long-text\">\r\n <mat-label>defs</mat-label>\r\n <textarea matInput [formControl]=\"defs\" rows=\"3\"></textarea>\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n <div id=\"image-view\">\r\n @if (imageUrl.value) {\r\n <img alt=\"background\" [src]=\"imageUrl.value\" width=\"600\" />\r\n }\r\n </div>\r\n </div>\r\n </mat-tab>\r\n\r\n <!-- timelines -->\r\n <mat-tab>\r\n <ng-template mat-tab-label>\r\n <mat-icon>animation</mat-icon> <span class=\"label\">animation</span>\r\n </ng-template>\r\n\r\n <gve-animation-timeline-set\r\n [tags]=\"opTags\"\r\n [elementIds]=\"opElementIds\"\r\n [timelines]=\"timelines.value\"\r\n (timelinesChange)=\"onTimelinesChange($event)\"\r\n />\r\n </mat-tab>\r\n </mat-tab-group>\r\n\r\n <!-- progress -->\r\n <div>\r\n <mat-progress-bar mode=\"indeterminate\" *ngIf=\"busy\" />\r\n </div>\r\n\r\n <!-- result -->\r\n @if (result) {\r\n <fieldset id=\"result\">\r\n <legend>result</legend>\r\n <gve-chain-result-view\r\n [result]=\"result\"\r\n [initialStepIndex]=\"initialStepIndex\"\r\n (stepPick)=\"onStepPick($event)\"\r\n />\r\n </fieldset>\r\n }\r\n\r\n <!-- snapshot view -->\r\n <fieldset id=\"preview\">\r\n <legend class=\"button-row\">\r\n <span>{{ viewTitle }}</span>\r\n </legend>\r\n <!-- snapshot view -->\r\n <gve-snapshot-view\r\n #snapshotView\r\n [debug]=\"debug\"\r\n [data]=\"viewData\"\r\n (snapshotRender)=\"onSnapshotRender($any($event))\"\r\n (visualEvent)=\"onVisualEvent($any($event))\"\r\n />\r\n <div class=\"button-row\">\r\n <!-- toggle ruler -->\r\n <mat-button-toggle\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n matTooltip=\"Toggle rulers\"\r\n [checked]=\"rulers\"\r\n (change)=\"toggleRulers()\"\r\n >\r\n <mat-icon class=\"mat-primary\">straighten</mat-icon>\r\n </mat-button-toggle>\r\n <!-- run to last -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n class=\"primary\"\r\n matTooltip=\"Run all operations\"\r\n [disabled]=\"!operations.value.length\"\r\n (click)=\"runToLast()\"\r\n >\r\n <mat-icon>play_circle</mat-icon>\r\n </button>\r\n </div>\r\n @if (visualInfo) {\r\n <div id=\"visual-info\">{{ visualInfo }}</div>\r\n }\r\n </fieldset>\r\n\r\n <!--buttons -->\r\n <div class=\"form-row-center\">\r\n <button\r\n type=\"button\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Discard changes\"\r\n (click)=\"close()\"\r\n >\r\n <mat-icon>clear</mat-icon>\r\n close\r\n </button>\r\n @if (!noSave) {\r\n <button\r\n type=\"submit\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Save changes\"\r\n [disabled]=\"form.invalid\"\r\n >\r\n <mat-icon>check_circle</mat-icon>\r\n save\r\n </button>\r\n }\r\n </div>\r\n</form>\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row-center{display:flex;gap:8px;align-items:center;justify-content:center;flex-wrap:wrap}.form-row,.form-row-center *{flex:0 0 auto}.form-row .right{margin-left:auto}.button-row{display:flex;align-items:center;flex-wrap:wrap}.button-row *{flex:0 0 auto}.long-text{width:100%;max-width:800px}#text-range{margin:8px;border:1px solid silver;border-radius:6px;padding:6px}mat-expansion-panel{margin:8px 0}div#visual-info{font-size:95%;color:#909090;margin:8px}#list{margin:8px 0}#opStyle{margin-top:8px}table{width:100%;border-collapse:collapse}th{color:#909090;font-weight:400;text-align:left;background-color:#e1e0e0}th,td{padding:4px;border-bottom:1px solid silver}tbody tr:nth-child(2n){background-color:#e8e8e8}td.fit-width{width:1px;white-space:nowrap}tr.selected{background-color:#c8d9eb}tr.edited{background-color:#f6f6e4}fieldset{border:1px solid silver;border-radius:6px;padding:8px;margin:8px 0}legend{color:#909090}.error{color:red}.input-nr{width:6em}.full-width{width:100%}.code{font-family:Courier New,Courier,monospace}.boxed{border:1px solid silver;border-radius:6px;padding:8px;margin:8px 0}span.label{margin-left:8px}gve-animation-timeline-set{margin:8px 0}div#image{display:grid;gap:8px;grid-template-rows:auto;grid-template-columns:1fr auto;grid-template-areas:\"image-ctl image-view\"}div#image-ctl{grid-area:image-ctl}div#image-view{grid-area:image-view}@media only screen and (max-width: 959px){div#image{grid-template-columns:1fr;grid-template-areas:\"image-ctl\" \"image-view\"}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i6.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.MinValidator, selector: "input[type=number][min][formControlName],input[type=number][min][formControl],input[type=number][min][ngModel]", inputs: ["min"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i7.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i7.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatButtonToggleModule }, { kind: "component", type: i8.MatButtonToggle, selector: "mat-button-toggle", inputs: ["aria-label", "aria-labelledby", "id", "name", "value", "tabIndex", "disableRipple", "appearance", "checked", "disabled", "disabledInteractive"], outputs: ["change"], exportAs: ["matButtonToggle"] }, { kind: "ngmodule", type: MatCheckboxModule }, { kind: "ngmodule", type: MatExpansionModule }, { kind: "component", type: i9.MatExpansionPanel, selector: "mat-expansion-panel", inputs: ["hideToggle", "togglePosition"], outputs: ["afterExpand", "afterCollapse"], exportAs: ["matExpansionPanel"] }, { kind: "component", type: i9.MatExpansionPanelHeader, selector: "mat-expansion-panel-header", inputs: ["expandedHeight", "collapsedHeight", "tabIndex"] }, { kind: "directive", type: i9.MatExpansionPanelTitle, selector: "mat-panel-title" }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i10.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i10.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i11.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i12.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "ngmodule", type: MatProgressBarModule }, { kind: "component", type: i13.MatProgressBar, selector: "mat-progress-bar", inputs: ["color", "value", "bufferValue", "mode"], outputs: ["animationEnd"], exportAs: ["matProgressBar"] }, { kind: "ngmodule", type: MatSnackBarModule }, { kind: "ngmodule", type: MatTabsModule }, { kind: "directive", type: i14.MatTabLabel, selector: "[mat-tab-label], [matTabLabel]" }, { kind: "component", type: i14.MatTab, selector: "mat-tab", inputs: ["disabled", "label", "aria-label", "aria-labelledby", "labelClass", "bodyClass"], exportAs: ["matTab"] }, { kind: "component", type: i14.MatTabGroup, selector: "mat-tab-group", inputs: ["color", "fitInkBarToContent", "mat-stretch-tabs", "dynamicHeight", "selectedIndex", "headerPosition", "animationDuration", "contentTabIndex", "disablePagination", "disableRipple", "preserveContent", "backgroundColor", "aria-label", "aria-labelledby"], outputs: ["selectedIndexChange", "focusChange", "animationDone", "selectedTabChange"], exportAs: ["matTabGroup"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i15.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: NgToolsModule }, { kind: "pipe", type: i16.FlatLookupPipe, name: "flatLookup" }, { kind: "component", type: BaseTextEditorComponent, selector: "gve-base-text-editor", inputs: ["text"], outputs: ["textChange"] }, { kind: "component", type: ChainOperationEditorComponent, selector: "gve-chain-operation-editor", inputs: ["operation", "snapshot", "hidePreview"], outputs: ["operationChange", "operationPreview", "operationCancel"] }, { kind: "component", type: ChainResultViewComponent, selector: "gve-chain-result-view", inputs: ["result", "initialStepIndex"], outputs: ["stepPick"] }, { kind: "component", type: LnHeightsEditorComponent, selector: "gve-ln-heights-editor", inputs: ["lineCount", "heights"], outputs: ["heightsChange"] }, { kind: "component", type: AnimationTimelineSetComponent, selector: "gve-animation-timeline-set", inputs: ["timelines", "elementIds", "tags"], outputs: ["timelinesChange", "timelinesCancel"] }] }); }
521
828
  }
522
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.6", ngImport: i0, type: SnapshotEditorComponent, decorators: [{
829
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: SnapshotEditorComponent, decorators: [{
523
830
  type: Component,
524
831
  args: [{ selector: 'gve-snapshot-editor', standalone: true, imports: [
525
832
  CommonModule,
@@ -543,16 +850,21 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.6", ngImpor
543
850
  ChainResultViewComponent,
544
851
  LnHeightsEditorComponent,
545
852
  AnimationTimelineSetComponent,
546
- ], schemas: [CUSTOM_ELEMENTS_SCHEMA], template: "<form [formGroup]=\"form\" (submit)=\"save()\">\r\n <mat-tab-group>\r\n <!-- text -->\r\n <mat-tab label=\"text\">\r\n <mat-expansion-panel [expanded]=\"!baseText.value\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title> base text </mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <gve-base-text-editor\r\n [text]=\"baseText.value\"\r\n (textChange)=\"onTextChange($event)\"\r\n />\r\n <fieldset>\r\n <div class=\"form-row\">\r\n <!-- offsetX -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>X offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"offsetX\"\r\n placeholder=\"X offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- offsetY -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>Y offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"offsetY\"\r\n placeholder=\"Y offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- lineHeightOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>ln h-offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"lineHeightOffset\"\r\n placeholder=\"ln h-offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- charSpacingOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>char spacing</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"charSpacingOffset\"\r\n placeholder=\"char spacing\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- spcWidthOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>spc w-offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"spcWidthOffset\"\r\n placeholder=\"spc w-offset\"\r\n />\r\n </mat-form-field>\r\n <!-- minLineHeights -->\r\n <div>\r\n <gve-ln-heights-editor\r\n [lineCount]=\"lineCount\"\r\n [heights]=\"lnHeights.value\"\r\n (heightsChange)=\"onHeightsChange($event)\"\r\n />\r\n </div>\r\n </div>\r\n <!-- textStyle -->\r\n <div>\r\n <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n <mat-label>text style</mat-label>\r\n <textarea\r\n matInput\r\n [formControl]=\"textStyle\"\r\n placeholder=\"text style\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n </fieldset>\r\n </mat-expansion-panel>\r\n </mat-tab>\r\n\r\n <!-- operations -->\r\n <mat-tab label=\"operations\">\r\n <div id=\"snapshot-container\">\r\n <div id=\"general\">\r\n <div class=\"form-row\">\r\n <!-- width -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>width</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"width\"\r\n placeholder=\"width\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- height -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"height\"\r\n placeholder=\"height\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- add op -->\r\n <button\r\n type=\"button\"\r\n mat-flat-button\r\n color=\"primary\"\r\n class=\"mat-primary right\"\r\n [disabled]=\"!baseText.value\"\r\n (click)=\"editNewOperation()\"\r\n >\r\n <mat-icon>add_circle</mat-icon> operation\r\n </button>\r\n </div>\r\n </div>\r\n\r\n <!-- ops -->\r\n <div id=\"ops\">\r\n <!-- operations list -->\r\n @if (operations.value.length) {\r\n <table id=\"list\">\r\n <thead>\r\n <tr>\r\n <th></th>\r\n <th>ID</th>\r\n <th>type</th>\r\n <th>at</th>\r\n <th>run</th>\r\n <th>value</th>\r\n <th>itag</th>\r\n <th>otag</th>\r\n <th>gid</th>\r\n <th>feats</th>\r\n </tr>\r\n </thead>\r\n <tbody>\r\n @for (operation of operations.value; track operation.id; let\r\n index=$index) {\r\n <tr\r\n [ngClass]=\"{ selected: operation.id === resultOperationId }\"\r\n [ngClass]=\"{ edited: operation.id === editedOp?.id }\"\r\n >\r\n <td class=\"fit-width\">\r\n <!-- edit -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n class=\"mat-primary\"\r\n (click)=\"editOperation(index)\"\r\n matTooltip=\"Edit operation\"\r\n >\r\n <mat-icon>edit</mat-icon>\r\n </button>\r\n <!-- delete -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"warn\"\r\n (click)=\"deleteOperation(index)\"\r\n matTooltip=\"Delete operation\"\r\n >\r\n <mat-icon class=\"mat-warn\">delete</mat-icon>\r\n </button>\r\n <!-- run to -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"accent\"\r\n (click)=\"runTo(index)\"\r\n matTooltip=\"Run to this operation\"\r\n >\r\n <mat-icon class=\"mat-accent\">subscriptions</mat-icon>\r\n </button>\r\n </td>\r\n <td>{{ operation.id }}</td>\r\n <td>{{ operation.type | flatLookup : opTypeMap }}</td>\r\n <td>{{ operation.at }}</td>\r\n <td>{{ operation.run }}</td>\r\n <td>{{ operation.value }}</td>\r\n <td>{{ operation.inputTag }}</td>\r\n <td>{{ operation.outputTag }}</td>\r\n <td>{{ operation.groupId }}</td>\r\n <td>{{ operation.features?.length }}</td>\r\n </tr>\r\n }\r\n </tbody>\r\n </table>\r\n }\r\n\r\n <!-- operation editor -->\r\n <mat-expansion-panel [expanded]=\"editedOp\" [disabled]=\"!editedOp\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title>operation {{ editedOp?.id }}</mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <fieldset>\r\n <gve-chain-operation-editor\r\n [operation]=\"editedOp\"\r\n (operationCancel)=\"onOperationCancel()\"\r\n (operationChange)=\"onOperationChange($event)\"\r\n (operationPreview)=\"onOperationPreview($event)\"\r\n />\r\n </fieldset>\r\n </mat-expansion-panel>\r\n\r\n <div>\r\n <mat-progress-bar mode=\"indeterminate\" *ngIf=\"busy\" />\r\n </div>\r\n\r\n <!-- opStyle -->\r\n <div id=\"opStyle\">\r\n <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n <mat-label>operations style</mat-label>\r\n <textarea\r\n matInput\r\n [formControl]=\"opStyle\"\r\n placeholder=\"operations style\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- batch ops -->\r\n <div>\r\n <mat-expansion-panel>\r\n <mat-expansion-panel-header>\r\n <mat-panel-title>batch operations</mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <fieldset>\r\n <div id=\"batch-container\">\r\n <div id=\"batch-input\">\r\n <div>\r\n <mat-form-field class=\"full-width\">\r\n <mat-label>operations</mat-label>\r\n <textarea\r\n class=\"code\"\r\n matInput\r\n [formControl]=\"inputOps\"\r\n placeholder=\"operations\"\r\n rows=\"8\"\r\n spellcheck=\"false\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n <div class=\"form-row\">\r\n <button\r\n type=\"button\"\r\n color=\"warn\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Remove all the operations\"\r\n [disabled]=\"!operations.value.length || busy\"\r\n (click)=\"clearOperations()\"\r\n >\r\n clear\r\n </button>\r\n <button\r\n type=\"button\"\r\n color=\"primary\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Parse text and add results to the operations\"\r\n [disabled]=\"!inputOps.value || busy\"\r\n (click)=\"parseOperations()\"\r\n >\r\n batch add\r\n </button>\r\n @if (parseError) {\r\n <span class=\"error\">{{ parseError }}</span>\r\n }\r\n </div>\r\n </div>\r\n <div id=\"batch-help\">\r\n <ul>\r\n <li>\r\n <strong>replace</strong>:\r\n <code>ATxRUN<strong>=</strong>\"VALUE\"</code>\r\n </li>\r\n <li>\r\n <strong>delete</strong>:\r\n <code>ATxRUN<strong>-</strong></code>\r\n </li>\r\n <li>\r\n <strong>add-before</strong>:\r\n <code>ATxRUN<strong>+[</strong>\"VALUE\"</code>\r\n </li>\r\n <li>\r\n <strong>add-after</strong>:\r\n <code>ATxRUN<strong>+]</strong>\"VALUE\"</code>\r\n </li>\r\n <li>\r\n <strong>move-before</strong>:\r\n <code>ATxRUN<strong>&gt;[</strong>TO</code>\r\n </li>\r\n <li>\r\n <strong>move-after</strong>:\r\n <code>ATxRUN<strong>&gt;]</strong>TO</code>\r\n </li>\r\n <li>\r\n <strong>swap</strong>:\r\n <code>ATxRUN<strong>&lt;&gt;</strong>TOxRUN</code>\r\n </li>\r\n <li>\r\n <strong>annotate</strong>:\r\n <code>ATxRUN<strong>:</strong></code>\r\n </li>\r\n </ul>\r\n <p>For all the operations:</p>\r\n <ul>\r\n <li>\r\n prefix <code>(ITAG:OTAG)</code> to define input and/or\r\n output tags, separated by colon.\r\n </li>\r\n <li>\r\n prepend <code>&#x40;</code> to AT to use character\r\n indexes (0-N) rather than IDs.\r\n </li>\r\n <li>RUN (where applicable) defaults to 1.</li>\r\n <li>\r\n append features like <code>[NAME OPERATOR VALUE]</code>,\r\n separated by space, where:\r\n </li>\r\n <ol>\r\n <li>\r\n NAME is an arbitrary string. Prefixes:\r\n <code>*</code>=global features, <code>!</code>=remove\r\n feature (no value). Suffixes:\r\n <code>^</code>=short-lived (like\r\n <code>*version^=alpha</code>).\r\n </li>\r\n <li>\r\n OPERATOR is <code>=</code> (multiple),\r\n <code>:=</code> (single),\r\n <code>==</code> (first-single).\r\n </li>\r\n <li>\r\n VALUE is a string. If it includes spaces, wrap it in\r\n <code>\"\"</code>.\r\n </li>\r\n </ol>\r\n </ul>\r\n </div>\r\n </div>\r\n </fieldset>\r\n </mat-expansion-panel>\r\n </div>\r\n </div>\r\n\r\n <!-- result -->\r\n @if (result) {\r\n <fieldset id=\"result\">\r\n <legend>result</legend>\r\n <gve-chain-result-view\r\n [result]=\"result\"\r\n (stepPick)=\"onStepPick($event)\"\r\n />\r\n </fieldset>\r\n }\r\n </div>\r\n </mat-tab>\r\n\r\n <!-- image -->\r\n <mat-tab label=\"image\">\r\n <div id=\"image\">\r\n <div id=\"image-ctl\">\r\n <!-- url -->\r\n <mat-form-field class=\"long-text\">\r\n <mat-label>URL</mat-label>\r\n <input matInput [formControl]=\"imageUrl\" />\r\n </mat-form-field>\r\n <div class=\"form-row\">\r\n <!-- x -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>X</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"imageX\"\r\n placeholder=\"X\"\r\n />\r\n </mat-form-field>\r\n <!-- y -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>Y</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"imageY\"\r\n placeholder=\"Y\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n <div class=\"form-row\">\r\n <!-- width -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>width</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"imageWidth\"\r\n />\r\n </mat-form-field>\r\n <!-- height -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"imageHeight\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n <!-- defs -->\r\n <div>\r\n <mat-form-field class=\"long-text\">\r\n <mat-label>defs</mat-label>\r\n <textarea matInput [formControl]=\"defs\" rows=\"3\"></textarea>\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n <div id=\"image-view\">\r\n @if (imageUrl.value) {\r\n <img alt=\"background\" [src]=\"imageUrl.value\" width=\"600\" />\r\n }\r\n </div>\r\n </div>\r\n </mat-tab>\r\n\r\n <!-- timelines -->\r\n <mat-tab label=\"animation\">\r\n <gve-animation-timeline-set\r\n [tags]=\"opTags\"\r\n [elementIds]=\"opElementIds\"\r\n [timelines]=\"timelines.value\"\r\n (timelinesChange)=\"onTimelinesChange($event)\"\r\n />\r\n </mat-tab>\r\n </mat-tab-group>\r\n\r\n <!-- preview -->\r\n @if (snapshot) {\r\n <fieldset id=\"preview\">\r\n <legend class=\"button-row\">\r\n <span>{{ viewTitle }}</span>\r\n <button\r\n color=\"primary\"\r\n mat-icon-button\r\n type=\"button\"\r\n matTooltip=\"Refresh preview\"\r\n (click)=\"updateViewData()\"\r\n >\r\n <mat-icon class=\"mat-primary\">refresh</mat-icon>\r\n </button>\r\n </legend>\r\n <gve-snapshot-view\r\n [data]=\"viewData\"\r\n (snapshotRender)=\"onSnapshotRender($any($event))\"\r\n (visualEvent)=\"onVisualEvent($any($event))\"\r\n />\r\n <div>\r\n <mat-button-toggle\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n matTooltip=\"Toggle rulers\"\r\n [checked]=\"rulers\"\r\n (change)=\"toggleRulers()\"\r\n >\r\n <mat-icon class=\"mat-primary\">straighten</mat-icon>\r\n </mat-button-toggle>\r\n </div>\r\n @if (visualInfo) {\r\n <div id=\"visual-info\">{{ visualInfo }}</div>\r\n }\r\n </fieldset>\r\n }\r\n\r\n <!--buttons -->\r\n <div class=\"form-row-center\">\r\n <button\r\n type=\"button\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Discard changes\"\r\n (click)=\"close()\"\r\n >\r\n <mat-icon>clear</mat-icon>\r\n close\r\n </button>\r\n @if (!noSave) {\r\n <button\r\n type=\"submit\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Save changes\"\r\n [disabled]=\"form.invalid || form.pristine\"\r\n >\r\n <mat-icon>check_circle</mat-icon>\r\n save\r\n </button>\r\n }\r\n </div>\r\n</form>\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row-center{display:flex;gap:8px;align-items:center;justify-content:center;flex-wrap:wrap}.form-row,.form-row-center *{flex:0 0 auto}.form-row *.right{margin-left:auto}.button-row{display:flex;align-items:center;flex-wrap:wrap}.button-row *{flex:0 0 auto}.long-text{width:100%;max-width:800px}#text-range{margin:8px;border:1px solid silver;border-radius:6px;padding:6px}mat-expansion-panel{margin:8px 0}div#visual-info{font-size:95%;color:#909090;margin:8px}div#batch-container{display:grid;gap:16px;grid-template-rows:auto;grid-template-columns:1fr auto;grid-template-areas:\"batch-input batch-help\"}div#batch-input{grid-area:batch-input}div#batch-help{grid-area:batch-help}div#batch-help strong{color:#ff8c00}#list{margin:8px 0}#opStyle{margin-top:8px}table{width:100%;border-collapse:collapse}th{color:#909090;font-weight:400;text-align:left;background-color:#e1e0e0}th,td{padding:4px;border-bottom:1px solid silver}tbody tr:nth-child(2n){background-color:#e8e8e8}td.fit-width{width:1px;white-space:nowrap}tr.selected{background-color:#c8d9eb}tr.edited{background-color:#f6f6e4}fieldset{border:1px solid silver;border-radius:6px;padding:8px;margin:8px 0}legend{color:#909090}.error{color:red}.input-nr{width:6em}.full-width{width:100%}.code{font-family:Courier New,Courier,monospace}gve-animation-timeline-set{margin:8px 0}div#image{display:grid;gap:8px;grid-template-rows:auto;grid-template-columns:1fr auto;grid-template-areas:\"image-ctl image-view\"}div#image-ctl{grid-area:image-ctl}div#image-view{grid-area:image-view}@media only screen and (max-width: 959px){div#batch-container{grid-template-columns:1fr;grid-template-areas:\"batch-input\" \"batch-help\"}div#image{grid-template-columns:1fr;grid-template-areas:\"image-ctl\" \"image-view\"}}\n"] }]
547
- }], ctorParameters: () => [{ type: i1.FormBuilder }, { type: i2.GveApiService }, { type: i3.DialogService }, { type: i4.MatSnackBar }], propDecorators: { snapshot: [{
853
+ ], schemas: [CUSTOM_ELEMENTS_SCHEMA], template: "<form [formGroup]=\"form\" (submit)=\"save()\">\r\n <mat-tab-group>\r\n <!-- text -->\r\n <mat-tab>\r\n <ng-template mat-tab-label>\r\n <mat-icon>article</mat-icon> <span class=\"label\">text</span>\r\n </ng-template>\r\n <mat-expansion-panel [expanded]=\"!baseText.value\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title> base text </mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <gve-base-text-editor\r\n [text]=\"baseText.value\"\r\n (textChange)=\"onTextChange($event)\"\r\n />\r\n <fieldset>\r\n <div class=\"form-row\">\r\n <!-- offsetX -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>X offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"offsetX\"\r\n placeholder=\"X offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- offsetY -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>Y offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"offsetY\"\r\n placeholder=\"Y offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- lineHeightOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>ln h-offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"lineHeightOffset\"\r\n placeholder=\"ln h-offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- charSpacingOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>char spacing</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"charSpacingOffset\"\r\n placeholder=\"char spacing\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- spcWidthOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>spc w-offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"spcWidthOffset\"\r\n placeholder=\"spc w-offset\"\r\n />\r\n </mat-form-field>\r\n <!-- minLineHeights -->\r\n <div class=\"boxed\">\r\n <gve-ln-heights-editor\r\n [lineCount]=\"lineCount\"\r\n [heights]=\"lnHeights.value\"\r\n (heightsChange)=\"onHeightsChange($event)\"\r\n />\r\n </div>\r\n </div>\r\n <!-- textStyle -->\r\n <div>\r\n <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n <mat-label>text style</mat-label>\r\n <textarea\r\n matInput\r\n [formControl]=\"textStyle\"\r\n placeholder=\"text style\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n </fieldset>\r\n </mat-expansion-panel>\r\n </mat-tab>\r\n\r\n <!-- operations -->\r\n <mat-tab>\r\n <ng-template mat-tab-label>\r\n <mat-icon>edit</mat-icon> <span class=\"label\">operations</span>\r\n </ng-template>\r\n\r\n <div id=\"snapshot-container\">\r\n <div id=\"general\">\r\n <div class=\"form-row\">\r\n <!-- width -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>width</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"width\"\r\n placeholder=\"width\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- height -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"height\"\r\n placeholder=\"height\"\r\n />\r\n </mat-form-field>\r\n\r\n <div class=\"form-row right\">\r\n <!-- remove ops -->\r\n <button\r\n type=\"button\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Remove all the operations\"\r\n [disabled]=\"!operations.value.length || busy\"\r\n (click)=\"clearOperations()\"\r\n >\r\n <mat-icon>delete_forever</mat-icon> clear\r\n </button>\r\n\r\n <!-- batch add ops -->\r\n <button\r\n type=\"button\"\r\n mat-flat-button\r\n matTooltip=\"Add a batch of operations\"\r\n class=\"mat-primary\"\r\n [disabled]=\"busy\"\r\n (click)=\"parseOperations()\"\r\n >\r\n <mat-icon>post_add</mat-icon> batch\r\n </button>\r\n\r\n <!-- add op -->\r\n <button\r\n type=\"button\"\r\n mat-flat-button\r\n class=\"mat-primary\"\r\n [disabled]=\"!baseText.value\"\r\n (click)=\"editNewOperation()\"\r\n >\r\n <mat-icon>add_circle</mat-icon> operation\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- ops -->\r\n <div id=\"ops\">\r\n <!-- operations list -->\r\n @if (operations.value.length) {\r\n <table id=\"list\">\r\n <thead>\r\n <tr>\r\n <th></th>\r\n <th>ID</th>\r\n <th>type</th>\r\n <th>at</th>\r\n <th>run</th>\r\n <th>value</th>\r\n <th>itag</th>\r\n <th>otag</th>\r\n <th>gid</th>\r\n <th>feats</th>\r\n </tr>\r\n </thead>\r\n <tbody>\r\n @for (operation of operations.value; track operation.id; let\r\n index=$index) {\r\n <tr\r\n [ngClass]=\"{ selected: operation.id === resultOperationId }\"\r\n [ngClass]=\"{ edited: operation.id === editedOp?.id }\"\r\n >\r\n <td class=\"fit-width\">\r\n <!-- edit -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n class=\"mat-primary\"\r\n (click)=\"editOperation(index)\"\r\n matTooltip=\"Edit operation\"\r\n >\r\n <mat-icon>edit</mat-icon>\r\n </button>\r\n <!-- delete -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n class=\"mat-warn\"\r\n (click)=\"deleteOperation(index)\"\r\n matTooltip=\"Delete operation\"\r\n >\r\n <mat-icon class=\"mat-warn\">delete</mat-icon>\r\n </button>\r\n <!-- run to -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n class=\"mat-accent\"\r\n (click)=\"runTo(index)\"\r\n matTooltip=\"Run to this operation\"\r\n >\r\n <mat-icon class=\"mat-accent\">subscriptions</mat-icon>\r\n </button>\r\n </td>\r\n <td>{{ operation.id }}</td>\r\n <td>{{ operation.type | flatLookup : opTypeMap }}</td>\r\n <td>{{ operation.at }}</td>\r\n <td>{{ operation.run }}</td>\r\n <td>{{ operation.value }}</td>\r\n <td>{{ operation.inputTag }}</td>\r\n <td>{{ operation.outputTag }}</td>\r\n <td>{{ operation.groupId }}</td>\r\n <td>{{ operation.features?.length }}</td>\r\n </tr>\r\n }\r\n </tbody>\r\n </table>\r\n }\r\n\r\n <!-- operation editor -->\r\n <mat-expansion-panel [expanded]=\"editedOp\" [disabled]=\"!editedOp\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title>operation {{ editedOp?.id }}</mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <fieldset>\r\n <gve-chain-operation-editor\r\n [hidePreview]=\"editedOpIndex === -1\"\r\n [operation]=\"editedOp\"\r\n (operationCancel)=\"onOperationCancel()\"\r\n (operationChange)=\"onOperationChange($event)\"\r\n (operationPreview)=\"onOperationPreview($event)\"\r\n />\r\n </fieldset>\r\n </mat-expansion-panel>\r\n\r\n <!-- opStyle -->\r\n <div id=\"opStyle\">\r\n <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n <mat-label>operations style</mat-label>\r\n <textarea\r\n matInput\r\n [formControl]=\"opStyle\"\r\n placeholder=\"operations style\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n </div>\r\n </mat-tab>\r\n\r\n <!-- image -->\r\n <mat-tab>\r\n <ng-template mat-tab-label>\r\n <mat-icon>image</mat-icon> <span class=\"label\">image</span>\r\n </ng-template>\r\n\r\n <div id=\"image\">\r\n <div id=\"image-ctl\">\r\n <!-- url -->\r\n <mat-form-field class=\"long-text\">\r\n <mat-label>URL</mat-label>\r\n <input matInput [formControl]=\"imageUrl\" />\r\n </mat-form-field>\r\n <div class=\"form-row\">\r\n <!-- x -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>X</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"imageX\"\r\n placeholder=\"X\"\r\n />\r\n </mat-form-field>\r\n <!-- y -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>Y</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"imageY\"\r\n placeholder=\"Y\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n <div class=\"form-row\">\r\n <!-- width -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>width</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"imageWidth\"\r\n />\r\n </mat-form-field>\r\n <!-- height -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"imageHeight\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n <!-- defs -->\r\n <div>\r\n <mat-form-field class=\"long-text\">\r\n <mat-label>defs</mat-label>\r\n <textarea matInput [formControl]=\"defs\" rows=\"3\"></textarea>\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n <div id=\"image-view\">\r\n @if (imageUrl.value) {\r\n <img alt=\"background\" [src]=\"imageUrl.value\" width=\"600\" />\r\n }\r\n </div>\r\n </div>\r\n </mat-tab>\r\n\r\n <!-- timelines -->\r\n <mat-tab>\r\n <ng-template mat-tab-label>\r\n <mat-icon>animation</mat-icon> <span class=\"label\">animation</span>\r\n </ng-template>\r\n\r\n <gve-animation-timeline-set\r\n [tags]=\"opTags\"\r\n [elementIds]=\"opElementIds\"\r\n [timelines]=\"timelines.value\"\r\n (timelinesChange)=\"onTimelinesChange($event)\"\r\n />\r\n </mat-tab>\r\n </mat-tab-group>\r\n\r\n <!-- progress -->\r\n <div>\r\n <mat-progress-bar mode=\"indeterminate\" *ngIf=\"busy\" />\r\n </div>\r\n\r\n <!-- result -->\r\n @if (result) {\r\n <fieldset id=\"result\">\r\n <legend>result</legend>\r\n <gve-chain-result-view\r\n [result]=\"result\"\r\n [initialStepIndex]=\"initialStepIndex\"\r\n (stepPick)=\"onStepPick($event)\"\r\n />\r\n </fieldset>\r\n }\r\n\r\n <!-- snapshot view -->\r\n <fieldset id=\"preview\">\r\n <legend class=\"button-row\">\r\n <span>{{ viewTitle }}</span>\r\n </legend>\r\n <!-- snapshot view -->\r\n <gve-snapshot-view\r\n #snapshotView\r\n [debug]=\"debug\"\r\n [data]=\"viewData\"\r\n (snapshotRender)=\"onSnapshotRender($any($event))\"\r\n (visualEvent)=\"onVisualEvent($any($event))\"\r\n />\r\n <div class=\"button-row\">\r\n <!-- toggle ruler -->\r\n <mat-button-toggle\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n matTooltip=\"Toggle rulers\"\r\n [checked]=\"rulers\"\r\n (change)=\"toggleRulers()\"\r\n >\r\n <mat-icon class=\"mat-primary\">straighten</mat-icon>\r\n </mat-button-toggle>\r\n <!-- run to last -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n class=\"primary\"\r\n matTooltip=\"Run all operations\"\r\n [disabled]=\"!operations.value.length\"\r\n (click)=\"runToLast()\"\r\n >\r\n <mat-icon>play_circle</mat-icon>\r\n </button>\r\n </div>\r\n @if (visualInfo) {\r\n <div id=\"visual-info\">{{ visualInfo }}</div>\r\n }\r\n </fieldset>\r\n\r\n <!--buttons -->\r\n <div class=\"form-row-center\">\r\n <button\r\n type=\"button\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Discard changes\"\r\n (click)=\"close()\"\r\n >\r\n <mat-icon>clear</mat-icon>\r\n close\r\n </button>\r\n @if (!noSave) {\r\n <button\r\n type=\"submit\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Save changes\"\r\n [disabled]=\"form.invalid\"\r\n >\r\n <mat-icon>check_circle</mat-icon>\r\n save\r\n </button>\r\n }\r\n </div>\r\n</form>\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row-center{display:flex;gap:8px;align-items:center;justify-content:center;flex-wrap:wrap}.form-row,.form-row-center *{flex:0 0 auto}.form-row .right{margin-left:auto}.button-row{display:flex;align-items:center;flex-wrap:wrap}.button-row *{flex:0 0 auto}.long-text{width:100%;max-width:800px}#text-range{margin:8px;border:1px solid silver;border-radius:6px;padding:6px}mat-expansion-panel{margin:8px 0}div#visual-info{font-size:95%;color:#909090;margin:8px}#list{margin:8px 0}#opStyle{margin-top:8px}table{width:100%;border-collapse:collapse}th{color:#909090;font-weight:400;text-align:left;background-color:#e1e0e0}th,td{padding:4px;border-bottom:1px solid silver}tbody tr:nth-child(2n){background-color:#e8e8e8}td.fit-width{width:1px;white-space:nowrap}tr.selected{background-color:#c8d9eb}tr.edited{background-color:#f6f6e4}fieldset{border:1px solid silver;border-radius:6px;padding:8px;margin:8px 0}legend{color:#909090}.error{color:red}.input-nr{width:6em}.full-width{width:100%}.code{font-family:Courier New,Courier,monospace}.boxed{border:1px solid silver;border-radius:6px;padding:8px;margin:8px 0}span.label{margin-left:8px}gve-animation-timeline-set{margin:8px 0}div#image{display:grid;gap:8px;grid-template-rows:auto;grid-template-columns:1fr auto;grid-template-areas:\"image-ctl image-view\"}div#image-ctl{grid-area:image-ctl}div#image-view{grid-area:image-view}@media only screen and (max-width: 959px){div#image{grid-template-columns:1fr;grid-template-areas:\"image-ctl\" \"image-view\"}}\n"] }]
854
+ }], ctorParameters: () => [{ type: i1.FormBuilder }, { type: i2.GveApiService }, { type: i3.MatDialog }, { type: i4.DialogService }, { type: i5.MatSnackBar }], propDecorators: { snapshotView: [{
855
+ type: ViewChild,
856
+ args: ['snapshotView', { static: false }]
857
+ }], snapshot: [{
548
858
  type: Input
549
859
  }], batchOps: [{
550
860
  type: Input
551
861
  }], noSave: [{
552
862
  type: Input
863
+ }], debug: [{
864
+ type: Input
553
865
  }], snapshotChange: [{
554
866
  type: Output
555
867
  }], snapshotCancel: [{
556
868
  type: Output
557
869
  }] } });
558
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snapshot-editor.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/gve-core/src/lib/components/snapshot-editor/snapshot-editor.component.ts","../../../../../../../projects/myrmidon/gve-core/src/lib/components/snapshot-editor/snapshot-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  CUSTOM_ELEMENTS_SCHEMA,\r\n  Component,\r\n  EventEmitter,\r\n  Input,\r\n  OnInit,\r\n  Output,\r\n} from '@angular/core';\r\nimport {\r\n  FormBuilder,\r\n  FormControl,\r\n  FormGroup,\r\n  FormsModule,\r\n  ReactiveFormsModule,\r\n  Validators,\r\n} from '@angular/forms';\r\nimport { CommonModule } from '@angular/common';\r\n\r\nimport { MatButtonModule } from '@angular/material/button';\r\nimport { MatButtonToggleModule } from '@angular/material/button-toggle';\r\nimport { MatCheckboxModule } from '@angular/material/checkbox';\r\nimport { MatExpansionModule } from '@angular/material/expansion';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatIconModule } from '@angular/material/icon';\r\nimport { MatInputModule } from '@angular/material/input';\r\nimport { MatProgressBarModule } from '@angular/material/progress-bar';\r\nimport { MatSnackBar, MatSnackBarModule } from '@angular/material/snack-bar';\r\nimport { MatTabsModule } from '@angular/material/tabs';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\n\r\nimport { customAlphabet } from 'nanoid';\r\n\r\nimport { NgToolsModule, deepCopy } from '@myrmidon/ng-tools';\r\nimport { DialogService } from '@myrmidon/ng-mat-tools';\r\n\r\nimport {\r\n  CharChainOperation,\r\n  CharNode,\r\n  DEFAULT_SVG_BASE_TEXT_OPTIONS,\r\n  Feature,\r\n  GveAnimationTimeline,\r\n  GveVisualEvent,\r\n  Snapshot,\r\n  SnapshotViewData,\r\n  SnapshotViewRenderEvent,\r\n  SnapshotViewService,\r\n} from '@myrmidon/gve-snapshot-view';\r\n\r\nimport {\r\n  BaseTextViewComponent,\r\n  VarBaseTextRange,\r\n} from '../base-text-view/base-text-view.component';\r\nimport { ChainOperationEditorComponent } from '../chain-operation-editor/chain-operation-editor.component';\r\nimport {\r\n  ChainOperationContextStep,\r\n  CharChainResult,\r\n  GveApiService,\r\n} from '../../services/gve-api.service';\r\nimport { ChainResultViewComponent } from '../chain-result-view/chain-result-view.component';\r\nimport { LnHeightsEditorComponent } from '../ln-heights-editor/ln-heights-editor.component';\r\nimport { BaseTextEditorComponent } from '../base-text-editor/base-text-editor.component';\r\nimport { AnimationTimelineSetComponent } from '../animation-timeline-set/animation-timeline-set.component';\r\n\r\nconst VIEW_TITLE = 'preview';\r\n\r\n/**\r\n * A component to edit a text snapshot.\r\n */\r\n@Component({\r\n  selector: 'gve-snapshot-editor',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    FormsModule,\r\n    ReactiveFormsModule,\r\n    MatButtonModule,\r\n    MatButtonToggleModule,\r\n    MatCheckboxModule,\r\n    MatExpansionModule,\r\n    MatFormFieldModule,\r\n    MatIconModule,\r\n    MatInputModule,\r\n    MatProgressBarModule,\r\n    MatSnackBarModule,\r\n    MatTabsModule,\r\n    MatTooltipModule,\r\n    NgToolsModule,\r\n    BaseTextEditorComponent,\r\n    BaseTextViewComponent,\r\n    ChainOperationEditorComponent,\r\n    ChainResultViewComponent,\r\n    LnHeightsEditorComponent,\r\n    AnimationTimelineSetComponent,\r\n  ],\r\n  schemas: [CUSTOM_ELEMENTS_SCHEMA],\r\n  templateUrl: './snapshot-editor.component.html',\r\n  styleUrl: './snapshot-editor.component.css',\r\n})\r\nexport class SnapshotEditorComponent implements OnInit {\r\n  private readonly _nanoid = customAlphabet('1234567890abcdef', 10);\r\n  private _snapshot?: Snapshot;\r\n  private _renderer?: SnapshotViewService;\r\n  private _previewing?: boolean;\r\n\r\n  /**\r\n   * The snapshot to edit.\r\n   */\r\n  @Input()\r\n  public get snapshot(): Snapshot | undefined {\r\n    return this._snapshot;\r\n  }\r\n  public set snapshot(value: Snapshot | undefined | null) {\r\n    if (this._snapshot === value) {\r\n      return;\r\n    }\r\n    this._snapshot = value || undefined;\r\n    this.updateForm(this._snapshot);\r\n    this.updateViewData();\r\n  }\r\n\r\n  /**\r\n   * The batch operations text to set for the editor.\r\n   */\r\n  @Input()\r\n  public batchOps?: string;\r\n\r\n  /**\r\n   * True to disable saving.\r\n   */\r\n  @Input()\r\n  public noSave?: boolean;\r\n\r\n  /**\r\n   * Emitted when the user saves the edited snapshot.\r\n   */\r\n  @Output()\r\n  public readonly snapshotChange: EventEmitter<Snapshot> = new EventEmitter();\r\n\r\n  /**\r\n   * Emitted when the user cancels the snapshot editing.\r\n   */\r\n  @Output()\r\n  public readonly snapshotCancel = new EventEmitter<void>();\r\n\r\n  // general\r\n  public width: FormControl<number>;\r\n  public height: FormControl<number>;\r\n  public style: FormControl<string | null>;\r\n  // base text\r\n  public baseText: FormControl<CharNode[] | string>;\r\n  public offsetX: FormControl<number>;\r\n  public offsetY: FormControl<number>;\r\n  public lineHeightOffset: FormControl<number>;\r\n  public lnHeights: FormControl<Record<number, number> | null>;\r\n  public charSpacingOffset: FormControl<number>;\r\n  public spcWidthOffset: FormControl<number>;\r\n  public textStyle: FormControl<string | null>;\r\n  // operations\r\n  public operations: FormControl<CharChainOperation[]>;\r\n  public inputOps: FormControl<string | null>;\r\n  public opStyle: FormControl<string | null>;\r\n  public form: FormGroup;\r\n  // image and defs\r\n  public imageUrl: FormControl<string | null>;\r\n  public imageOpacity: FormControl<number>;\r\n  public imageX: FormControl<number>;\r\n  public imageY: FormControl<number>;\r\n  public imageWidth: FormControl<number>;\r\n  public imageHeight: FormControl<number>;\r\n  public defs: FormControl<string | null>;\r\n  // timelines\r\n  public timelines: FormControl<GveAnimationTimeline[]>;\r\n\r\n  public textRange?: VarBaseTextRange;\r\n  public editedOp?: CharChainOperation;\r\n  public editedOpIndex = -1;\r\n\r\n  public opTags: string[] = [];\r\n  public opElementIds: string[] = [];\r\n  // TODO: add code to update opTags and opElementIds\r\n\r\n  public busy?: boolean;\r\n  public parseError?: string;\r\n  public opTypeMap = {\r\n    0: 'replace',\r\n    1: 'delete',\r\n    2: 'add-before',\r\n    3: 'add-after',\r\n    4: 'move-before',\r\n    5: 'move-after',\r\n    6: 'swap',\r\n    7: 'annotate',\r\n  };\r\n  public lineCount = 0;\r\n\r\n  public viewTitle = VIEW_TITLE;\r\n  public viewData?: SnapshotViewData;\r\n  public visualInfo?: string;\r\n  public rulers = true;\r\n  // run result\r\n  public result?: CharChainResult;\r\n  public resultOperationId?: string;\r\n\r\n  constructor(\r\n    formBuilder: FormBuilder,\r\n    private _api: GveApiService,\r\n    private _dialogService: DialogService,\r\n    private _snackbar: MatSnackBar\r\n  ) {\r\n    // general\r\n    this.width = new FormControl<number>(800, { nonNullable: true });\r\n    this.height = new FormControl<number>(600, { nonNullable: true });\r\n    this.style = new FormControl<string | null>(null);\r\n    // base text\r\n    this.baseText = new FormControl<CharNode[] | string>([], {\r\n      nonNullable: true,\r\n      validators: [Validators.required],\r\n    });\r\n    this.offsetX = new FormControl<number>(0, { nonNullable: true });\r\n    this.offsetY = new FormControl<number>(0, { nonNullable: true });\r\n    this.lineHeightOffset = new FormControl<number>(\r\n      DEFAULT_SVG_BASE_TEXT_OPTIONS.lineHeightOffset,\r\n      { nonNullable: true }\r\n    );\r\n    this.lnHeights = new FormControl<Record<number, number> | null>(null);\r\n    this.charSpacingOffset = new FormControl<number>(0, { nonNullable: true });\r\n    this.spcWidthOffset = new FormControl<number>(0, { nonNullable: true });\r\n    this.textStyle = new FormControl<string | null>(null);\r\n\r\n    // operations\r\n    this.operations = new FormControl<CharChainOperation[]>([], {\r\n      nonNullable: true,\r\n    });\r\n    this.inputOps = new FormControl<string | null>(null);\r\n    this.opStyle = new FormControl<string | null>(null);\r\n    // image\r\n    this.imageUrl = new FormControl<string | null>(null);\r\n    this.imageOpacity = new FormControl<number>(1, { nonNullable: true });\r\n    this.imageX = new FormControl<number>(0, { nonNullable: true });\r\n    this.imageY = new FormControl<number>(0, { nonNullable: true });\r\n    this.imageWidth = new FormControl<number>(0, { nonNullable: true });\r\n    this.imageHeight = new FormControl<number>(0, { nonNullable: true });\r\n    this.defs = new FormControl<string | null>(null);\r\n\r\n    // timelines\r\n    this.timelines = new FormControl<GveAnimationTimeline[]>([], {\r\n      nonNullable: true,\r\n    });\r\n\r\n    this.form = formBuilder.group({\r\n      width: this.width,\r\n      height: this.height,\r\n      style: this.style,\r\n      baseText: this.baseText,\r\n      offsetX: this.offsetX,\r\n      offsetY: this.offsetY,\r\n      lineHeightOffset: this.lineHeightOffset,\r\n      lnHeights: this.lnHeights,\r\n      charSpacingOffset: this.charSpacingOffset,\r\n      spcWidthOffset: this.spcWidthOffset,\r\n      textStyle: this.textStyle,\r\n      operations: this.operations,\r\n      inputOps: this.inputOps,\r\n      opStyle: this.opStyle,\r\n      // image\r\n      imageUrl: this.imageUrl,\r\n      imageOpacity: this.imageOpacity,\r\n      imageX: this.imageX,\r\n      imageY: this.imageY,\r\n      imageWidth: this.imageWidth,\r\n      imageHeight: this.imageHeight,\r\n      defs: this.defs,\r\n      // timelines\r\n      timelines: this.timelines,\r\n    });\r\n  }\r\n\r\n  public ngOnInit(): void {\r\n    if (this.batchOps) {\r\n      this.inputOps.setValue(this.batchOps);\r\n      this.parseOperations();\r\n    }\r\n  }\r\n\r\n  public updateViewData(\r\n    newOperation?: CharChainOperation,\r\n    snapshot?: Snapshot,\r\n    title = VIEW_TITLE\r\n  ): void {\r\n    console.log('update view data');\r\n    if (!snapshot) {\r\n      snapshot = this.getSnapshot();\r\n    }\r\n    // update or add new operation in copy\r\n    if (newOperation) {\r\n      const index = snapshot.operations.findIndex(\r\n        (op) => op.id === newOperation.id\r\n      );\r\n      if (index > -1) {\r\n        snapshot.operations.splice(index, 1, newOperation);\r\n      } else {\r\n        snapshot.operations.push(newOperation);\r\n      }\r\n    }\r\n\r\n    // update view data\r\n    this.viewTitle = title;\r\n    this.visualInfo = undefined;\r\n    this.viewData = {\r\n      snapshot: snapshot,\r\n      options: {\r\n        // debug: true,\r\n        delayedRender: true,\r\n        showRulers: true,\r\n        showGrid: true,\r\n        panZoom: true,\r\n      },\r\n    };\r\n    console.log(`view data: ${this.viewData}`);\r\n  }\r\n\r\n  private updateForm(snapshot: Snapshot | undefined) {\r\n    if (!snapshot) {\r\n      this.form.reset();\r\n    } else {\r\n      this.width.setValue(snapshot.size.width);\r\n      this.height.setValue(snapshot.size.height);\r\n      this.style.setValue(snapshot.style || null);\r\n      this.imageUrl.setValue(snapshot.image?.url || null);\r\n      this.imageOpacity.setValue(snapshot.image?.opacity || 0);\r\n      this.imageX.setValue(snapshot.image?.canvas?.x || 0);\r\n      this.imageY.setValue(snapshot.image?.canvas?.y || 0);\r\n      this.imageWidth.setValue(snapshot.image?.canvas?.width || 0);\r\n      this.imageHeight.setValue(snapshot.image?.canvas?.height || 0);\r\n      this.defs.setValue(snapshot.defs || null);\r\n      this.baseText.setValue(snapshot.text);\r\n      this.offsetX.setValue(snapshot.textOptions?.offset?.x || 0);\r\n      this.offsetY.setValue(snapshot.textOptions?.offset?.y || 0);\r\n      this.lineHeightOffset.setValue(snapshot.textOptions?.lineHeightOffset);\r\n      this.lnHeights.setValue(snapshot.textOptions?.minLineHeights || null);\r\n      this.charSpacingOffset.setValue(snapshot.textOptions?.charSpacingOffset);\r\n      this.spcWidthOffset.setValue(snapshot.textOptions?.spcWidthOffset);\r\n      this.textStyle.setValue(snapshot.textStyle || null);\r\n      this.operations.setValue(snapshot.operations);\r\n      this.inputOps.reset();\r\n      this.opStyle.setValue(snapshot.opStyle || null);\r\n      // timelines\r\n      const timelines: GveAnimationTimeline[] = [];\r\n      if (snapshot.timelines) {\r\n        for (let tag in snapshot.timelines) {\r\n          timelines.push(snapshot.timelines[tag]);\r\n        }\r\n      }\r\n      this.timelines.setValue(timelines);\r\n    }\r\n    this.form.markAsPristine();\r\n  }\r\n\r\n  public editNewOperation() {\r\n    this.editedOp = {\r\n      id: this._nanoid(),\r\n      type: 0,\r\n      at: this.textRange?.at || 0,\r\n      run: this.textRange?.run || 1,\r\n    };\r\n    this.editedOpIndex = -1;\r\n  }\r\n\r\n  public editOperation(index: number) {\r\n    this.editedOpIndex = index;\r\n    this.editedOp = deepCopy(this.operations.value[index]);\r\n  }\r\n\r\n  private closeEditedOp() {\r\n    this.editedOpIndex = -1;\r\n    this.editedOp = undefined;\r\n  }\r\n\r\n  public onOperationCancel() {\r\n    this.closeEditedOp();\r\n  }\r\n\r\n  public onOperationChange(op: CharChainOperation) {\r\n    console.log('operation change');\r\n    const operations = [...this.operations.value];\r\n\r\n    if (this.editedOpIndex > -1) {\r\n      operations.splice(this.editedOpIndex, 1, op);\r\n    } else {\r\n      operations.push(op);\r\n    }\r\n    this.operations.setValue(operations);\r\n    this.operations.markAsDirty();\r\n    this.operations.updateValueAndValidity();\r\n\r\n    this.closeEditedOp();\r\n  }\r\n\r\n  public deleteOperation(index: number) {\r\n    this._dialogService\r\n      .confirm(\r\n        'Confirm Deletion',\r\n        `Delete operation \"${this.operations.value[index].id}\"?`\r\n      )\r\n      .subscribe((yes: boolean) => {\r\n        if (yes) {\r\n          if (this.resultOperationId === this.operations.value[index].id) {\r\n            this.resultOperationId = undefined;\r\n          }\r\n          const operations = [...this.operations.value];\r\n          operations.splice(index, 1);\r\n          this.operations.setValue(operations);\r\n          this.operations.markAsDirty();\r\n          this.operations.updateValueAndValidity();\r\n        }\r\n      });\r\n  }\r\n\r\n  public onTextChange(text: CharNode[]) {\r\n    console.log('text change: ' + text);\r\n    this.baseText.setValue(text);\r\n    this.baseText.updateValueAndValidity();\r\n    this.baseText.markAsDirty();\r\n    if (!text.length) {\r\n      this.lineCount = 0;\r\n    } else {\r\n      this.lineCount = text.filter((c) => c.data === '\\n').length + 1;\r\n    }\r\n    this.textRange = undefined;\r\n    this.closeEditedOp();\r\n    this.operations.reset();\r\n    this.updateViewData();\r\n  }\r\n\r\n  public onRangePick(range: VarBaseTextRange) {\r\n    this.textRange = range;\r\n  }\r\n\r\n  public parseOperations() {\r\n    if (this.busy || !this.inputOps.value) {\r\n      return;\r\n    }\r\n    this._dialogService\r\n      .confirm('Confirm', 'Parse operations?')\r\n      .subscribe((yes: boolean) => {\r\n        if (yes) {\r\n          this.busy = true;\r\n          this.parseError = undefined;\r\n\r\n          this._api.parseOperations(this.inputOps.value!).subscribe({\r\n            next: (wrapper) => {\r\n              const operations = [...this.operations.value];\r\n              operations.push(...wrapper.result!);\r\n              this.operations.setValue(operations);\r\n              this.operations.markAsDirty();\r\n              this.operations.updateValueAndValidity();\r\n              this.updateViewData();\r\n            },\r\n            error: (error) => {\r\n              this.parseError = error.message;\r\n            },\r\n            complete: () => {\r\n              this.busy = false;\r\n            },\r\n          });\r\n        }\r\n      });\r\n  }\r\n\r\n  public clearOperations(): void {\r\n    this._dialogService\r\n      .confirm('Confirm', 'Remove all operations?')\r\n      .subscribe((yes: boolean) => {\r\n        if (yes) {\r\n          this.resultOperationId = undefined;\r\n          this.operations.reset();\r\n          this.operations.markAsDirty();\r\n          this.operations.updateValueAndValidity();\r\n          this.updateViewData();\r\n        }\r\n      });\r\n  }\r\n\r\n  public onOperationPreview(operation: CharChainOperation): void {\r\n    if (this._previewing) {\r\n      return;\r\n    }\r\n    this._previewing = true;\r\n    setTimeout(() => {\r\n      this.updateViewData(operation);\r\n      this._previewing = false;\r\n    }, 0);\r\n  }\r\n\r\n  public runTo(index: number): void {\r\n    const snapshot = this.getSnapshot();\r\n    if (!snapshot.text) {\r\n      return;\r\n    }\r\n    const operations = this.operations.value.slice(0, index + 1);\r\n    this.busy = true;\r\n    this._api.runOperations(snapshot.text as CharNode[], operations).subscribe({\r\n      next: (wrapper) => {\r\n        if (wrapper.error) {\r\n          this._snackbar.open(wrapper.error, 'OK');\r\n          return;\r\n        }\r\n        this.result = wrapper.result!;\r\n        // TODO update view\r\n        // this.updateViewData(\r\n        //   undefined,\r\n        //   snapshot,\r\n        //   this.operations.value[index].outputTag\r\n        // );\r\n      },\r\n      error: (error) => {\r\n        console.error(error);\r\n        this._snackbar.open('Error running operations', 'OK');\r\n      },\r\n      complete: () => {\r\n        this.busy = false;\r\n      },\r\n    });\r\n  }\r\n\r\n  public onStepPick(step: ChainOperationContextStep): void {\r\n    if (!this.result) {\r\n      return;\r\n    }\r\n    this.resultOperationId = step.operation.id;\r\n\r\n    // get snapshot for preview by updating its text and\r\n    // removing operations past the last executed\r\n    const stepNodes = this.result.taggedNodes[step.outputTag];\r\n\r\n    // we always display the original base text, just updating\r\n    // its nodes features to those of the picked step, and adding\r\n    // new nodes if they are not in the original text. It is assumed\r\n    // that these new nodes have a manually set position.\r\n    const snapshot = this.getSnapshot();\r\n    const nodes: CharNode[] = deepCopy(snapshot.text as CharNode[]);\r\n    for (const sn of stepNodes) {\r\n      const i = nodes.findIndex((n) => n.id === sn.id);\r\n      if (i > -1) {\r\n        nodes[i].features = sn.features;\r\n      } else {\r\n        nodes.push(sn);\r\n      }\r\n    }\r\n\r\n    // find max ID in snapshot.text (=original) nodes,\r\n    // so that any ID greater than it belongs to new nodes\r\n    const maxOriginalId = (snapshot.text as CharNode[]).reduce(\r\n      (max, n) => Math.max(max, n.id),\r\n      0\r\n    );\r\n\r\n    // append nodes introduced before the picked step if any\r\n    // (those introduced by the picked step have already been added)\r\n    const stepIndex = this.result.steps.indexOf(step);\r\n    for (let i = 0; i < stepIndex; i++) {\r\n      const stepNodes = this.result.taggedNodes[this.result.steps[i].outputTag];\r\n      for (const n of stepNodes) {\r\n        if (n.id > maxOriginalId && !nodes.find((x) => x.id === n.id)) {\r\n          nodes.push(n);\r\n        }\r\n      }\r\n    }\r\n\r\n    snapshot.text = nodes;\r\n    const i = snapshot.operations.findIndex((o) => o.id === step.operation.id);\r\n    if (i > -1) {\r\n      snapshot.operations = snapshot.operations.slice(0, i + 1);\r\n    }\r\n    this.updateViewData(undefined, snapshot, step.outputTag);\r\n  }\r\n\r\n  public onVisualEvent(event: CustomEvent<GveVisualEvent>): void {\r\n    const d = event.detail;\r\n    if (d.event.type === 'mouseover') {\r\n      const visual = d.source;\r\n      const sb: string[] = [];\r\n      sb.push('#' + visual.id);\r\n      sb.push(` (${visual.type})`);\r\n      if (visual.data?.label) {\r\n        sb.push(': ');\r\n        sb.push(visual.data.label);\r\n      }\r\n      if (visual.data?.features?.length) {\r\n        sb.push(\r\n          visual.data.features\r\n            .map((f: Feature) => `${f.name}=${f.value}`)\r\n            .join('\\n')\r\n        );\r\n      }\r\n      this.visualInfo = sb.join('');\r\n    }\r\n  }\r\n\r\n  public onHeightsChange(heights: Record<number, number> | undefined): void {\r\n    this.lnHeights.setValue(heights || null);\r\n    this.lnHeights.markAsDirty();\r\n    this.lnHeights.updateValueAndValidity();\r\n  }\r\n\r\n  public close(): void {\r\n    this.snapshotCancel.emit();\r\n  }\r\n\r\n  public onSnapshotRender(event: CustomEvent<SnapshotViewRenderEvent>): void {\r\n    this._renderer = (event.detail as SnapshotViewRenderEvent).renderer;\r\n  }\r\n\r\n  public toggleRulers(): void {\r\n    if (!this._renderer) {\r\n      return;\r\n    }\r\n    this.rulers = this._renderer.toggleRulers();\r\n  }\r\n\r\n  private getSnapshot(): Snapshot {\r\n    const snapshot: Snapshot = {\r\n      size: {\r\n        width: this.width.value,\r\n        height: this.height.value,\r\n      },\r\n      style: this.style.value || undefined,\r\n      defs: this.defs.value || undefined,\r\n      text: this.baseText.value!,\r\n      textStyle: this.textStyle.value || undefined,\r\n      textOptions: {\r\n        lineHeightOffset: this.lineHeightOffset.value,\r\n        minLineHeights: this.lnHeights.value || undefined,\r\n        charSpacingOffset: this.charSpacingOffset.value,\r\n        spcWidthOffset: this.spcWidthOffset.value,\r\n        offset: {\r\n          x: this.offsetX.value,\r\n          y: this.offsetY.value,\r\n        },\r\n      },\r\n      operations: [...this.operations.value],\r\n      opStyle: this.opStyle.value || undefined,\r\n      timelines: {}, // TODO\r\n    };\r\n\r\n    // image\r\n    if (this.imageUrl.value) {\r\n      snapshot.image = {\r\n        url: this.imageUrl.value,\r\n        opacity: this.imageOpacity.value || undefined,\r\n      };\r\n      if (\r\n        this.imageX.value ||\r\n        this.imageY.value ||\r\n        this.imageWidth.value ||\r\n        this.imageHeight.value\r\n      ) {\r\n        snapshot.image.canvas = {\r\n          x: this.imageX.value,\r\n          y: this.imageY.value,\r\n          width: this.imageWidth.value,\r\n          height: this.imageHeight.value,\r\n        };\r\n      }\r\n    }\r\n\r\n    // timelines\r\n    if (this.timelines.value.length) {\r\n      snapshot.timelines = {};\r\n      this.timelines.value.forEach((t) => {\r\n        snapshot.timelines![t.tag] = t;\r\n      });\r\n    } else {\r\n      snapshot.timelines = undefined;\r\n    }\r\n\r\n    return snapshot;\r\n  }\r\n\r\n  public onTimelinesChange(timelines: GveAnimationTimeline[]): void {\r\n    this.timelines.setValue(timelines);\r\n    this.timelines.markAsDirty();\r\n    this.timelines.updateValueAndValidity();\r\n  }\r\n\r\n  public save(): void {\r\n    if (this.form.invalid || this.noSave) {\r\n      return;\r\n    }\r\n    this._snapshot = this.getSnapshot();\r\n    this.snapshotChange.emit(this._snapshot);\r\n  }\r\n}\r\n","<form [formGroup]=\"form\" (submit)=\"save()\">\r\n  <mat-tab-group>\r\n    <!-- text -->\r\n    <mat-tab label=\"text\">\r\n      <mat-expansion-panel [expanded]=\"!baseText.value\">\r\n        <mat-expansion-panel-header>\r\n          <mat-panel-title> base text </mat-panel-title>\r\n        </mat-expansion-panel-header>\r\n        <gve-base-text-editor\r\n          [text]=\"baseText.value\"\r\n          (textChange)=\"onTextChange($event)\"\r\n        />\r\n        <fieldset>\r\n          <div class=\"form-row\">\r\n            <!-- offsetX -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>X offset</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"offsetX\"\r\n                placeholder=\"X offset\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- offsetY -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>Y offset</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"offsetY\"\r\n                placeholder=\"Y offset\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- lineHeightOffset -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>ln h-offset</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"lineHeightOffset\"\r\n                placeholder=\"ln h-offset\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- charSpacingOffset -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>char spacing</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"charSpacingOffset\"\r\n                placeholder=\"char spacing\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- spcWidthOffset -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>spc w-offset</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"spcWidthOffset\"\r\n                placeholder=\"spc w-offset\"\r\n              />\r\n            </mat-form-field>\r\n            <!-- minLineHeights -->\r\n            <div>\r\n              <gve-ln-heights-editor\r\n                [lineCount]=\"lineCount\"\r\n                [heights]=\"lnHeights.value\"\r\n                (heightsChange)=\"onHeightsChange($event)\"\r\n              />\r\n            </div>\r\n          </div>\r\n          <!-- textStyle -->\r\n          <div>\r\n            <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n              <mat-label>text style</mat-label>\r\n              <textarea\r\n                matInput\r\n                [formControl]=\"textStyle\"\r\n                placeholder=\"text style\"\r\n              ></textarea>\r\n            </mat-form-field>\r\n          </div>\r\n        </fieldset>\r\n      </mat-expansion-panel>\r\n    </mat-tab>\r\n\r\n    <!-- operations -->\r\n    <mat-tab label=\"operations\">\r\n      <div id=\"snapshot-container\">\r\n        <div id=\"general\">\r\n          <div class=\"form-row\">\r\n            <!-- width -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>width</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                min=\"0\"\r\n                [formControl]=\"width\"\r\n                placeholder=\"width\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- height -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>height</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                min=\"0\"\r\n                [formControl]=\"height\"\r\n                placeholder=\"height\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- add op -->\r\n            <button\r\n              type=\"button\"\r\n              mat-flat-button\r\n              color=\"primary\"\r\n              class=\"mat-primary right\"\r\n              [disabled]=\"!baseText.value\"\r\n              (click)=\"editNewOperation()\"\r\n            >\r\n              <mat-icon>add_circle</mat-icon> operation\r\n            </button>\r\n          </div>\r\n        </div>\r\n\r\n        <!-- ops -->\r\n        <div id=\"ops\">\r\n          <!-- operations list -->\r\n          @if (operations.value.length) {\r\n          <table id=\"list\">\r\n            <thead>\r\n              <tr>\r\n                <th></th>\r\n                <th>ID</th>\r\n                <th>type</th>\r\n                <th>at</th>\r\n                <th>run</th>\r\n                <th>value</th>\r\n                <th>itag</th>\r\n                <th>otag</th>\r\n                <th>gid</th>\r\n                <th>feats</th>\r\n              </tr>\r\n            </thead>\r\n            <tbody>\r\n              @for (operation of operations.value; track operation.id; let\r\n              index=$index) {\r\n              <tr\r\n                [ngClass]=\"{ selected: operation.id === resultOperationId }\"\r\n                [ngClass]=\"{ edited: operation.id === editedOp?.id }\"\r\n              >\r\n                <td class=\"fit-width\">\r\n                  <!-- edit -->\r\n                  <button\r\n                    type=\"button\"\r\n                    mat-icon-button\r\n                    color=\"primary\"\r\n                    class=\"mat-primary\"\r\n                    (click)=\"editOperation(index)\"\r\n                    matTooltip=\"Edit operation\"\r\n                  >\r\n                    <mat-icon>edit</mat-icon>\r\n                  </button>\r\n                  <!-- delete -->\r\n                  <button\r\n                    type=\"button\"\r\n                    mat-icon-button\r\n                    color=\"warn\"\r\n                    (click)=\"deleteOperation(index)\"\r\n                    matTooltip=\"Delete operation\"\r\n                  >\r\n                    <mat-icon class=\"mat-warn\">delete</mat-icon>\r\n                  </button>\r\n                  <!-- run to -->\r\n                  <button\r\n                    type=\"button\"\r\n                    mat-icon-button\r\n                    color=\"accent\"\r\n                    (click)=\"runTo(index)\"\r\n                    matTooltip=\"Run to this operation\"\r\n                  >\r\n                    <mat-icon class=\"mat-accent\">subscriptions</mat-icon>\r\n                  </button>\r\n                </td>\r\n                <td>{{ operation.id }}</td>\r\n                <td>{{ operation.type | flatLookup : opTypeMap }}</td>\r\n                <td>{{ operation.at }}</td>\r\n                <td>{{ operation.run }}</td>\r\n                <td>{{ operation.value }}</td>\r\n                <td>{{ operation.inputTag }}</td>\r\n                <td>{{ operation.outputTag }}</td>\r\n                <td>{{ operation.groupId }}</td>\r\n                <td>{{ operation.features?.length }}</td>\r\n              </tr>\r\n              }\r\n            </tbody>\r\n          </table>\r\n          }\r\n\r\n          <!-- operation editor -->\r\n          <mat-expansion-panel [expanded]=\"editedOp\" [disabled]=\"!editedOp\">\r\n            <mat-expansion-panel-header>\r\n              <mat-panel-title>operation {{ editedOp?.id }}</mat-panel-title>\r\n            </mat-expansion-panel-header>\r\n            <fieldset>\r\n              <gve-chain-operation-editor\r\n                [operation]=\"editedOp\"\r\n                (operationCancel)=\"onOperationCancel()\"\r\n                (operationChange)=\"onOperationChange($event)\"\r\n                (operationPreview)=\"onOperationPreview($event)\"\r\n              />\r\n            </fieldset>\r\n          </mat-expansion-panel>\r\n\r\n          <div>\r\n            <mat-progress-bar mode=\"indeterminate\" *ngIf=\"busy\" />\r\n          </div>\r\n\r\n          <!-- opStyle -->\r\n          <div id=\"opStyle\">\r\n            <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n              <mat-label>operations style</mat-label>\r\n              <textarea\r\n                matInput\r\n                [formControl]=\"opStyle\"\r\n                placeholder=\"operations style\"\r\n              ></textarea>\r\n            </mat-form-field>\r\n          </div>\r\n\r\n          <!-- batch ops -->\r\n          <div>\r\n            <mat-expansion-panel>\r\n              <mat-expansion-panel-header>\r\n                <mat-panel-title>batch operations</mat-panel-title>\r\n              </mat-expansion-panel-header>\r\n              <fieldset>\r\n                <div id=\"batch-container\">\r\n                  <div id=\"batch-input\">\r\n                    <div>\r\n                      <mat-form-field class=\"full-width\">\r\n                        <mat-label>operations</mat-label>\r\n                        <textarea\r\n                          class=\"code\"\r\n                          matInput\r\n                          [formControl]=\"inputOps\"\r\n                          placeholder=\"operations\"\r\n                          rows=\"8\"\r\n                          spellcheck=\"false\"\r\n                        ></textarea>\r\n                      </mat-form-field>\r\n                    </div>\r\n                    <div class=\"form-row\">\r\n                      <button\r\n                        type=\"button\"\r\n                        color=\"warn\"\r\n                        class=\"mat-warn\"\r\n                        mat-flat-button\r\n                        matTooltip=\"Remove all the operations\"\r\n                        [disabled]=\"!operations.value.length || busy\"\r\n                        (click)=\"clearOperations()\"\r\n                      >\r\n                        clear\r\n                      </button>\r\n                      <button\r\n                        type=\"button\"\r\n                        color=\"primary\"\r\n                        class=\"mat-primary\"\r\n                        mat-flat-button\r\n                        matTooltip=\"Parse text and add results to the operations\"\r\n                        [disabled]=\"!inputOps.value || busy\"\r\n                        (click)=\"parseOperations()\"\r\n                      >\r\n                        batch add\r\n                      </button>\r\n                      @if (parseError) {\r\n                      <span class=\"error\">{{ parseError }}</span>\r\n                      }\r\n                    </div>\r\n                  </div>\r\n                  <div id=\"batch-help\">\r\n                    <ul>\r\n                      <li>\r\n                        <strong>replace</strong>:\r\n                        <code>ATxRUN<strong>=</strong>\"VALUE\"</code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>delete</strong>:\r\n                        <code>ATxRUN<strong>-</strong></code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>add-before</strong>:\r\n                        <code>ATxRUN<strong>+[</strong>\"VALUE\"</code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>add-after</strong>:\r\n                        <code>ATxRUN<strong>+]</strong>\"VALUE\"</code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>move-before</strong>:\r\n                        <code>ATxRUN<strong>&gt;[</strong>TO</code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>move-after</strong>:\r\n                        <code>ATxRUN<strong>&gt;]</strong>TO</code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>swap</strong>:\r\n                        <code>ATxRUN<strong>&lt;&gt;</strong>TOxRUN</code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>annotate</strong>:\r\n                        <code>ATxRUN<strong>:</strong></code>\r\n                      </li>\r\n                    </ul>\r\n                    <p>For all the operations:</p>\r\n                    <ul>\r\n                      <li>\r\n                        prefix <code>(ITAG:OTAG)</code> to define input and/or\r\n                        output tags, separated by colon.\r\n                      </li>\r\n                      <li>\r\n                        prepend <code>&#x40;</code> to AT to use character\r\n                        indexes (0-N) rather than IDs.\r\n                      </li>\r\n                      <li>RUN (where applicable) defaults to 1.</li>\r\n                      <li>\r\n                        append features like <code>[NAME OPERATOR VALUE]</code>,\r\n                        separated by space, where:\r\n                      </li>\r\n                      <ol>\r\n                        <li>\r\n                          NAME is an arbitrary string. Prefixes:\r\n                          <code>*</code>=global features, <code>!</code>=remove\r\n                          feature (no value). Suffixes:\r\n                          <code>^</code>=short-lived (like\r\n                          <code>*version^=alpha</code>).\r\n                        </li>\r\n                        <li>\r\n                          OPERATOR is <code>=</code> (multiple),\r\n                          <code>:=</code> (single),\r\n                          <code>==</code> (first-single).\r\n                        </li>\r\n                        <li>\r\n                          VALUE is a string. If it includes spaces, wrap it in\r\n                          <code>\"\"</code>.\r\n                        </li>\r\n                      </ol>\r\n                    </ul>\r\n                  </div>\r\n                </div>\r\n              </fieldset>\r\n            </mat-expansion-panel>\r\n          </div>\r\n        </div>\r\n\r\n        <!-- result -->\r\n        @if (result) {\r\n        <fieldset id=\"result\">\r\n          <legend>result</legend>\r\n          <gve-chain-result-view\r\n            [result]=\"result\"\r\n            (stepPick)=\"onStepPick($event)\"\r\n          />\r\n        </fieldset>\r\n        }\r\n      </div>\r\n    </mat-tab>\r\n\r\n    <!-- image -->\r\n    <mat-tab label=\"image\">\r\n      <div id=\"image\">\r\n        <div id=\"image-ctl\">\r\n          <!-- url -->\r\n          <mat-form-field class=\"long-text\">\r\n            <mat-label>URL</mat-label>\r\n            <input matInput [formControl]=\"imageUrl\" />\r\n          </mat-form-field>\r\n          <div class=\"form-row\">\r\n            <!-- x -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>X</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"imageX\"\r\n                placeholder=\"X\"\r\n              />\r\n            </mat-form-field>\r\n            <!-- y -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>Y</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"imageY\"\r\n                placeholder=\"Y\"\r\n              />\r\n            </mat-form-field>\r\n          </div>\r\n          <div class=\"form-row\">\r\n            <!-- width -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>width</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                min=\"0\"\r\n                [formControl]=\"imageWidth\"\r\n              />\r\n            </mat-form-field>\r\n            <!-- height -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>height</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                min=\"0\"\r\n                [formControl]=\"imageHeight\"\r\n              />\r\n            </mat-form-field>\r\n          </div>\r\n          <!-- defs -->\r\n          <div>\r\n            <mat-form-field class=\"long-text\">\r\n              <mat-label>defs</mat-label>\r\n              <textarea matInput [formControl]=\"defs\" rows=\"3\"></textarea>\r\n            </mat-form-field>\r\n          </div>\r\n        </div>\r\n        <div id=\"image-view\">\r\n          @if (imageUrl.value) {\r\n          <img alt=\"background\" [src]=\"imageUrl.value\" width=\"600\" />\r\n          }\r\n        </div>\r\n      </div>\r\n    </mat-tab>\r\n\r\n    <!-- timelines -->\r\n    <mat-tab label=\"animation\">\r\n      <gve-animation-timeline-set\r\n        [tags]=\"opTags\"\r\n        [elementIds]=\"opElementIds\"\r\n        [timelines]=\"timelines.value\"\r\n        (timelinesChange)=\"onTimelinesChange($event)\"\r\n      />\r\n    </mat-tab>\r\n  </mat-tab-group>\r\n\r\n  <!-- preview -->\r\n  @if (snapshot) {\r\n  <fieldset id=\"preview\">\r\n    <legend class=\"button-row\">\r\n      <span>{{ viewTitle }}</span>\r\n      <button\r\n        color=\"primary\"\r\n        mat-icon-button\r\n        type=\"button\"\r\n        matTooltip=\"Refresh preview\"\r\n        (click)=\"updateViewData()\"\r\n      >\r\n        <mat-icon class=\"mat-primary\">refresh</mat-icon>\r\n      </button>\r\n    </legend>\r\n    <gve-snapshot-view\r\n      [data]=\"viewData\"\r\n      (snapshotRender)=\"onSnapshotRender($any($event))\"\r\n      (visualEvent)=\"onVisualEvent($any($event))\"\r\n    />\r\n    <div>\r\n      <mat-button-toggle\r\n        type=\"button\"\r\n        mat-icon-button\r\n        color=\"primary\"\r\n        matTooltip=\"Toggle rulers\"\r\n        [checked]=\"rulers\"\r\n        (change)=\"toggleRulers()\"\r\n      >\r\n        <mat-icon class=\"mat-primary\">straighten</mat-icon>\r\n      </mat-button-toggle>\r\n    </div>\r\n    @if (visualInfo) {\r\n    <div id=\"visual-info\">{{ visualInfo }}</div>\r\n    }\r\n  </fieldset>\r\n  }\r\n\r\n  <!--buttons -->\r\n  <div class=\"form-row-center\">\r\n    <button\r\n      type=\"button\"\r\n      class=\"mat-warn\"\r\n      mat-flat-button\r\n      matTooltip=\"Discard changes\"\r\n      (click)=\"close()\"\r\n    >\r\n      <mat-icon>clear</mat-icon>\r\n      close\r\n    </button>\r\n    @if (!noSave) {\r\n    <button\r\n      type=\"submit\"\r\n      class=\"mat-primary\"\r\n      mat-flat-button\r\n      matTooltip=\"Save changes\"\r\n      [disabled]=\"form.invalid || form.pristine\"\r\n    >\r\n      <mat-icon>check_circle</mat-icon>\r\n      save\r\n    </button>\r\n    }\r\n  </div>\r\n</form>\r\n"]}
870
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snapshot-editor.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/gve-core/src/lib/components/snapshot-editor/snapshot-editor.component.ts","../../../../../../../projects/myrmidon/gve-core/src/lib/components/snapshot-editor/snapshot-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  CUSTOM_ELEMENTS_SCHEMA,\r\n  Component,\r\n  ElementRef,\r\n  EventEmitter,\r\n  Input,\r\n  Output,\r\n  ViewChild,\r\n} from '@angular/core';\r\nimport {\r\n  FormBuilder,\r\n  FormControl,\r\n  FormGroup,\r\n  FormsModule,\r\n  ReactiveFormsModule,\r\n  Validators,\r\n} from '@angular/forms';\r\nimport { CommonModule } from '@angular/common';\r\n\r\nimport { MatButtonModule } from '@angular/material/button';\r\nimport { MatButtonToggleModule } from '@angular/material/button-toggle';\r\nimport { MatCheckboxModule } from '@angular/material/checkbox';\r\nimport { MatExpansionModule } from '@angular/material/expansion';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatIconModule } from '@angular/material/icon';\r\nimport { MatInputModule } from '@angular/material/input';\r\nimport { MatProgressBarModule } from '@angular/material/progress-bar';\r\nimport { MatSnackBar, MatSnackBarModule } from '@angular/material/snack-bar';\r\nimport { MatTabsModule } from '@angular/material/tabs';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\nimport { MatDialog } from '@angular/material/dialog';\r\n\r\nimport { customAlphabet } from 'nanoid';\r\n\r\nimport { NgToolsModule, deepCopy } from '@myrmidon/ng-tools';\r\nimport { DialogService } from '@myrmidon/ng-mat-tools';\r\n\r\nimport {\r\n  CharChainOperation,\r\n  CharNode,\r\n  DEFAULT_SVG_BASE_TEXT_OPTIONS,\r\n  Feature,\r\n  GveAnimationTimeline,\r\n  GveVisualEvent,\r\n  Snapshot,\r\n  SnapshotViewComponent,\r\n  SnapshotViewData,\r\n  SnapshotViewRenderEvent,\r\n  SnapshotViewService,\r\n} from '@myrmidon/gve-snapshot-view';\r\n\r\nimport {\r\n  BaseTextViewComponent,\r\n  VarBaseTextRange,\r\n} from '../base-text-view/base-text-view.component';\r\nimport { ChainOperationEditorComponent } from '../chain-operation-editor/chain-operation-editor.component';\r\nimport {\r\n  ChainOperationContextStep,\r\n  CharChainResult,\r\n  GveApiService,\r\n} from '../../services/gve-api.service';\r\nimport { ChainResultViewComponent } from '../chain-result-view/chain-result-view.component';\r\nimport { LnHeightsEditorComponent } from '../ln-heights-editor/ln-heights-editor.component';\r\nimport { BaseTextEditorComponent } from '../base-text-editor/base-text-editor.component';\r\nimport { AnimationTimelineSetComponent } from '../animation-timeline-set/animation-timeline-set.component';\r\nimport { BatchOperationEditorComponent } from '../batch-operation-editor/batch-operation-editor.component';\r\n\r\n// default snapshot view title\r\nconst VIEW_TITLE = 'preview';\r\n\r\n// suffix for IDs of SVG elements to be made transparent at first rendition\r\nconst SVG_TRANSP_SUFFIX = '_t';\r\n\r\n/**\r\n * 🔑 `gve-snapshot-editor`\r\n *\r\n * A component to edit a text snapshot. This is the top level component in the GVE core\r\n * library.\r\n *\r\n * - ▶️ `snapshot` (`Snapshot`): the snapshot to edit.\r\n * - ▶️ `batchOps` (`string`): the batch operations text to set for the editor.\r\n * - ▶️ `noSave` (`boolean`): true to disable saving.\r\n * - 🔥 `snapshotChange` (`Snapshot`): emitted when the user saves the edited snapshot.\r\n * - 🔥 `snapshotCancel` (`void`): emitted when the user cancels the snapshot editing.\r\n */\r\n@Component({\r\n  selector: 'gve-snapshot-editor',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    FormsModule,\r\n    ReactiveFormsModule,\r\n    MatButtonModule,\r\n    MatButtonToggleModule,\r\n    MatCheckboxModule,\r\n    MatExpansionModule,\r\n    MatFormFieldModule,\r\n    MatIconModule,\r\n    MatInputModule,\r\n    MatProgressBarModule,\r\n    MatSnackBarModule,\r\n    MatTabsModule,\r\n    MatTooltipModule,\r\n    NgToolsModule,\r\n    BaseTextEditorComponent,\r\n    BaseTextViewComponent,\r\n    ChainOperationEditorComponent,\r\n    ChainResultViewComponent,\r\n    LnHeightsEditorComponent,\r\n    AnimationTimelineSetComponent,\r\n  ],\r\n  schemas: [CUSTOM_ELEMENTS_SCHEMA],\r\n  templateUrl: './snapshot-editor.component.html',\r\n  styleUrl: './snapshot-editor.component.css',\r\n})\r\nexport class SnapshotEditorComponent {\r\n  private readonly _nanoid = customAlphabet('1234567890abcdef', 10);\r\n  private _snapshot?: Snapshot;\r\n  private _renderer?: SnapshotViewService;\r\n  private _previewing?: boolean;\r\n  private _transparentIds?: string[];\r\n  private _stepPickFrozen?: boolean;\r\n\r\n  @ViewChild('snapshotView', { static: false })\r\n  public snapshotView?: ElementRef<SnapshotViewComponent>;\r\n\r\n  /**\r\n   * The snapshot to edit.\r\n   */\r\n  @Input()\r\n  public get snapshot(): Snapshot | undefined {\r\n    return this._snapshot;\r\n  }\r\n  public set snapshot(value: Snapshot | undefined | null) {\r\n    if (this._snapshot === value) {\r\n      return;\r\n    }\r\n    this._snapshot = value || undefined;\r\n    this.updateForm(this._snapshot);\r\n    setTimeout(() => {\r\n      this.runToLast();\r\n    }, 0);\r\n  }\r\n\r\n  /**\r\n   * The batch operations text to set for the editor.\r\n   */\r\n  @Input()\r\n  public batchOps?: string;\r\n\r\n  /**\r\n   * True to disable saving.\r\n   */\r\n  @Input()\r\n  public noSave?: boolean;\r\n\r\n  /**\r\n   * True to enable debug mode for view rendition.\r\n   */\r\n  @Input()\r\n  public debug?: boolean;\r\n\r\n  /**\r\n   * Emitted when the user saves the edited snapshot.\r\n   */\r\n  @Output()\r\n  public readonly snapshotChange: EventEmitter<Snapshot> = new EventEmitter();\r\n\r\n  /**\r\n   * Emitted when the user cancels the snapshot editing.\r\n   */\r\n  @Output()\r\n  public readonly snapshotCancel = new EventEmitter<void>();\r\n\r\n  // general\r\n  public width: FormControl<number>;\r\n  public height: FormControl<number>;\r\n  public style: FormControl<string | null>;\r\n  // base text\r\n  public baseText: FormControl<CharNode[] | string>;\r\n  public offsetX: FormControl<number>;\r\n  public offsetY: FormControl<number>;\r\n  public lineHeightOffset: FormControl<number>;\r\n  public lnHeights: FormControl<Record<number, number> | null>;\r\n  public charSpacingOffset: FormControl<number>;\r\n  public spcWidthOffset: FormControl<number>;\r\n  public textStyle: FormControl<string | null>;\r\n  // operations\r\n  public operations: FormControl<CharChainOperation[]>;\r\n  public opStyle: FormControl<string | null>;\r\n  public form: FormGroup;\r\n  // image and defs\r\n  public imageUrl: FormControl<string | null>;\r\n  public imageOpacity: FormControl<number>;\r\n  public imageX: FormControl<number>;\r\n  public imageY: FormControl<number>;\r\n  public imageWidth: FormControl<number>;\r\n  public imageHeight: FormControl<number>;\r\n  public defs: FormControl<string | null>;\r\n  // timelines\r\n  public timelines: FormControl<GveAnimationTimeline[]>;\r\n  // list of operations output tags\r\n  public opTags: string[] = [];\r\n  // list of operation diplomatic.g element IDs\r\n  public opElementIds: string[] = [];\r\n\r\n  // the currently picked base text range\r\n  public textRange?: VarBaseTextRange;\r\n  // the lines count for the current base text\r\n  public lineCount = 0;\r\n\r\n  // the currently edited operation\r\n  public editedOp?: CharChainOperation;\r\n  public editedOpIndex = -1;\r\n\r\n  public busy?: boolean;\r\n  public parseError?: string;\r\n  public opTypeMap = {\r\n    0: 'replace',\r\n    1: 'delete',\r\n    2: 'add-before',\r\n    3: 'add-after',\r\n    4: 'move-before',\r\n    5: 'move-after',\r\n    6: 'swap',\r\n    7: 'annotate',\r\n  };\r\n\r\n  // snapshot view data\r\n  public viewTitle = VIEW_TITLE;\r\n  public viewData?: SnapshotViewData;\r\n  public visualInfo?: string;\r\n  public rulers = true;\r\n\r\n  // operations execution result\r\n  public result?: CharChainResult;\r\n  public resultOperationId?: string;\r\n  public initialStepIndex = -1;\r\n\r\n  constructor(\r\n    formBuilder: FormBuilder,\r\n    private _api: GveApiService,\r\n    private _dialog: MatDialog,\r\n    private _dialogService: DialogService,\r\n    private _snackbar: MatSnackBar\r\n  ) {\r\n    // general\r\n    this.width = new FormControl<number>(800, { nonNullable: true });\r\n    this.height = new FormControl<number>(600, { nonNullable: true });\r\n    this.style = new FormControl<string | null>(null);\r\n    // base text\r\n    this.baseText = new FormControl<CharNode[] | string>([], {\r\n      nonNullable: true,\r\n      validators: [Validators.required],\r\n    });\r\n    this.offsetX = new FormControl<number>(0, { nonNullable: true });\r\n    this.offsetY = new FormControl<number>(0, { nonNullable: true });\r\n    this.lineHeightOffset = new FormControl<number>(\r\n      DEFAULT_SVG_BASE_TEXT_OPTIONS.lineHeightOffset,\r\n      { nonNullable: true }\r\n    );\r\n    this.lnHeights = new FormControl<Record<number, number> | null>(null);\r\n    this.charSpacingOffset = new FormControl<number>(0, { nonNullable: true });\r\n    this.spcWidthOffset = new FormControl<number>(0, { nonNullable: true });\r\n    this.textStyle = new FormControl<string | null>(null);\r\n\r\n    // operations\r\n    this.operations = new FormControl<CharChainOperation[]>([], {\r\n      nonNullable: true,\r\n    });\r\n    this.opStyle = new FormControl<string | null>(null);\r\n    // image\r\n    this.imageUrl = new FormControl<string | null>(null);\r\n    this.imageOpacity = new FormControl<number>(1, { nonNullable: true });\r\n    this.imageX = new FormControl<number>(0, { nonNullable: true });\r\n    this.imageY = new FormControl<number>(0, { nonNullable: true });\r\n    this.imageWidth = new FormControl<number>(0, { nonNullable: true });\r\n    this.imageHeight = new FormControl<number>(0, { nonNullable: true });\r\n    this.defs = new FormControl<string | null>(null);\r\n\r\n    // timelines\r\n    this.timelines = new FormControl<GveAnimationTimeline[]>([], {\r\n      nonNullable: true,\r\n    });\r\n\r\n    // form\r\n    this.form = formBuilder.group({\r\n      width: this.width,\r\n      height: this.height,\r\n      style: this.style,\r\n      baseText: this.baseText,\r\n      offsetX: this.offsetX,\r\n      offsetY: this.offsetY,\r\n      lineHeightOffset: this.lineHeightOffset,\r\n      lnHeights: this.lnHeights,\r\n      charSpacingOffset: this.charSpacingOffset,\r\n      spcWidthOffset: this.spcWidthOffset,\r\n      textStyle: this.textStyle,\r\n      operations: this.operations,\r\n      opStyle: this.opStyle,\r\n      // image\r\n      imageUrl: this.imageUrl,\r\n      imageOpacity: this.imageOpacity,\r\n      imageX: this.imageX,\r\n      imageY: this.imageY,\r\n      imageWidth: this.imageWidth,\r\n      imageHeight: this.imageHeight,\r\n      defs: this.defs,\r\n      // timelines\r\n      timelines: this.timelines,\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Set the view data for the snapshot view.\r\n   *\r\n   * @param snapshot The optional snapshot data to use. If not provided,\r\n   * a snapshot will be created from the form data.\r\n   * @param title The optional title to set for the view.\r\n   */\r\n  private setViewData(snapshot?: Snapshot, title = VIEW_TITLE): void {\r\n    console.log('set view data');\r\n\r\n    // get the snapshot to use\r\n    if (!snapshot) {\r\n      snapshot = this.getSnapshot();\r\n    }\r\n\r\n    // update view data\r\n    this.viewTitle = title;\r\n    this.visualInfo = undefined;\r\n    this.viewData = {\r\n      snapshot: snapshot,\r\n      options: {\r\n        debug: this.debug,\r\n        delayedRender: true,\r\n        showRulers: true,\r\n        showGrid: true,\r\n        panZoom: true,\r\n        transparentIds: this._transparentIds,\r\n      },\r\n    };\r\n\r\n    console.log('view data: ', this.viewData);\r\n  }\r\n\r\n  private updateForm(snapshot: Snapshot | undefined) {\r\n    this._transparentIds = undefined;\r\n    this.initialStepIndex = -1;\r\n\r\n    if (!snapshot) {\r\n      this.form.reset();\r\n    } else {\r\n      this.width.setValue(snapshot.size.width);\r\n      this.height.setValue(snapshot.size.height);\r\n      this.style.setValue(snapshot.style || null);\r\n      this.imageUrl.setValue(snapshot.image?.url || null);\r\n      this.imageOpacity.setValue(snapshot.image?.opacity || 0);\r\n      this.imageX.setValue(snapshot.image?.canvas?.x || 0);\r\n      this.imageY.setValue(snapshot.image?.canvas?.y || 0);\r\n      this.imageWidth.setValue(snapshot.image?.canvas?.width || 0);\r\n      this.imageHeight.setValue(snapshot.image?.canvas?.height || 0);\r\n      this.defs.setValue(snapshot.defs || null);\r\n      this.baseText.setValue(snapshot.text);\r\n      this.offsetX.setValue(snapshot.textOptions?.offset?.x || 0);\r\n      this.offsetY.setValue(snapshot.textOptions?.offset?.y || 0);\r\n      this.lineHeightOffset.setValue(snapshot.textOptions?.lineHeightOffset);\r\n      this.lnHeights.setValue(snapshot.textOptions?.minLineHeights || null);\r\n      this.charSpacingOffset.setValue(snapshot.textOptions?.charSpacingOffset);\r\n      this.spcWidthOffset.setValue(snapshot.textOptions?.spcWidthOffset);\r\n      this.textStyle.setValue(snapshot.textStyle || null);\r\n      this.operations.setValue(snapshot.operations);\r\n      this.opStyle.setValue(snapshot.opStyle || null);\r\n      // timelines\r\n      const timelines: GveAnimationTimeline[] = [];\r\n      if (snapshot.timelines) {\r\n        for (let tag in snapshot.timelines) {\r\n          timelines.push(snapshot.timelines[tag]);\r\n        }\r\n      }\r\n      this.timelines.setValue(timelines);\r\n    }\r\n    this.updateLineCount(this.baseText.value as CharNode[]);\r\n    this.updateOpLists();\r\n\r\n    this.form.markAsPristine();\r\n  }\r\n\r\n  // #region base text\r\n  /**\r\n   * Update the line count based on the received text.\r\n   * @param text The text to use for counting lines.\r\n   */\r\n  private updateLineCount(text: CharNode[]) {\r\n    if (!text.length) {\r\n      this.lineCount = 0;\r\n    } else {\r\n      this.lineCount = text.filter((c) => c.data === '\\n').length + 1;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Handle the event fired by the base text editor to pick a text range.\r\n   * @param range The picked range.\r\n   */\r\n  public onRangePick(range: VarBaseTextRange) {\r\n    this.textRange = range;\r\n  }\r\n\r\n  /**\r\n   * Handle the event fired by the base text editor to change the base text.\r\n   * @param text The text to set.\r\n   */\r\n  public onTextChange(text: CharNode[]) {\r\n    console.log('text change', text);\r\n\r\n    // update the form control and reset the current text range\r\n    this.baseText.setValue(text);\r\n    this.baseText.updateValueAndValidity();\r\n    this.baseText.markAsDirty();\r\n    this.textRange = undefined;\r\n\r\n    // update the line count\r\n    this.updateLineCount(text);\r\n\r\n    // remove all operations and update the view data\r\n    this.removeAllOperations();\r\n\r\n    // remove all timelines\r\n    this.timelines.reset();\r\n  }\r\n  // #endregion\r\n\r\n  // #region operations\r\n  /**\r\n   * Update the lists of operation output tags and element IDs by collecting\r\n   * all the operation tags and the IDs of the elements in their diplomatic.g\r\n   * SVG code if any.\r\n   */\r\n  private updateOpLists() {\r\n    const tags = new Set<string>();\r\n    const ids = new Set<string>();\r\n\r\n    let n = 0;\r\n    for (const op of this.operations.value) {\r\n      // output tag\r\n      n++;\r\n      tags.add(op.outputTag || `v${n}`);\r\n\r\n      // element IDs\r\n      if (op.diplomatics?.g) {\r\n        this.parseSvgIds(op.diplomatics.g)?.forEach((id) => ids.add(id));\r\n      }\r\n    }\r\n\r\n    this.opTags = [...tags];\r\n    this.opElementIds = [...ids];\r\n  }\r\n\r\n  /**\r\n   * Edit a new operation.\r\n   */\r\n  public editNewOperation() {\r\n    // create a new operation and edit it\r\n    this.editedOp = {\r\n      id: this._nanoid(),\r\n      type: 0,\r\n      at: this.textRange?.at || 0,\r\n      run: this.textRange?.run || 1,\r\n    };\r\n    this.editedOpIndex = -1;\r\n  }\r\n\r\n  /**\r\n   * Edit (a copy of) the operation at the specified index.\r\n   * @param index The operation index.\r\n   */\r\n  public editOperation(index: number) {\r\n    this.editedOpIndex = index;\r\n    this.editedOp = deepCopy(this.operations.value[index]);\r\n  }\r\n\r\n  /**\r\n   * Close the currently edited operation.\r\n   */\r\n  private closeEditedOperation() {\r\n    this.editedOpIndex = -1;\r\n    this.editedOp = undefined;\r\n  }\r\n\r\n  /**\r\n   * Handle the event fired by the operation editor to cancel\r\n   * the current operation edits.\r\n   */\r\n  public onOperationCancel() {\r\n    this.closeEditedOperation();\r\n  }\r\n\r\n  /**\r\n   * Handle the event fired by the operation editor to change\r\n   * the currently edited operation, or add a new one if that\r\n   * was a new operation.\r\n   * @param op The changed operation.\r\n   */\r\n  public async onOperationChange(op: CharChainOperation) {\r\n    console.log('operation change');\r\n    const operations = [...this.operations.value];\r\n\r\n    // replace or add the operation\r\n    let i = this.editedOpIndex;\r\n    if (this.editedOpIndex > -1) {\r\n      operations.splice(this.editedOpIndex, 1, op);\r\n    } else {\r\n      operations.push(op);\r\n      i = operations.length - 1;\r\n    }\r\n\r\n    // update the operations form control\r\n    this.operations.setValue(operations);\r\n    this.operations.markAsDirty();\r\n    this.operations.updateValueAndValidity();\r\n\r\n    // update the operation lists\r\n    this.updateOpLists();\r\n\r\n    // update the view data\r\n    this.result = await this.runTo(i);\r\n\r\n    // close the edited operation\r\n    this.closeEditedOperation();\r\n  }\r\n\r\n  /**\r\n   * Delete the operation at the specified index.\r\n   * @param index The index of the operation to delete.\r\n   */\r\n  public deleteOperation(index: number) {\r\n    this._dialogService\r\n      .confirm(\r\n        'Confirm Deletion',\r\n        `Delete operation \"${this.operations.value[index].id}\"?`\r\n      )\r\n      .subscribe((yes: boolean) => {\r\n        if (yes) {\r\n          // close the edited operation if it is the one being deleted\r\n          if (this.editedOpIndex === index) {\r\n            this.closeEditedOperation();\r\n          }\r\n          // reset the result operation ID if it is the one being deleted\r\n          if (this.resultOperationId === this.operations.value[index].id) {\r\n            this.resultOperationId = undefined;\r\n          }\r\n          // delete the operation and update the form control\r\n          const operations = [...this.operations.value];\r\n          operations.splice(index, 1);\r\n          this.operations.setValue(operations);\r\n          this.operations.markAsDirty();\r\n          this.operations.updateValueAndValidity();\r\n\r\n          // update the operation lists\r\n          this.updateOpLists();\r\n\r\n          // update the view data\r\n          setTimeout(() => {\r\n            this.runToLast();\r\n          }, 0);\r\n        }\r\n      });\r\n  }\r\n\r\n  /**\r\n   * Parse the operations from their text and append them to the current\r\n   * snapshot operations.\r\n   */\r\n  public parseOperations() {\r\n    if (this.busy) {\r\n      return;\r\n    }\r\n\r\n    const dialogRef = this._dialog.open(BatchOperationEditorComponent, {\r\n      data: { preset: this.batchOps },\r\n    });\r\n\r\n    dialogRef.afterClosed().subscribe((batchOps: CharChainOperation[]) => {\r\n      if (batchOps) {\r\n        const operations = [...this.operations.value];\r\n        operations.push(...batchOps);\r\n        this.operations.setValue(operations);\r\n        this.operations.markAsDirty();\r\n        this.operations.updateValueAndValidity();\r\n        // update the operation lists\r\n        this.updateOpLists();\r\n        // update the view data\r\n        setTimeout(() => {\r\n          this.runToLast();\r\n        }, 0);\r\n      }\r\n    });\r\n  }\r\n\r\n  private removeAllOperations(): void {\r\n    this.resultOperationId = undefined;\r\n    this.closeEditedOperation();\r\n    this.operations.reset();\r\n    this.operations.markAsDirty();\r\n    this.operations.updateValueAndValidity();\r\n    this.opTags = [];\r\n    this.opElementIds = [];\r\n    this.setViewData();\r\n    this.result = undefined;\r\n  }\r\n\r\n  /**\r\n   * Remove all the operations.\r\n   */\r\n  public clearOperations(): void {\r\n    this._dialogService\r\n      .confirm('Confirm', 'Remove all operations?')\r\n      .subscribe((yes: boolean) => {\r\n        if (yes) {\r\n          this.removeAllOperations();\r\n        }\r\n      });\r\n  }\r\n\r\n  /**\r\n   * Parse all the id attributes from the received SVG code and\r\n   * return them as an array.\r\n   *\r\n   * @param svg The SVG content to parse or undefined.\r\n   * @returns Array of IDs found in the SVG content or undefined.\r\n   */\r\n  private parseSvgIds(svg?: string): string[] | undefined {\r\n    if (!svg) {\r\n      return undefined;\r\n    }\r\n\r\n    const regex = /<[^>]+\\bid=([\"'])(.*?)\\1/g;\r\n    const ids: string[] = [];\r\n    let match: RegExpExecArray | null;\r\n\r\n    while ((match = regex.exec(svg)) !== null) {\r\n      ids.push(match[2]);\r\n    }\r\n\r\n    return ids;\r\n  }\r\n\r\n  private getTransparentIds(g?: string): string[] | undefined {\r\n    if (!g) {\r\n      return undefined;\r\n    }\r\n    return this.parseSvgIds(g)?.filter((id) => id.endsWith(SVG_TRANSP_SUFFIX));\r\n  }\r\n\r\n  /**\r\n   * Run the operations up to the specified index (included).\r\n   * This is called when:\r\n   * - a preview is requested by the operation editor.\r\n   * - the currently edited operation is saved.\r\n   * - the user picks a step in the chain result view.\r\n   * - runToLast is called, which happens when:\r\n   *   - setting the snapshot.\r\n   *   - parsing a batch of operations.\r\n   *   - deleting an operation.\r\n   *\r\n   * @param index The index of the operation to run to.\r\n   * @param lastOperation The operation to use in place of the existing\r\n   * operation in the snapshot at index. This is used when previewing\r\n   * the edited operation from within the operation editor.\r\n   */\r\n  public runTo(\r\n    index: number,\r\n    lastOperation?: CharChainOperation\r\n  ): Promise<CharChainResult> {\r\n    return new Promise((resolve, reject) => {\r\n      if (this.busy) {\r\n        return reject('Busy');\r\n      }\r\n      console.log('run to: ' + index);\r\n\r\n      // get the snapshot (=text and ops) to run operations on\r\n      const snapshot = this.getSnapshot();\r\n      if (!snapshot.text) {\r\n        return reject('No snapshot text');\r\n      }\r\n\r\n      // remove from the snapshot the operations past the specified index,\r\n      // also replacing the last operation when this was received as a parameter\r\n      snapshot.operations = [...this.operations.value];\r\n\r\n      if (lastOperation) {\r\n        snapshot.operations = snapshot.operations.slice(0, index);\r\n        snapshot.operations.push(lastOperation);\r\n      } else {\r\n        snapshot.operations = snapshot.operations.slice(0, index + 1);\r\n      }\r\n\r\n      // run the operations\r\n      this.busy = true;\r\n      this.initialStepIndex = index;\r\n      this._api\r\n        .runOperations(snapshot.text as CharNode[], snapshot.operations)\r\n        .subscribe({\r\n          next: (wrapper) => {\r\n            // handle operation (non-fatal) error or result\r\n            if (wrapper.error) {\r\n              this._snackbar.open(wrapper.error, 'OK');\r\n              reject(wrapper.error);\r\n              return;\r\n            }\r\n\r\n            // extract the IDs from the last operation's diplomatics and filter\r\n            // them so that only the ones ending with _t are kept. By convention,\r\n            // all the IDs ending with this suffix are to be made invisible at\r\n            // their first rendition (opacity=0). An animation will then make\r\n            // them visible.\r\n            const lastOp = snapshot.operations[snapshot.operations.length - 1];\r\n            this._transparentIds = this.getTransparentIds(\r\n              lastOp.diplomatics?.g\r\n            );\r\n\r\n            // update the view data\r\n            this.setViewData(snapshot, lastOp.outputTag);\r\n\r\n            // return the result\r\n            resolve(wrapper.result!);\r\n          },\r\n          error: (error) => {\r\n            console.error(error);\r\n            this._transparentIds = undefined;\r\n            this._snackbar.open('Error running operations', 'OK');\r\n            reject(error);\r\n          },\r\n          complete: () => {\r\n            this.busy = false;\r\n          },\r\n        });\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Run the operations up to the last operation if any, updating the\r\n   * execution result. The execution result is always the result from\r\n   * all the operations, as users must be able to browse across its steps.\r\n   */\r\n  public async runToLast(): Promise<void> {\r\n    if (this.operations.value.length) {\r\n      this.result = await this.runTo(this.operations.value.length - 1);\r\n    } else {\r\n      this.setViewData();\r\n      this.result = undefined;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Update the snapshot view by running the operations up to the\r\n   * currently edited operation if any.\r\n   *\r\n   * @param operation The operation being previewed.\r\n   */\r\n  public onOperationPreview(operation: CharChainOperation): void {\r\n    // no multiple previews or previewing a new operation\r\n    if (this._previewing || this.editedOpIndex < 0) {\r\n      return;\r\n    }\r\n    this._previewing = true;\r\n    setTimeout(() => {\r\n      this.runTo(this.editedOpIndex, operation);\r\n      this._previewing = false;\r\n    }, 0);\r\n  }\r\n  // #endregion\r\n\r\n  /**\r\n   * Build the snapshot at the specified execution step. This implies adding\r\n   * the nodes introduced by the operations up to the specified step, and\r\n   * updating the features of the nodes introduced by the specified step.\r\n   * Also, the operations after the specified step are removed from the\r\n   * snapshot.\r\n   *\r\n   * @param step The step to build the snapshot at.\r\n   * @returns The snapshot at the specified step.\r\n   */\r\n  private buildSnapshotAtStep(\r\n    step: ChainOperationContextStep\r\n  ): Snapshot | null {\r\n    // no result, nothing to do\r\n    if (!this.result) {\r\n      return null;\r\n    }\r\n\r\n    // get the currently edited snapshot\r\n    const snapshot = this.getSnapshot();\r\n    // find the max ID in snapshot.text (=original) nodes,\r\n    // so that any ID greater than it belongs to new nodes\r\n    const maxOriginalId = (snapshot.text as CharNode[]).reduce(\r\n      (max, n) => Math.max(max, n.id),\r\n      0\r\n    );\r\n    // get a copy of the nodes of the snapshot base text\r\n    const snapNodes: CharNode[] = deepCopy(snapshot.text as CharNode[]);\r\n\r\n    // get the nodes of the picked step\r\n    const stepNodes = this.result.taggedNodes[step.outputTag];\r\n\r\n    // update the snapshot text nodes copies with the picked step nodes,\r\n    // by updating their features if existing, or adding new nodes if not.\r\n    for (const sn of stepNodes) {\r\n      // find the node in the original snapshot text\r\n      const i = snapNodes.findIndex((n) => n.id === sn.id);\r\n      // if found, update its features, else add it\r\n      if (i > -1) {\r\n        snapNodes[i].features = sn.features;\r\n      } else {\r\n        // new node: we assume that it has a manually set position (x,y features)\r\n        snapNodes.push(sn);\r\n      }\r\n    }\r\n\r\n    // append nodes introduced *before* the picked step, if any\r\n    // (those introduced by the picked step have already been added)\r\n    const stepIndex = this.result.steps.indexOf(step);\r\n\r\n    for (let i = 0; i < stepIndex; i++) {\r\n      const stepNodes = this.result.taggedNodes[this.result.steps[i].outputTag];\r\n\r\n      for (const n of stepNodes) {\r\n        if (n.id > maxOriginalId && !snapNodes.find((x) => x.id === n.id)) {\r\n          snapNodes.push(n);\r\n        }\r\n      }\r\n    }\r\n\r\n    // update the snapshot text nodes\r\n    snapshot.text = snapNodes;\r\n\r\n    // remove from the snapshot the operations after the picked step\r\n    const i = snapshot.operations.findIndex((o) => o.id === step.operation.id);\r\n    if (i > -1) {\r\n      snapshot.operations = snapshot.operations.slice(0, i + 1);\r\n    }\r\n\r\n    return snapshot;\r\n  }\r\n\r\n  private playTimeline(tl: GveAnimationTimeline): void {\r\n    const shadowRoot = this.snapshotView?.nativeElement?.shadowRoot;\r\n    if (tl && shadowRoot) {\r\n      console.log('play timeline', tl);\r\n      this._renderer?.playTimeline(tl, undefined, shadowRoot);\r\n    } else {\r\n      if (!this.snapshotView) {\r\n        console.warn('no snapshotView for timeline');\r\n      } else {\r\n        console.warn('no shadowRoot for timeline');\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Handle the event fired by the chain result view to pick a step.\r\n   *\r\n   * @param step The step to pick.\r\n   */\r\n  public async onStepPick(step: ChainOperationContextStep): Promise<void> {\r\n    if (!this.result || this._stepPickFrozen) {\r\n      return;\r\n    }\r\n\r\n    // get the currently edited snapshot\r\n    const snapshot = this.buildSnapshotAtStep(step)!;\r\n    // update result operation ID\r\n    this.resultOperationId = step.operation.id;\r\n    // update transparent IDs\r\n    this._transparentIds = this.getTransparentIds(\r\n      step.operation.diplomatics?.g\r\n    );\r\n    // update view data\r\n    this.setViewData(snapshot, step.outputTag);\r\n\r\n    // play animation if any\r\n    const tl = this.timelines.value.find((t) => t.tag === step.outputTag);\r\n    if (tl) {\r\n      this._stepPickFrozen = true;\r\n\r\n      // find the index of the picked step in the snapshot operations\r\n      const i = snapshot.operations.findIndex(\r\n        (o) => o.id === step.operation.id\r\n      );\r\n      // run the operations up to the picked step as we need to hide\r\n      // those visuals to be revealed by the animation\r\n      await this.runTo(i);\r\n      this._stepPickFrozen = false;\r\n\r\n      // play the timeline\r\n      setTimeout(() => {\r\n        this.playTimeline(tl);\r\n      }, 0);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Handle the event fired by a visual in the snapshot view.\r\n   *\r\n   * @param event The event.\r\n   */\r\n  public onVisualEvent(event: CustomEvent<GveVisualEvent>): void {\r\n    const d = event.detail;\r\n\r\n    if (d.event.type === 'mouseover') {\r\n      const visual = d.source;\r\n      const sb: string[] = [];\r\n      // id (type)\r\n      sb.push('#' + visual.id);\r\n      sb.push(` (${visual.type})`);\r\n      // : label and features\r\n      if (visual.data?.label) {\r\n        sb.push(': ');\r\n        sb.push(visual.data.label);\r\n      }\r\n      if (visual.data?.features?.length) {\r\n        sb.push(' ');\r\n        sb.push(\r\n          visual.data.features\r\n            .map((f: Feature) => `${f.name}=${f.value}`)\r\n            .join('\\n')\r\n        );\r\n      }\r\n      // (@x,y width×height)\r\n      sb.push(\r\n        ` (@${visual.x.toFixed(1)},${visual.y.toFixed(1)} ` +\r\n        `◻${visual.width.toFixed(1)}×${visual.height.toFixed(1)})`\r\n      );\r\n      this.visualInfo = sb.join('');\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Handle the change of line heights by updating the form control.\r\n   *\r\n   * @param heights The line heights.\r\n   */\r\n  public onHeightsChange(heights: Record<number, number> | undefined): void {\r\n    this.lnHeights.setValue(heights || null);\r\n    this.lnHeights.markAsDirty();\r\n    this.lnHeights.updateValueAndValidity();\r\n  }\r\n\r\n  /**\r\n   * Handle the change of timelines by updating the form control.\r\n   *\r\n   * @param timelines The timelines.\r\n   */\r\n  public onTimelinesChange(timelines: GveAnimationTimeline[]): void {\r\n    this.timelines.setValue(timelines);\r\n    this.timelines.markAsDirty();\r\n    this.timelines.updateValueAndValidity();\r\n  }\r\n\r\n  /**\r\n   * Emit the cancel event for this snapshot edit.\r\n   */\r\n  public close(): void {\r\n    this.snapshotCancel.emit();\r\n  }\r\n\r\n  /**\r\n   * Handle the render event from the snapshot view to get a reference\r\n   * to the renderer.\r\n   *\r\n   * @param event The rendition event.\r\n   */\r\n  public onSnapshotRender(event: CustomEvent<SnapshotViewRenderEvent>): void {\r\n    this._renderer = (event.detail as SnapshotViewRenderEvent).renderer;\r\n  }\r\n\r\n  /**\r\n   * Toggle rulers in the snapshot view.\r\n   */\r\n  public toggleRulers(): void {\r\n    if (!this._renderer) {\r\n      return;\r\n    }\r\n    this.rulers = this._renderer.toggleRulers();\r\n  }\r\n\r\n  /**\r\n   * Get a snapshot from the form data.\r\n   *\r\n   * @returns New snapshot object.\r\n   */\r\n  private getSnapshot(): Snapshot {\r\n    const snapshot: Snapshot = {\r\n      size: {\r\n        width: this.width.value,\r\n        height: this.height.value,\r\n      },\r\n      style: this.style.value || undefined,\r\n      defs: this.defs.value || undefined,\r\n      text: this.baseText.value!,\r\n      textStyle: this.textStyle.value || undefined,\r\n      textOptions: {\r\n        lineHeightOffset: this.lineHeightOffset.value,\r\n        minLineHeights: this.lnHeights.value || undefined,\r\n        charSpacingOffset: this.charSpacingOffset.value,\r\n        spcWidthOffset: this.spcWidthOffset.value,\r\n        offset: {\r\n          x: this.offsetX.value,\r\n          y: this.offsetY.value,\r\n        },\r\n      },\r\n      operations: [...this.operations.value],\r\n      opStyle: this.opStyle.value || undefined,\r\n    };\r\n\r\n    // image\r\n    if (this.imageUrl.value) {\r\n      snapshot.image = {\r\n        url: this.imageUrl.value,\r\n        opacity: this.imageOpacity.value || undefined,\r\n      };\r\n      if (\r\n        this.imageX.value ||\r\n        this.imageY.value ||\r\n        this.imageWidth.value ||\r\n        this.imageHeight.value\r\n      ) {\r\n        snapshot.image.canvas = {\r\n          x: this.imageX.value,\r\n          y: this.imageY.value,\r\n          width: this.imageWidth.value,\r\n          height: this.imageHeight.value,\r\n        };\r\n      }\r\n    }\r\n\r\n    // timelines\r\n    if (this.timelines.value.length) {\r\n      snapshot.timelines = {};\r\n      this.timelines.value.forEach((t) => {\r\n        snapshot.timelines![t.tag] = t;\r\n      });\r\n    } else {\r\n      snapshot.timelines = undefined;\r\n    }\r\n\r\n    return snapshot;\r\n  }\r\n\r\n  /**\r\n   * Get a snapshot from the form data and emit it\r\n   * in the snapshot change event.\r\n   */\r\n  public save(): void {\r\n    if (this.form.invalid || this.noSave) {\r\n      return;\r\n    }\r\n    this._snapshot = this.getSnapshot();\r\n    this.snapshotChange.emit(this._snapshot);\r\n  }\r\n}\r\n","<form [formGroup]=\"form\" (submit)=\"save()\">\r\n  <mat-tab-group>\r\n    <!-- text -->\r\n    <mat-tab>\r\n      <ng-template mat-tab-label>\r\n        <mat-icon>article</mat-icon> <span class=\"label\">text</span>\r\n      </ng-template>\r\n      <mat-expansion-panel [expanded]=\"!baseText.value\">\r\n        <mat-expansion-panel-header>\r\n          <mat-panel-title> base text </mat-panel-title>\r\n        </mat-expansion-panel-header>\r\n        <gve-base-text-editor\r\n          [text]=\"baseText.value\"\r\n          (textChange)=\"onTextChange($event)\"\r\n        />\r\n        <fieldset>\r\n          <div class=\"form-row\">\r\n            <!-- offsetX -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>X offset</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"offsetX\"\r\n                placeholder=\"X offset\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- offsetY -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>Y offset</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"offsetY\"\r\n                placeholder=\"Y offset\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- lineHeightOffset -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>ln h-offset</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"lineHeightOffset\"\r\n                placeholder=\"ln h-offset\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- charSpacingOffset -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>char spacing</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"charSpacingOffset\"\r\n                placeholder=\"char spacing\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- spcWidthOffset -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>spc w-offset</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"spcWidthOffset\"\r\n                placeholder=\"spc w-offset\"\r\n              />\r\n            </mat-form-field>\r\n            <!-- minLineHeights -->\r\n            <div class=\"boxed\">\r\n              <gve-ln-heights-editor\r\n                [lineCount]=\"lineCount\"\r\n                [heights]=\"lnHeights.value\"\r\n                (heightsChange)=\"onHeightsChange($event)\"\r\n              />\r\n            </div>\r\n          </div>\r\n          <!-- textStyle -->\r\n          <div>\r\n            <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n              <mat-label>text style</mat-label>\r\n              <textarea\r\n                matInput\r\n                [formControl]=\"textStyle\"\r\n                placeholder=\"text style\"\r\n              ></textarea>\r\n            </mat-form-field>\r\n          </div>\r\n        </fieldset>\r\n      </mat-expansion-panel>\r\n    </mat-tab>\r\n\r\n    <!-- operations -->\r\n    <mat-tab>\r\n      <ng-template mat-tab-label>\r\n        <mat-icon>edit</mat-icon> <span class=\"label\">operations</span>\r\n      </ng-template>\r\n\r\n      <div id=\"snapshot-container\">\r\n        <div id=\"general\">\r\n          <div class=\"form-row\">\r\n            <!-- width -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>width</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                min=\"0\"\r\n                [formControl]=\"width\"\r\n                placeholder=\"width\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- height -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>height</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                min=\"0\"\r\n                [formControl]=\"height\"\r\n                placeholder=\"height\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <div class=\"form-row right\">\r\n              <!-- remove ops -->\r\n              <button\r\n                type=\"button\"\r\n                class=\"mat-warn\"\r\n                mat-flat-button\r\n                matTooltip=\"Remove all the operations\"\r\n                [disabled]=\"!operations.value.length || busy\"\r\n                (click)=\"clearOperations()\"\r\n              >\r\n                <mat-icon>delete_forever</mat-icon> clear\r\n              </button>\r\n\r\n              <!-- batch add ops -->\r\n              <button\r\n                type=\"button\"\r\n                mat-flat-button\r\n                matTooltip=\"Add a batch of operations\"\r\n                class=\"mat-primary\"\r\n                [disabled]=\"busy\"\r\n                (click)=\"parseOperations()\"\r\n              >\r\n                <mat-icon>post_add</mat-icon> batch\r\n              </button>\r\n\r\n              <!-- add op -->\r\n              <button\r\n                type=\"button\"\r\n                mat-flat-button\r\n                class=\"mat-primary\"\r\n                [disabled]=\"!baseText.value\"\r\n                (click)=\"editNewOperation()\"\r\n              >\r\n                <mat-icon>add_circle</mat-icon> operation\r\n              </button>\r\n            </div>\r\n          </div>\r\n        </div>\r\n\r\n        <!-- ops -->\r\n        <div id=\"ops\">\r\n          <!-- operations list -->\r\n          @if (operations.value.length) {\r\n          <table id=\"list\">\r\n            <thead>\r\n              <tr>\r\n                <th></th>\r\n                <th>ID</th>\r\n                <th>type</th>\r\n                <th>at</th>\r\n                <th>run</th>\r\n                <th>value</th>\r\n                <th>itag</th>\r\n                <th>otag</th>\r\n                <th>gid</th>\r\n                <th>feats</th>\r\n              </tr>\r\n            </thead>\r\n            <tbody>\r\n              @for (operation of operations.value; track operation.id; let\r\n              index=$index) {\r\n              <tr\r\n                [ngClass]=\"{ selected: operation.id === resultOperationId }\"\r\n                [ngClass]=\"{ edited: operation.id === editedOp?.id }\"\r\n              >\r\n                <td class=\"fit-width\">\r\n                  <!-- edit -->\r\n                  <button\r\n                    type=\"button\"\r\n                    mat-icon-button\r\n                    class=\"mat-primary\"\r\n                    (click)=\"editOperation(index)\"\r\n                    matTooltip=\"Edit operation\"\r\n                  >\r\n                    <mat-icon>edit</mat-icon>\r\n                  </button>\r\n                  <!-- delete -->\r\n                  <button\r\n                    type=\"button\"\r\n                    mat-icon-button\r\n                    class=\"mat-warn\"\r\n                    (click)=\"deleteOperation(index)\"\r\n                    matTooltip=\"Delete operation\"\r\n                  >\r\n                    <mat-icon class=\"mat-warn\">delete</mat-icon>\r\n                  </button>\r\n                  <!-- run to -->\r\n                  <button\r\n                    type=\"button\"\r\n                    mat-icon-button\r\n                    class=\"mat-accent\"\r\n                    (click)=\"runTo(index)\"\r\n                    matTooltip=\"Run to this operation\"\r\n                  >\r\n                    <mat-icon class=\"mat-accent\">subscriptions</mat-icon>\r\n                  </button>\r\n                </td>\r\n                <td>{{ operation.id }}</td>\r\n                <td>{{ operation.type | flatLookup : opTypeMap }}</td>\r\n                <td>{{ operation.at }}</td>\r\n                <td>{{ operation.run }}</td>\r\n                <td>{{ operation.value }}</td>\r\n                <td>{{ operation.inputTag }}</td>\r\n                <td>{{ operation.outputTag }}</td>\r\n                <td>{{ operation.groupId }}</td>\r\n                <td>{{ operation.features?.length }}</td>\r\n              </tr>\r\n              }\r\n            </tbody>\r\n          </table>\r\n          }\r\n\r\n          <!-- operation editor -->\r\n          <mat-expansion-panel [expanded]=\"editedOp\" [disabled]=\"!editedOp\">\r\n            <mat-expansion-panel-header>\r\n              <mat-panel-title>operation {{ editedOp?.id }}</mat-panel-title>\r\n            </mat-expansion-panel-header>\r\n            <fieldset>\r\n              <gve-chain-operation-editor\r\n                [hidePreview]=\"editedOpIndex === -1\"\r\n                [operation]=\"editedOp\"\r\n                (operationCancel)=\"onOperationCancel()\"\r\n                (operationChange)=\"onOperationChange($event)\"\r\n                (operationPreview)=\"onOperationPreview($event)\"\r\n              />\r\n            </fieldset>\r\n          </mat-expansion-panel>\r\n\r\n          <!-- opStyle -->\r\n          <div id=\"opStyle\">\r\n            <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n              <mat-label>operations style</mat-label>\r\n              <textarea\r\n                matInput\r\n                [formControl]=\"opStyle\"\r\n                placeholder=\"operations style\"\r\n              ></textarea>\r\n            </mat-form-field>\r\n          </div>\r\n        </div>\r\n      </div>\r\n    </mat-tab>\r\n\r\n    <!-- image -->\r\n    <mat-tab>\r\n      <ng-template mat-tab-label>\r\n        <mat-icon>image</mat-icon> <span class=\"label\">image</span>\r\n      </ng-template>\r\n\r\n      <div id=\"image\">\r\n        <div id=\"image-ctl\">\r\n          <!-- url -->\r\n          <mat-form-field class=\"long-text\">\r\n            <mat-label>URL</mat-label>\r\n            <input matInput [formControl]=\"imageUrl\" />\r\n          </mat-form-field>\r\n          <div class=\"form-row\">\r\n            <!-- x -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>X</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"imageX\"\r\n                placeholder=\"X\"\r\n              />\r\n            </mat-form-field>\r\n            <!-- y -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>Y</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"imageY\"\r\n                placeholder=\"Y\"\r\n              />\r\n            </mat-form-field>\r\n          </div>\r\n          <div class=\"form-row\">\r\n            <!-- width -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>width</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                min=\"0\"\r\n                [formControl]=\"imageWidth\"\r\n              />\r\n            </mat-form-field>\r\n            <!-- height -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>height</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                min=\"0\"\r\n                [formControl]=\"imageHeight\"\r\n              />\r\n            </mat-form-field>\r\n          </div>\r\n          <!-- defs -->\r\n          <div>\r\n            <mat-form-field class=\"long-text\">\r\n              <mat-label>defs</mat-label>\r\n              <textarea matInput [formControl]=\"defs\" rows=\"3\"></textarea>\r\n            </mat-form-field>\r\n          </div>\r\n        </div>\r\n        <div id=\"image-view\">\r\n          @if (imageUrl.value) {\r\n          <img alt=\"background\" [src]=\"imageUrl.value\" width=\"600\" />\r\n          }\r\n        </div>\r\n      </div>\r\n    </mat-tab>\r\n\r\n    <!-- timelines -->\r\n    <mat-tab>\r\n      <ng-template mat-tab-label>\r\n        <mat-icon>animation</mat-icon> <span class=\"label\">animation</span>\r\n      </ng-template>\r\n\r\n      <gve-animation-timeline-set\r\n        [tags]=\"opTags\"\r\n        [elementIds]=\"opElementIds\"\r\n        [timelines]=\"timelines.value\"\r\n        (timelinesChange)=\"onTimelinesChange($event)\"\r\n      />\r\n    </mat-tab>\r\n  </mat-tab-group>\r\n\r\n  <!-- progress -->\r\n  <div>\r\n    <mat-progress-bar mode=\"indeterminate\" *ngIf=\"busy\" />\r\n  </div>\r\n\r\n  <!-- result -->\r\n  @if (result) {\r\n  <fieldset id=\"result\">\r\n    <legend>result</legend>\r\n    <gve-chain-result-view\r\n      [result]=\"result\"\r\n      [initialStepIndex]=\"initialStepIndex\"\r\n      (stepPick)=\"onStepPick($event)\"\r\n    />\r\n  </fieldset>\r\n  }\r\n\r\n  <!-- snapshot view -->\r\n  <fieldset id=\"preview\">\r\n    <legend class=\"button-row\">\r\n      <span>{{ viewTitle }}</span>\r\n    </legend>\r\n    <!-- snapshot view -->\r\n    <gve-snapshot-view\r\n      #snapshotView\r\n      [debug]=\"debug\"\r\n      [data]=\"viewData\"\r\n      (snapshotRender)=\"onSnapshotRender($any($event))\"\r\n      (visualEvent)=\"onVisualEvent($any($event))\"\r\n    />\r\n    <div class=\"button-row\">\r\n      <!-- toggle ruler -->\r\n      <mat-button-toggle\r\n        type=\"button\"\r\n        mat-icon-button\r\n        color=\"primary\"\r\n        matTooltip=\"Toggle rulers\"\r\n        [checked]=\"rulers\"\r\n        (change)=\"toggleRulers()\"\r\n      >\r\n        <mat-icon class=\"mat-primary\">straighten</mat-icon>\r\n      </mat-button-toggle>\r\n      <!-- run to last -->\r\n      <button\r\n        type=\"button\"\r\n        mat-icon-button\r\n        class=\"primary\"\r\n        matTooltip=\"Run all operations\"\r\n        [disabled]=\"!operations.value.length\"\r\n        (click)=\"runToLast()\"\r\n      >\r\n        <mat-icon>play_circle</mat-icon>\r\n      </button>\r\n    </div>\r\n    @if (visualInfo) {\r\n    <div id=\"visual-info\">{{ visualInfo }}</div>\r\n    }\r\n  </fieldset>\r\n\r\n  <!--buttons -->\r\n  <div class=\"form-row-center\">\r\n    <button\r\n      type=\"button\"\r\n      class=\"mat-warn\"\r\n      mat-flat-button\r\n      matTooltip=\"Discard changes\"\r\n      (click)=\"close()\"\r\n    >\r\n      <mat-icon>clear</mat-icon>\r\n      close\r\n    </button>\r\n    @if (!noSave) {\r\n    <button\r\n      type=\"submit\"\r\n      class=\"mat-primary\"\r\n      mat-flat-button\r\n      matTooltip=\"Save changes\"\r\n      [disabled]=\"form.invalid\"\r\n    >\r\n      <mat-icon>check_circle</mat-icon>\r\n      save\r\n    </button>\r\n    }\r\n  </div>\r\n</form>\r\n"]}