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