@myrmidon/gve-core 0.0.5 → 1.0.0

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 (35) hide show
  1. package/README.md +2 -0
  2. package/fesm2022/myrmidon-gve-core.mjs +370 -259
  3. package/fesm2022/myrmidon-gve-core.mjs.map +1 -1
  4. package/lib/components/batch-operation-editor/batch-operation-editor.component.d.ts +46 -0
  5. package/lib/components/chain-result-view/chain-result-view.component.d.ts +4 -1
  6. package/lib/components/feature-editor/feature-editor.component.d.ts +1 -1
  7. package/lib/components/feature-set-editor/feature-set-editor.component.d.ts +1 -1
  8. package/lib/components/snapshot-editor/snapshot-editor.component.d.ts +11 -9
  9. package/lib/components/steps-map/steps-map.component.d.ts +2 -0
  10. package/lib/models.d.ts +8 -0
  11. package/lib/services/gve-api.service.d.ts +33 -0
  12. package/package.json +10 -12
  13. package/public-api.d.ts +1 -0
  14. package/esm2022/lib/components/animation-timeline/animation-timeline.component.mjs +0 -207
  15. package/esm2022/lib/components/animation-timeline-set/animation-timeline-set.component.mjs +0 -154
  16. package/esm2022/lib/components/animation-tween/animation-tween.component.mjs +0 -184
  17. package/esm2022/lib/components/base-text-char/base-text-char.component.mjs +0 -46
  18. package/esm2022/lib/components/base-text-editor/base-text-editor.component.mjs +0 -115
  19. package/esm2022/lib/components/base-text-view/base-text-view.component.mjs +0 -159
  20. package/esm2022/lib/components/chain-operation-editor/chain-operation-editor.component.mjs +0 -570
  21. package/esm2022/lib/components/chain-result-view/chain-result-view.component.mjs +0 -222
  22. package/esm2022/lib/components/feature-editor/feature-editor.component.mjs +0 -200
  23. package/esm2022/lib/components/feature-set-editor/feature-set-editor.component.mjs +0 -175
  24. package/esm2022/lib/components/feature-set-view/feature-set-view.component.mjs +0 -100
  25. package/esm2022/lib/components/ln-heights-editor/ln-heights-editor.component.mjs +0 -126
  26. package/esm2022/lib/components/operation-source-editor/operation-source-editor.component.mjs +0 -131
  27. package/esm2022/lib/components/simple-tree/simple-tree.component.mjs +0 -72
  28. package/esm2022/lib/components/snapshot-editor/snapshot-editor.component.mjs +0 -863
  29. package/esm2022/lib/components/steps-map/steps-map.component.mjs +0 -77
  30. package/esm2022/lib/models.mjs +0 -2
  31. package/esm2022/lib/services/gve-api.service.mjs +0 -65
  32. package/esm2022/lib/services/settings.service.mjs +0 -87
  33. package/esm2022/lib/validators/svg-validators.mjs +0 -34
  34. package/esm2022/myrmidon-gve-core.mjs +0 -5
  35. package/esm2022/public-api.mjs +0 -22
@@ -1,863 +0,0 @@
1
- import { CUSTOM_ELEMENTS_SCHEMA, Component, EventEmitter, Input, Output, ViewChild, } from '@angular/core';
2
- import { FormControl, FormsModule, ReactiveFormsModule, Validators, } from '@angular/forms';
3
- import { CommonModule } from '@angular/common';
4
- import { MatButtonModule } from '@angular/material/button';
5
- import { MatButtonToggleModule } from '@angular/material/button-toggle';
6
- import { MatCheckboxModule } from '@angular/material/checkbox';
7
- import { MatExpansionModule } from '@angular/material/expansion';
8
- import { MatFormFieldModule } from '@angular/material/form-field';
9
- import { MatIconModule } from '@angular/material/icon';
10
- import { MatInputModule } from '@angular/material/input';
11
- import { MatProgressBarModule } from '@angular/material/progress-bar';
12
- import { MatSnackBarModule } from '@angular/material/snack-bar';
13
- import { MatTabsModule } from '@angular/material/tabs';
14
- import { MatTooltipModule } from '@angular/material/tooltip';
15
- import { customAlphabet } from 'nanoid';
16
- import { NgToolsModule, deepCopy } from '@myrmidon/ng-tools';
17
- import { DEFAULT_SVG_BASE_TEXT_OPTIONS, } from '@myrmidon/gve-snapshot-view';
18
- import { BaseTextViewComponent, } from '../base-text-view/base-text-view.component';
19
- import { ChainOperationEditorComponent } from '../chain-operation-editor/chain-operation-editor.component';
20
- import { ChainResultViewComponent } from '../chain-result-view/chain-result-view.component';
21
- import { LnHeightsEditorComponent } from '../ln-heights-editor/ln-heights-editor.component';
22
- import { BaseTextEditorComponent } from '../base-text-editor/base-text-editor.component';
23
- import { AnimationTimelineSetComponent } from '../animation-timeline-set/animation-timeline-set.component';
24
- import * as i0 from "@angular/core";
25
- import * as i1 from "@angular/forms";
26
- 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";
40
- // default snapshot view title
41
- const VIEW_TITLE = 'preview';
42
- // suffix for IDs of SVG elements to be made transparent at first rendition
43
- const SVG_TRANSP_SUFFIX = '_t';
44
- /**
45
- * 🔑 `gve-snapshot-editor`
46
- *
47
- * A component to edit a text snapshot. This is the top level component in the GVE core
48
- * library.
49
- *
50
- * - ▶️ `snapshot` (`Snapshot`): the snapshot to edit.
51
- * - ▶️ `batchOps` (`string`): the batch operations text to set for the editor.
52
- * - ▶️ `noSave` (`boolean`): true to disable saving.
53
- * - 🔥 `snapshotChange` (`Snapshot`): emitted when the user saves the edited snapshot.
54
- * - 🔥 `snapshotCancel` (`void`): emitted when the user cancels the snapshot editing.
55
- */
56
- export class SnapshotEditorComponent {
57
- /**
58
- * The snapshot to edit.
59
- */
60
- get snapshot() {
61
- return this._snapshot;
62
- }
63
- set snapshot(value) {
64
- if (this._snapshot === value) {
65
- return;
66
- }
67
- this._snapshot = value || undefined;
68
- this.updateForm(this._snapshot);
69
- setTimeout(() => {
70
- this.runToLast();
71
- }, 0);
72
- }
73
- constructor(formBuilder, _api, _dialogService, _snackbar) {
74
- this._api = _api;
75
- this._dialogService = _dialogService;
76
- this._snackbar = _snackbar;
77
- this._nanoid = customAlphabet('1234567890abcdef', 10);
78
- /**
79
- * Emitted when the user saves the edited snapshot.
80
- */
81
- this.snapshotChange = new EventEmitter();
82
- /**
83
- * Emitted when the user cancels the snapshot editing.
84
- */
85
- this.snapshotCancel = new EventEmitter();
86
- // list of operations output tags
87
- this.opTags = [];
88
- // list of operation diplomatic.g element IDs
89
- this.opElementIds = [];
90
- // the lines count for the current base text
91
- this.lineCount = 0;
92
- this.editedOpIndex = -1;
93
- this.opTypeMap = {
94
- 0: 'replace',
95
- 1: 'delete',
96
- 2: 'add-before',
97
- 3: 'add-after',
98
- 4: 'move-before',
99
- 5: 'move-after',
100
- 6: 'swap',
101
- 7: 'annotate',
102
- };
103
- // snapshot view data
104
- this.viewTitle = VIEW_TITLE;
105
- this.rulers = true;
106
- this.initialStepIndex = -1;
107
- // general
108
- this.width = new FormControl(800, { nonNullable: true });
109
- this.height = new FormControl(600, { nonNullable: true });
110
- this.style = new FormControl(null);
111
- // base text
112
- this.baseText = new FormControl([], {
113
- nonNullable: true,
114
- validators: [Validators.required],
115
- });
116
- this.offsetX = new FormControl(0, { nonNullable: true });
117
- this.offsetY = new FormControl(0, { nonNullable: true });
118
- this.lineHeightOffset = new FormControl(DEFAULT_SVG_BASE_TEXT_OPTIONS.lineHeightOffset, { nonNullable: true });
119
- this.lnHeights = new FormControl(null);
120
- this.charSpacingOffset = new FormControl(0, { nonNullable: true });
121
- this.spcWidthOffset = new FormControl(0, { nonNullable: true });
122
- this.textStyle = new FormControl(null);
123
- // operations
124
- this.operations = new FormControl([], {
125
- nonNullable: true,
126
- });
127
- this.inputOps = new FormControl(null);
128
- this.opStyle = new FormControl(null);
129
- // image
130
- this.imageUrl = new FormControl(null);
131
- this.imageOpacity = new FormControl(1, { nonNullable: true });
132
- this.imageX = new FormControl(0, { nonNullable: true });
133
- this.imageY = new FormControl(0, { nonNullable: true });
134
- this.imageWidth = new FormControl(0, { nonNullable: true });
135
- this.imageHeight = new FormControl(0, { nonNullable: true });
136
- this.defs = new FormControl(null);
137
- // timelines
138
- this.timelines = new FormControl([], {
139
- nonNullable: true,
140
- });
141
- // form
142
- this.form = formBuilder.group({
143
- width: this.width,
144
- height: this.height,
145
- style: this.style,
146
- baseText: this.baseText,
147
- offsetX: this.offsetX,
148
- offsetY: this.offsetY,
149
- lineHeightOffset: this.lineHeightOffset,
150
- lnHeights: this.lnHeights,
151
- charSpacingOffset: this.charSpacingOffset,
152
- spcWidthOffset: this.spcWidthOffset,
153
- textStyle: this.textStyle,
154
- operations: this.operations,
155
- inputOps: this.inputOps,
156
- opStyle: this.opStyle,
157
- // image
158
- imageUrl: this.imageUrl,
159
- imageOpacity: this.imageOpacity,
160
- imageX: this.imageX,
161
- imageY: this.imageY,
162
- imageWidth: this.imageWidth,
163
- imageHeight: this.imageHeight,
164
- defs: this.defs,
165
- // timelines
166
- timelines: this.timelines,
167
- });
168
- }
169
- ngOnInit() {
170
- // if there are any batch operations, parse them into snapshot operations
171
- if (this.batchOps) {
172
- this.inputOps.setValue(this.batchOps);
173
- this.parseOperations();
174
- }
175
- }
176
- /**
177
- * Set the view data for the snapshot view.
178
- *
179
- * @param snapshot The optional snapshot data to use. If not provided,
180
- * a snapshot will be created from the form data.
181
- * @param title The optional title to set for the view.
182
- */
183
- setViewData(snapshot, title = VIEW_TITLE) {
184
- console.log('set view data');
185
- // get the snapshot to use
186
- if (!snapshot) {
187
- snapshot = this.getSnapshot();
188
- }
189
- // update view data
190
- this.viewTitle = title;
191
- this.visualInfo = undefined;
192
- this.viewData = {
193
- snapshot: snapshot,
194
- options: {
195
- debug: this.debug,
196
- delayedRender: true,
197
- showRulers: true,
198
- showGrid: true,
199
- panZoom: true,
200
- transparentIds: this._transparentIds,
201
- },
202
- };
203
- console.log('view data: ', this.viewData);
204
- }
205
- updateForm(snapshot) {
206
- this._transparentIds = undefined;
207
- this.initialStepIndex = -1;
208
- if (!snapshot) {
209
- this.form.reset();
210
- }
211
- else {
212
- this.width.setValue(snapshot.size.width);
213
- this.height.setValue(snapshot.size.height);
214
- this.style.setValue(snapshot.style || null);
215
- this.imageUrl.setValue(snapshot.image?.url || null);
216
- this.imageOpacity.setValue(snapshot.image?.opacity || 0);
217
- this.imageX.setValue(snapshot.image?.canvas?.x || 0);
218
- this.imageY.setValue(snapshot.image?.canvas?.y || 0);
219
- this.imageWidth.setValue(snapshot.image?.canvas?.width || 0);
220
- this.imageHeight.setValue(snapshot.image?.canvas?.height || 0);
221
- this.defs.setValue(snapshot.defs || null);
222
- this.baseText.setValue(snapshot.text);
223
- this.offsetX.setValue(snapshot.textOptions?.offset?.x || 0);
224
- this.offsetY.setValue(snapshot.textOptions?.offset?.y || 0);
225
- this.lineHeightOffset.setValue(snapshot.textOptions?.lineHeightOffset);
226
- this.lnHeights.setValue(snapshot.textOptions?.minLineHeights || null);
227
- this.charSpacingOffset.setValue(snapshot.textOptions?.charSpacingOffset);
228
- this.spcWidthOffset.setValue(snapshot.textOptions?.spcWidthOffset);
229
- this.textStyle.setValue(snapshot.textStyle || null);
230
- this.operations.setValue(snapshot.operations);
231
- this.inputOps.reset();
232
- this.opStyle.setValue(snapshot.opStyle || null);
233
- // timelines
234
- const timelines = [];
235
- if (snapshot.timelines) {
236
- for (let tag in snapshot.timelines) {
237
- timelines.push(snapshot.timelines[tag]);
238
- }
239
- }
240
- this.timelines.setValue(timelines);
241
- }
242
- this.updateLineCount(this.baseText.value);
243
- this.updateOpLists();
244
- this.form.markAsPristine();
245
- }
246
- // #region base text
247
- /**
248
- * Update the line count based on the received text.
249
- * @param text The text to use for counting lines.
250
- */
251
- updateLineCount(text) {
252
- if (!text.length) {
253
- this.lineCount = 0;
254
- }
255
- else {
256
- this.lineCount = text.filter((c) => c.data === '\n').length + 1;
257
- }
258
- }
259
- /**
260
- * Handle the event fired by the base text editor to pick a text range.
261
- * @param range The picked range.
262
- */
263
- onRangePick(range) {
264
- this.textRange = range;
265
- }
266
- /**
267
- * Handle the event fired by the base text editor to change the base text.
268
- * @param text The text to set.
269
- */
270
- onTextChange(text) {
271
- console.log('text change: ' + text);
272
- // update the form control and reset the current text range
273
- this.baseText.setValue(text);
274
- this.baseText.updateValueAndValidity();
275
- this.baseText.markAsDirty();
276
- this.textRange = undefined;
277
- // update the line count
278
- this.updateLineCount(text);
279
- // remove all operations and update the view data
280
- this.removeAllOperations();
281
- }
282
- // #endregion
283
- // #region operations
284
- /**
285
- * Update the lists of operation output tags and element IDs by collecting
286
- * all the operation tags and the IDs of the elements in their diplomatic.g
287
- * SVG code if any.
288
- */
289
- updateOpLists() {
290
- const tags = new Set();
291
- const ids = new Set();
292
- let n = 0;
293
- for (const op of this.operations.value) {
294
- // output tag
295
- n++;
296
- tags.add(op.outputTag || `v${n}`);
297
- // element IDs
298
- if (op.diplomatics?.g) {
299
- this.parseSvgIds(op.diplomatics.g)?.forEach((id) => ids.add(id));
300
- }
301
- }
302
- this.opTags = [...tags];
303
- this.opElementIds = [...ids];
304
- }
305
- /**
306
- * Edit a new operation.
307
- */
308
- editNewOperation() {
309
- // create a new operation and edit it
310
- this.editedOp = {
311
- id: this._nanoid(),
312
- type: 0,
313
- at: this.textRange?.at || 0,
314
- run: this.textRange?.run || 1,
315
- };
316
- this.editedOpIndex = -1;
317
- }
318
- /**
319
- * Edit (a copy of) the operation at the specified index.
320
- * @param index The operation index.
321
- */
322
- editOperation(index) {
323
- this.editedOpIndex = index;
324
- this.editedOp = deepCopy(this.operations.value[index]);
325
- }
326
- /**
327
- * Close the currently edited operation.
328
- */
329
- closeEditedOperation() {
330
- this.editedOpIndex = -1;
331
- this.editedOp = undefined;
332
- }
333
- /**
334
- * Handle the event fired by the operation editor to cancel
335
- * the current operation edits.
336
- */
337
- onOperationCancel() {
338
- this.closeEditedOperation();
339
- }
340
- /**
341
- * Handle the event fired by the operation editor to change
342
- * the currently edited operation, or add a new one if that
343
- * was a new operation.
344
- * @param op The changed operation.
345
- */
346
- async onOperationChange(op) {
347
- console.log('operation change');
348
- const operations = [...this.operations.value];
349
- // replace or add the operation
350
- let i = this.editedOpIndex;
351
- if (this.editedOpIndex > -1) {
352
- operations.splice(this.editedOpIndex, 1, op);
353
- }
354
- else {
355
- operations.push(op);
356
- i = operations.length - 1;
357
- }
358
- // update the operations form control
359
- this.operations.setValue(operations);
360
- this.operations.markAsDirty();
361
- this.operations.updateValueAndValidity();
362
- // update the operation lists
363
- this.updateOpLists();
364
- // update the view data
365
- await this.runTo(i);
366
- // close the edited operation
367
- this.closeEditedOperation();
368
- }
369
- /**
370
- * Delete the operation at the specified index.
371
- * @param index The index of the operation to delete.
372
- */
373
- deleteOperation(index) {
374
- this._dialogService
375
- .confirm('Confirm Deletion', `Delete operation "${this.operations.value[index].id}"?`)
376
- .subscribe((yes) => {
377
- if (yes) {
378
- // close the edited operation if it is the one being deleted
379
- if (this.editedOpIndex === index) {
380
- this.closeEditedOperation();
381
- }
382
- // reset the result operation ID if it is the one being deleted
383
- if (this.resultOperationId === this.operations.value[index].id) {
384
- this.resultOperationId = undefined;
385
- }
386
- // delete the operation and update the form control
387
- const operations = [...this.operations.value];
388
- operations.splice(index, 1);
389
- this.operations.setValue(operations);
390
- this.operations.markAsDirty();
391
- this.operations.updateValueAndValidity();
392
- // update the operation lists
393
- this.updateOpLists();
394
- // update the view data
395
- this.runToLast();
396
- }
397
- });
398
- }
399
- /**
400
- * Parse the operations from their text and append them to the current
401
- * snapshot operations.
402
- */
403
- parseOperations() {
404
- if (this.busy || !this.inputOps.value) {
405
- return;
406
- }
407
- this._dialogService
408
- .confirm('Confirm', 'Parse operations?')
409
- .subscribe((yes) => {
410
- if (yes) {
411
- this.busy = true;
412
- this.parseError = undefined;
413
- this._api.parseOperations(this.inputOps.value).subscribe({
414
- next: (wrapper) => {
415
- const operations = [...this.operations.value];
416
- operations.push(...wrapper.result);
417
- this.operations.setValue(operations);
418
- this.operations.markAsDirty();
419
- this.operations.updateValueAndValidity();
420
- // update the operation lists
421
- this.updateOpLists();
422
- // update the view data
423
- this.runToLast();
424
- },
425
- error: (error) => {
426
- this.parseError = error.message;
427
- },
428
- complete: () => {
429
- this.busy = false;
430
- },
431
- });
432
- }
433
- });
434
- }
435
- removeAllOperations() {
436
- this.resultOperationId = undefined;
437
- this.closeEditedOperation();
438
- this.operations.reset();
439
- this.operations.markAsDirty();
440
- this.operations.updateValueAndValidity();
441
- this.opTags = [];
442
- this.opElementIds = [];
443
- this.setViewData();
444
- }
445
- /**
446
- * Remove all the operations.
447
- */
448
- clearOperations() {
449
- this._dialogService
450
- .confirm('Confirm', 'Remove all operations?')
451
- .subscribe((yes) => {
452
- if (yes) {
453
- this.removeAllOperations();
454
- }
455
- });
456
- }
457
- /**
458
- * Parse all the id attributes from the received SVG code and
459
- * return them as an array.
460
- *
461
- * @param svg The SVG content to parse or undefined.
462
- * @returns Array of IDs found in the SVG content or undefined.
463
- */
464
- parseSvgIds(svg) {
465
- if (!svg) {
466
- return undefined;
467
- }
468
- const regex = /<[^>]+\bid=(["'])(.*?)\1/g;
469
- const ids = [];
470
- let match;
471
- while ((match = regex.exec(svg)) !== null) {
472
- ids.push(match[2]);
473
- }
474
- return ids;
475
- }
476
- /**
477
- * Run the operations up to the specified index (included).
478
- * This is called when:
479
- * - a preview is requested by the operation editor.
480
- * - the currently edited operation is saved.
481
- * - the user picks a step in the chain result view.
482
- * - runToLast is called, which happens when:
483
- * - setting the snapshot.
484
- * - parsing a batch of operations.
485
- * - deleting an operation.
486
- *
487
- * @param index The index of the operation to run to.
488
- * @param lastOperation The operation to use in place of the existing
489
- * operation in the snapshot at index. This is used when previewing
490
- * the edited operation from within the operation editor.
491
- */
492
- runTo(index, lastOperation) {
493
- return new Promise((resolve, reject) => {
494
- if (this.busy) {
495
- return reject('Busy');
496
- }
497
- console.log('run to: ' + index);
498
- // get the snapshot to run operations on
499
- const snapshot = this.getSnapshot();
500
- if (!snapshot.text) {
501
- return reject('No snapshot text');
502
- }
503
- // run operations up to the specified index, but replace the last
504
- // operation when this was received as a parameter
505
- const operations = [...this.operations.value];
506
- if (lastOperation) {
507
- operations.slice(0, index);
508
- operations.push(lastOperation);
509
- }
510
- else {
511
- operations.slice(0, index + 1);
512
- }
513
- this.busy = true;
514
- this.initialStepIndex = index;
515
- this._api
516
- .runOperations(snapshot.text, operations)
517
- .subscribe({
518
- next: (wrapper) => {
519
- // handle operation (non-fatal) error or result
520
- if (wrapper.error) {
521
- this._snackbar.open(wrapper.error, 'OK');
522
- reject(wrapper.error);
523
- return;
524
- }
525
- // extract the IDs from the last operation's diplomatics and filter
526
- // them so that only the ones ending with _t are kept. By convention,
527
- // all the IDs ending with this suffix are to be made invisible at
528
- // their first rendition (opacity=0). An animation will then make
529
- // them visible.
530
- this._transparentIds = this.parseSvgIds(operations[index].diplomatics?.g)?.filter((id) => id.endsWith(SVG_TRANSP_SUFFIX));
531
- this.setViewData(snapshot, snapshot.operations[index].outputTag);
532
- this.result = wrapper.result;
533
- resolve();
534
- },
535
- error: (error) => {
536
- console.error(error);
537
- this._transparentIds = undefined;
538
- this._snackbar.open('Error running operations', 'OK');
539
- reject(error);
540
- },
541
- complete: () => {
542
- this.busy = false;
543
- },
544
- });
545
- });
546
- }
547
- /**
548
- * Run the operations up to the last operation if any.
549
- * Otherwise, just update the snapshot view.
550
- */
551
- runToLast() {
552
- if (this.operations.value.length) {
553
- this.runTo(this.operations.value.length - 1);
554
- }
555
- else {
556
- this.setViewData();
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
- */
565
- onOperationPreview(operation) {
566
- // no multiple previews or previewing a new operation
567
- if (this._previewing || this.editedOpIndex < 0) {
568
- return;
569
- }
570
- this._previewing = true;
571
- setTimeout(() => {
572
- this.runTo(this.editedOpIndex, operation);
573
- this._previewing = false;
574
- }, 0);
575
- }
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
589
- if (!this.result) {
590
- return null;
591
- }
592
- // get the currently edited snapshot
593
- const snapshot = this.getSnapshot();
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.
603
- for (const sn of stepNodes) {
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
607
- if (i > -1) {
608
- snapNodes[i].features = sn.features;
609
- }
610
- else {
611
- // new node: we assume that it has a manually set position (x,y features)
612
- snapNodes.push(sn);
613
- }
614
- }
615
- // append nodes introduced *before* the picked step, if any
616
- // (those introduced by the picked step have already been added)
617
- const stepIndex = this.result.steps.indexOf(step);
618
- for (let i = 0; i < stepIndex; i++) {
619
- const stepNodes = this.result.taggedNodes[this.result.steps[i].outputTag];
620
- for (const n of stepNodes) {
621
- if (n.id > maxOriginalId && !snapNodes.find((x) => x.id === n.id)) {
622
- snapNodes.push(n);
623
- }
624
- }
625
- }
626
- // update the snapshot text nodes
627
- snapshot.text = snapNodes;
628
- // remove from the snapshot the operations after the picked step
629
- const i = snapshot.operations.findIndex((o) => o.id === step.operation.id);
630
- if (i > -1) {
631
- snapshot.operations = snapshot.operations.slice(0, i + 1);
632
- }
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
- }
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 view data
664
- this.setViewData(snapshot, step.outputTag);
665
- // play animation if any
666
- const tl = this.timelines.value.find((t) => t.tag === step.outputTag);
667
- if (tl) {
668
- this._stepPickFrozen = true;
669
- // find the index of the picked step in the snapshot operations
670
- const i = snapshot.operations.findIndex((o) => o.id === step.operation.id);
671
- // run the operations up to the picked step as we need to hide
672
- // those visuals to be revealed by the animation
673
- await this.runTo(i);
674
- this._stepPickFrozen = false;
675
- // play the timeline
676
- setTimeout(() => {
677
- this.playTimeline(tl);
678
- }, 0);
679
- }
680
- }
681
- /**
682
- * Handle the event fired by a visual in the snapshot view.
683
- *
684
- * @param event The event.
685
- */
686
- onVisualEvent(event) {
687
- const d = event.detail;
688
- if (d.event.type === 'mouseover') {
689
- const visual = d.source;
690
- const sb = [];
691
- sb.push('#' + visual.id);
692
- sb.push(` (${visual.type})`);
693
- if (visual.data?.label) {
694
- sb.push(': ');
695
- sb.push(visual.data.label);
696
- }
697
- if (visual.data?.features?.length) {
698
- sb.push(' ');
699
- sb.push(visual.data.features
700
- .map((f) => `${f.name}=${f.value}`)
701
- .join('\n'));
702
- }
703
- this.visualInfo = sb.join('');
704
- }
705
- }
706
- /**
707
- * Handle the change of line heights by updating the form control.
708
- *
709
- * @param heights The line heights.
710
- */
711
- onHeightsChange(heights) {
712
- this.lnHeights.setValue(heights || null);
713
- this.lnHeights.markAsDirty();
714
- this.lnHeights.updateValueAndValidity();
715
- }
716
- /**
717
- * Handle the change of timelines by updating the form control.
718
- *
719
- * @param timelines The timelines.
720
- */
721
- onTimelinesChange(timelines) {
722
- this.timelines.setValue(timelines);
723
- this.timelines.markAsDirty();
724
- this.timelines.updateValueAndValidity();
725
- }
726
- /**
727
- * Emit the cancel event for this snapshot edit.
728
- */
729
- close() {
730
- this.snapshotCancel.emit();
731
- }
732
- /**
733
- * Handle the render event from the snapshot view to get a reference
734
- * to the renderer.
735
- *
736
- * @param event The rendition event.
737
- */
738
- onSnapshotRender(event) {
739
- this._renderer = event.detail.renderer;
740
- }
741
- /**
742
- * Toggle rulers in the snapshot view.
743
- */
744
- toggleRulers() {
745
- if (!this._renderer) {
746
- return;
747
- }
748
- this.rulers = this._renderer.toggleRulers();
749
- }
750
- /**
751
- * Get a snapshot from the form data.
752
- *
753
- * @returns New snapshot object.
754
- */
755
- getSnapshot() {
756
- const snapshot = {
757
- size: {
758
- width: this.width.value,
759
- height: this.height.value,
760
- },
761
- style: this.style.value || undefined,
762
- defs: this.defs.value || undefined,
763
- text: this.baseText.value,
764
- textStyle: this.textStyle.value || undefined,
765
- textOptions: {
766
- lineHeightOffset: this.lineHeightOffset.value,
767
- minLineHeights: this.lnHeights.value || undefined,
768
- charSpacingOffset: this.charSpacingOffset.value,
769
- spcWidthOffset: this.spcWidthOffset.value,
770
- offset: {
771
- x: this.offsetX.value,
772
- y: this.offsetY.value,
773
- },
774
- },
775
- operations: [...this.operations.value],
776
- opStyle: this.opStyle.value || undefined,
777
- };
778
- // image
779
- if (this.imageUrl.value) {
780
- snapshot.image = {
781
- url: this.imageUrl.value,
782
- opacity: this.imageOpacity.value || undefined,
783
- };
784
- if (this.imageX.value ||
785
- this.imageY.value ||
786
- this.imageWidth.value ||
787
- this.imageHeight.value) {
788
- snapshot.image.canvas = {
789
- x: this.imageX.value,
790
- y: this.imageY.value,
791
- width: this.imageWidth.value,
792
- height: this.imageHeight.value,
793
- };
794
- }
795
- }
796
- // timelines
797
- if (this.timelines.value.length) {
798
- snapshot.timelines = {};
799
- this.timelines.value.forEach((t) => {
800
- snapshot.timelines[t.tag] = t;
801
- });
802
- }
803
- else {
804
- snapshot.timelines = undefined;
805
- }
806
- return snapshot;
807
- }
808
- /**
809
- * Get a snapshot from the form data and emit it
810
- * in the snapshot change event.
811
- */
812
- save() {
813
- if (this.form.invalid || this.noSave) {
814
- return;
815
- }
816
- this._snapshot = this.getSnapshot();
817
- this.snapshotChange.emit(this._snapshot);
818
- }
819
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.8", ngImport: i0, type: SnapshotEditorComponent, deps: [{ token: i1.FormBuilder }, { token: i2.GveApiService }, { token: i3.DialogService }, { token: i4.MatSnackBar }], target: i0.ɵɵFactoryTarget.Component }); }
820
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.8", 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 <!-- 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 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\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 </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 <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>\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 <!--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}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}.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#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", "disabledInteractive"], 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: "directive", type: i13.MatTabLabel, selector: "[mat-tab-label], [matTabLabel]" }, { 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", "aria-label", "aria-labelledby"], 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", "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"] }] }); }
821
- }
822
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImport: i0, type: SnapshotEditorComponent, decorators: [{
823
- type: Component,
824
- args: [{ selector: 'gve-snapshot-editor', standalone: true, imports: [
825
- CommonModule,
826
- FormsModule,
827
- ReactiveFormsModule,
828
- MatButtonModule,
829
- MatButtonToggleModule,
830
- MatCheckboxModule,
831
- MatExpansionModule,
832
- MatFormFieldModule,
833
- MatIconModule,
834
- MatInputModule,
835
- MatProgressBarModule,
836
- MatSnackBarModule,
837
- MatTabsModule,
838
- MatTooltipModule,
839
- NgToolsModule,
840
- BaseTextEditorComponent,
841
- BaseTextViewComponent,
842
- ChainOperationEditorComponent,
843
- ChainResultViewComponent,
844
- LnHeightsEditorComponent,
845
- AnimationTimelineSetComponent,
846
- ], 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 <!-- 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 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\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 </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 <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>\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 <!--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}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}.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#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"] }]
847
- }], ctorParameters: () => [{ type: i1.FormBuilder }, { type: i2.GveApiService }, { type: i3.DialogService }, { type: i4.MatSnackBar }], propDecorators: { snapshotView: [{
848
- type: ViewChild,
849
- args: ['snapshotView', { static: false }]
850
- }], snapshot: [{
851
- type: Input
852
- }], batchOps: [{
853
- type: Input
854
- }], noSave: [{
855
- type: Input
856
- }], debug: [{
857
- type: Input
858
- }], snapshotChange: [{
859
- type: Output
860
- }], snapshotCancel: [{
861
- type: Output
862
- }] } });
863
- //# 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  OnInit,\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\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\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 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  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 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  // 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 _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    // 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      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 there are any batch operations, parse them into snapshot operations\r\n    if (this.batchOps) {\r\n      this.inputOps.setValue(this.batchOps);\r\n      this.parseOperations();\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.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.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  // #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    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          this.runToLast();\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 || !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              // update the operation lists\r\n              this.updateOpLists();\r\n              // update the view data\r\n              this.runToLast();\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  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  }\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  /**\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<void> {\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 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      // run operations up to the specified index, but replace the last\r\n      // operation when this was received as a parameter\r\n      const operations = [...this.operations.value];\r\n\r\n      if (lastOperation) {\r\n        operations.slice(0, index);\r\n        operations.push(lastOperation);\r\n      } else {\r\n        operations.slice(0, index + 1);\r\n      }\r\n\r\n      this.busy = true;\r\n      this.initialStepIndex = index;\r\n      this._api\r\n        .runOperations(snapshot.text as CharNode[], 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            this._transparentIds = this.parseSvgIds(\r\n              operations[index].diplomatics?.g\r\n            )?.filter((id) => id.endsWith(SVG_TRANSP_SUFFIX));\r\n\r\n            this.setViewData(snapshot, snapshot.operations[index].outputTag);\r\n            this.result = wrapper.result!;\r\n\r\n            resolve();\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.\r\n   * Otherwise, just update the snapshot view.\r\n   */\r\n  public runToLast(): void {\r\n    if (this.operations.value.length) {\r\n      this.runTo(this.operations.value.length - 1);\r\n    } else {\r\n      this.setViewData();\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 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      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        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  /**\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            <!-- 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                    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\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      </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    <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>\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  <!--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"]}