@myrmidon/gve-core 0.0.1
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/README.md +83 -0
- package/esm2022/lib/components/animation-timeline/animation-timeline.component.mjs +176 -0
- package/esm2022/lib/components/animation-tween/animation-tween.component.mjs +161 -0
- package/esm2022/lib/components/animation-vars/animation-vars.component.mjs +141 -0
- package/esm2022/lib/components/base-text-char/base-text-char.component.mjs +37 -0
- package/esm2022/lib/components/base-text-editor/base-text-editor.component.mjs +111 -0
- package/esm2022/lib/components/base-text-view/base-text-view.component.mjs +125 -0
- package/esm2022/lib/components/chain-operation-editor/chain-operation-editor.component.mjs +512 -0
- package/esm2022/lib/components/chain-result-view/chain-result-view.component.mjs +190 -0
- package/esm2022/lib/components/feature-editor/feature-editor.component.mjs +185 -0
- package/esm2022/lib/components/feature-set-editor/feature-set-editor.component.mjs +152 -0
- package/esm2022/lib/components/feature-set-view/feature-set-view.component.mjs +80 -0
- package/esm2022/lib/components/ln-heights-editor/ln-heights-editor.component.mjs +95 -0
- package/esm2022/lib/components/operation-source-editor/operation-source-editor.component.mjs +106 -0
- package/esm2022/lib/components/simple-tree/simple-tree.component.mjs +56 -0
- package/esm2022/lib/components/snapshot-editor/snapshot-editor.component.mjs +510 -0
- package/esm2022/lib/components/steps-map/steps-map.component.mjs +55 -0
- package/esm2022/lib/models.mjs +2 -0
- package/esm2022/lib/services/gve-api.service.mjs +52 -0
- package/esm2022/lib/services/settings.service.mjs +86 -0
- package/esm2022/lib/validators/svg-validators.mjs +28 -0
- package/esm2022/myrmidon-gve-core.mjs +5 -0
- package/esm2022/public-api.mjs +23 -0
- package/fesm2022/myrmidon-gve-core.mjs +2592 -0
- package/fesm2022/myrmidon-gve-core.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/components/animation-timeline/animation-timeline.component.d.ts +42 -0
- package/lib/components/animation-tween/animation-tween.component.d.ts +47 -0
- package/lib/components/animation-vars/animation-vars.component.d.ts +30 -0
- package/lib/components/base-text-char/base-text-char.component.d.ts +29 -0
- package/lib/components/base-text-editor/base-text-editor.component.d.ts +40 -0
- package/lib/components/base-text-view/base-text-view.component.d.ts +29 -0
- package/lib/components/chain-operation-editor/chain-operation-editor.component.d.ts +102 -0
- package/lib/components/chain-result-view/chain-result-view.component.d.ts +44 -0
- package/lib/components/feature-editor/feature-editor.component.d.ts +72 -0
- package/lib/components/feature-set-editor/feature-set-editor.component.d.ts +55 -0
- package/lib/components/feature-set-view/feature-set-view.component.d.ts +36 -0
- package/lib/components/ln-heights-editor/ln-heights-editor.component.d.ts +22 -0
- package/lib/components/operation-source-editor/operation-source-editor.component.d.ts +24 -0
- package/lib/components/simple-tree/simple-tree.component.d.ts +16 -0
- package/lib/components/snapshot-editor/snapshot-editor.component.d.ts +105 -0
- package/lib/components/steps-map/steps-map.component.d.ts +29 -0
- package/lib/models.d.ts +7 -0
- package/lib/services/gve-api.service.d.ts +57 -0
- package/lib/services/settings.service.d.ts +53 -0
- package/lib/validators/svg-validators.d.ts +4 -0
- package/package.json +45 -0
- package/public-api.d.ts +19 -0
|
@@ -0,0 +1,510 @@
|
|
|
1
|
+
import { CUSTOM_ELEMENTS_SCHEMA, Component, EventEmitter, Input, Output, } 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 { NgToolsModule, deepCopy } from '@myrmidon/ng-tools';
|
|
16
|
+
import { DEFAULT_SVG_BASE_TEXT_OPTIONS, } from '@myrmidon/gve-snapshot-view';
|
|
17
|
+
import { BaseTextViewComponent, } from '../base-text-view/base-text-view.component';
|
|
18
|
+
import { ChainOperationEditorComponent } from '../chain-operation-editor/chain-operation-editor.component';
|
|
19
|
+
import { ChainResultViewComponent } from '../chain-result-view/chain-result-view.component';
|
|
20
|
+
import { LnHeightsEditorComponent } from '../ln-heights-editor/ln-heights-editor.component';
|
|
21
|
+
import { BaseTextEditorComponent } from '../base-text-editor/base-text-editor.component';
|
|
22
|
+
import * as i0 from "@angular/core";
|
|
23
|
+
import * as i1 from "@angular/forms";
|
|
24
|
+
import * as i2 from "../../services/gve-api.service";
|
|
25
|
+
import * as i3 from "@myrmidon/ng-mat-tools";
|
|
26
|
+
import * as i4 from "@angular/material/snack-bar";
|
|
27
|
+
import * as i5 from "@angular/common";
|
|
28
|
+
import * as i6 from "@angular/material/button";
|
|
29
|
+
import * as i7 from "@angular/material/button-toggle";
|
|
30
|
+
import * as i8 from "@angular/material/expansion";
|
|
31
|
+
import * as i9 from "@angular/material/form-field";
|
|
32
|
+
import * as i10 from "@angular/material/icon";
|
|
33
|
+
import * as i11 from "@angular/material/input";
|
|
34
|
+
import * as i12 from "@angular/material/progress-bar";
|
|
35
|
+
import * as i13 from "@angular/material/tabs";
|
|
36
|
+
import * as i14 from "@angular/material/tooltip";
|
|
37
|
+
import * as i15 from "@myrmidon/ng-tools";
|
|
38
|
+
const VIEW_TITLE = 'preview';
|
|
39
|
+
/**
|
|
40
|
+
* A component to edit a text snapshot.
|
|
41
|
+
*/
|
|
42
|
+
export class SnapshotEditorComponent {
|
|
43
|
+
/**
|
|
44
|
+
* The snapshot to edit.
|
|
45
|
+
*/
|
|
46
|
+
get snapshot() {
|
|
47
|
+
return this._snapshot;
|
|
48
|
+
}
|
|
49
|
+
set snapshot(value) {
|
|
50
|
+
if (this._snapshot === value) {
|
|
51
|
+
return;
|
|
52
|
+
}
|
|
53
|
+
this._snapshot = value || undefined;
|
|
54
|
+
this.updateForm(this._snapshot);
|
|
55
|
+
this.updateViewData();
|
|
56
|
+
}
|
|
57
|
+
constructor(formBuilder, _api, _dialogService, _snackbar) {
|
|
58
|
+
this._api = _api;
|
|
59
|
+
this._dialogService = _dialogService;
|
|
60
|
+
this._snackbar = _snackbar;
|
|
61
|
+
/**
|
|
62
|
+
* Emitted when the user saves the edited snapshot.
|
|
63
|
+
*/
|
|
64
|
+
this.snapshotChange = new EventEmitter();
|
|
65
|
+
/**
|
|
66
|
+
* Emitted when the user cancels the snapshot editing.
|
|
67
|
+
*/
|
|
68
|
+
this.snapshotCancel = new EventEmitter();
|
|
69
|
+
this.editedOpIndex = -1;
|
|
70
|
+
this.opTypeMap = {
|
|
71
|
+
0: 'replace',
|
|
72
|
+
1: 'delete',
|
|
73
|
+
2: 'addBefore',
|
|
74
|
+
3: 'addAfter',
|
|
75
|
+
4: 'moveBefore',
|
|
76
|
+
5: 'moveAfter',
|
|
77
|
+
6: 'swap',
|
|
78
|
+
7: 'annotate',
|
|
79
|
+
};
|
|
80
|
+
this.lineCount = 0;
|
|
81
|
+
this.viewTitle = VIEW_TITLE;
|
|
82
|
+
this.rulers = true;
|
|
83
|
+
// general
|
|
84
|
+
this.width = new FormControl(800, { nonNullable: true });
|
|
85
|
+
this.height = new FormControl(600, { nonNullable: true });
|
|
86
|
+
this.style = new FormControl(null);
|
|
87
|
+
// base text
|
|
88
|
+
this.baseText = new FormControl([], {
|
|
89
|
+
nonNullable: true,
|
|
90
|
+
validators: [Validators.required],
|
|
91
|
+
});
|
|
92
|
+
this.offsetX = new FormControl(0, { nonNullable: true });
|
|
93
|
+
this.offsetY = new FormControl(0, { nonNullable: true });
|
|
94
|
+
this.lineHeightOffset = new FormControl(DEFAULT_SVG_BASE_TEXT_OPTIONS.lineHeightOffset, { nonNullable: true });
|
|
95
|
+
this.lnHeights = new FormControl(null);
|
|
96
|
+
this.charSpacingOffset = new FormControl(0, { nonNullable: true });
|
|
97
|
+
this.spcWidthOffset = new FormControl(0, { nonNullable: true });
|
|
98
|
+
this.textStyle = new FormControl(null);
|
|
99
|
+
// operations
|
|
100
|
+
this.operations = new FormControl([], {
|
|
101
|
+
nonNullable: true,
|
|
102
|
+
});
|
|
103
|
+
this.inputOps = new FormControl(null);
|
|
104
|
+
this.opStyle = new FormControl(null);
|
|
105
|
+
// image
|
|
106
|
+
this.imageUrl = new FormControl(null);
|
|
107
|
+
this.imageOpacity = new FormControl(1, { nonNullable: true });
|
|
108
|
+
this.imageX = new FormControl(0, { nonNullable: true });
|
|
109
|
+
this.imageY = new FormControl(0, { nonNullable: true });
|
|
110
|
+
this.imageWidth = new FormControl(0, { nonNullable: true });
|
|
111
|
+
this.imageHeight = new FormControl(0, { nonNullable: true });
|
|
112
|
+
this.defs = new FormControl(null);
|
|
113
|
+
this.form = formBuilder.group({
|
|
114
|
+
width: this.width,
|
|
115
|
+
height: this.height,
|
|
116
|
+
style: this.style,
|
|
117
|
+
baseText: this.baseText,
|
|
118
|
+
offsetX: this.offsetX,
|
|
119
|
+
offsetY: this.offsetY,
|
|
120
|
+
lineHeightOffset: this.lineHeightOffset,
|
|
121
|
+
lnHeights: this.lnHeights,
|
|
122
|
+
charSpacingOffset: this.charSpacingOffset,
|
|
123
|
+
spcWidthOffset: this.spcWidthOffset,
|
|
124
|
+
textStyle: this.textStyle,
|
|
125
|
+
operations: this.operations,
|
|
126
|
+
inputOps: this.inputOps,
|
|
127
|
+
opStyle: this.opStyle,
|
|
128
|
+
// image
|
|
129
|
+
imageUrl: this.imageUrl,
|
|
130
|
+
imageOpacity: this.imageOpacity,
|
|
131
|
+
imageX: this.imageX,
|
|
132
|
+
imageY: this.imageY,
|
|
133
|
+
imageWidth: this.imageWidth,
|
|
134
|
+
imageHeight: this.imageHeight,
|
|
135
|
+
defs: this.defs,
|
|
136
|
+
});
|
|
137
|
+
}
|
|
138
|
+
ngOnInit() {
|
|
139
|
+
if (this.batchOps) {
|
|
140
|
+
this.inputOps.setValue(this.batchOps);
|
|
141
|
+
this.parseOperations();
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
updateViewData(newOperation, snapshot, title = VIEW_TITLE) {
|
|
145
|
+
if (!snapshot) {
|
|
146
|
+
snapshot = this.getSnapshot();
|
|
147
|
+
}
|
|
148
|
+
// update or add new operation in copy
|
|
149
|
+
if (newOperation) {
|
|
150
|
+
const index = snapshot.operations.findIndex((op) => op.id === newOperation.id);
|
|
151
|
+
if (index > -1) {
|
|
152
|
+
snapshot.operations.splice(index, 1, newOperation);
|
|
153
|
+
}
|
|
154
|
+
else {
|
|
155
|
+
snapshot.operations.push(newOperation);
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
// update view data
|
|
159
|
+
this.viewTitle = title;
|
|
160
|
+
this.visualInfo = undefined;
|
|
161
|
+
this.viewData = {
|
|
162
|
+
snapshot: snapshot,
|
|
163
|
+
options: {
|
|
164
|
+
// debug: true,
|
|
165
|
+
delayedRender: true,
|
|
166
|
+
showRulers: true,
|
|
167
|
+
showGrid: true,
|
|
168
|
+
panZoom: true,
|
|
169
|
+
},
|
|
170
|
+
};
|
|
171
|
+
}
|
|
172
|
+
updateForm(snapshot) {
|
|
173
|
+
if (!snapshot) {
|
|
174
|
+
this.form.reset();
|
|
175
|
+
}
|
|
176
|
+
else {
|
|
177
|
+
this.width.setValue(snapshot.size.width);
|
|
178
|
+
this.height.setValue(snapshot.size.height);
|
|
179
|
+
this.style.setValue(snapshot.style || null);
|
|
180
|
+
this.imageUrl.setValue(snapshot.image?.url || null);
|
|
181
|
+
this.imageOpacity.setValue(snapshot.image?.opacity || 0);
|
|
182
|
+
this.imageX.setValue(snapshot.image?.canvas?.x || 0);
|
|
183
|
+
this.imageY.setValue(snapshot.image?.canvas?.y || 0);
|
|
184
|
+
this.imageWidth.setValue(snapshot.image?.canvas?.width || 0);
|
|
185
|
+
this.imageHeight.setValue(snapshot.image?.canvas?.height || 0);
|
|
186
|
+
this.defs.setValue(snapshot.defs || null);
|
|
187
|
+
this.baseText.setValue(snapshot.text);
|
|
188
|
+
this.offsetX.setValue(snapshot.textOptions?.offset?.x || 0);
|
|
189
|
+
this.offsetY.setValue(snapshot.textOptions?.offset?.y || 0);
|
|
190
|
+
this.lineHeightOffset.setValue(snapshot.textOptions?.lineHeightOffset);
|
|
191
|
+
this.lnHeights.setValue(snapshot.textOptions?.minLineHeights || null);
|
|
192
|
+
this.charSpacingOffset.setValue(snapshot.textOptions?.charSpacingOffset);
|
|
193
|
+
this.spcWidthOffset.setValue(snapshot.textOptions?.spcWidthOffset);
|
|
194
|
+
this.textStyle.setValue(snapshot.textStyle || null);
|
|
195
|
+
this.operations.setValue(snapshot.operations);
|
|
196
|
+
this.inputOps.reset();
|
|
197
|
+
this.opStyle.setValue(snapshot.opStyle || null);
|
|
198
|
+
}
|
|
199
|
+
this.form.markAsPristine();
|
|
200
|
+
}
|
|
201
|
+
editNewOperation() {
|
|
202
|
+
this.editedOp = {
|
|
203
|
+
id: '',
|
|
204
|
+
type: 0,
|
|
205
|
+
at: this.textRange?.at || 0,
|
|
206
|
+
run: this.textRange?.run || 1,
|
|
207
|
+
};
|
|
208
|
+
this.editedOpIndex = -1;
|
|
209
|
+
}
|
|
210
|
+
editOperation(index) {
|
|
211
|
+
this.editedOp = this.operations.value[index];
|
|
212
|
+
this.editedOpIndex = index;
|
|
213
|
+
}
|
|
214
|
+
closeEditedOp() {
|
|
215
|
+
this.editedOp = undefined;
|
|
216
|
+
this.editedOpIndex = -1;
|
|
217
|
+
}
|
|
218
|
+
onOperationCancel() {
|
|
219
|
+
this.closeEditedOp();
|
|
220
|
+
}
|
|
221
|
+
onOperationChange(op) {
|
|
222
|
+
const operations = [...this.operations.value];
|
|
223
|
+
if (this.editedOpIndex > -1) {
|
|
224
|
+
operations.splice(this.editedOpIndex, 1, op);
|
|
225
|
+
}
|
|
226
|
+
else {
|
|
227
|
+
operations.push(op);
|
|
228
|
+
}
|
|
229
|
+
this.operations.setValue(operations);
|
|
230
|
+
this.operations.markAsDirty();
|
|
231
|
+
this.operations.updateValueAndValidity();
|
|
232
|
+
this.closeEditedOp();
|
|
233
|
+
}
|
|
234
|
+
deleteOperation(index) {
|
|
235
|
+
this._dialogService
|
|
236
|
+
.confirm('Confirm Deletion', `Delete operation "${this.operations.value[index].id}"?`)
|
|
237
|
+
.subscribe((yes) => {
|
|
238
|
+
if (yes) {
|
|
239
|
+
if (this.resultOperationId === this.operations.value[index].id) {
|
|
240
|
+
this.resultOperationId = undefined;
|
|
241
|
+
}
|
|
242
|
+
const operations = [...this.operations.value];
|
|
243
|
+
operations.splice(index, 1);
|
|
244
|
+
this.operations.setValue(operations);
|
|
245
|
+
this.operations.markAsDirty();
|
|
246
|
+
this.operations.updateValueAndValidity();
|
|
247
|
+
}
|
|
248
|
+
});
|
|
249
|
+
}
|
|
250
|
+
onTextChange(text) {
|
|
251
|
+
this.baseText.setValue(text);
|
|
252
|
+
this.baseText.updateValueAndValidity();
|
|
253
|
+
this.baseText.markAsDirty();
|
|
254
|
+
if (!text.length) {
|
|
255
|
+
this.lineCount = 0;
|
|
256
|
+
}
|
|
257
|
+
else {
|
|
258
|
+
this.lineCount = text.filter((c) => c.data === '\n').length + 1;
|
|
259
|
+
}
|
|
260
|
+
this.textRange = undefined;
|
|
261
|
+
this.closeEditedOp();
|
|
262
|
+
this.operations.reset();
|
|
263
|
+
this.updateViewData();
|
|
264
|
+
}
|
|
265
|
+
onRangePick(range) {
|
|
266
|
+
this.textRange = range;
|
|
267
|
+
}
|
|
268
|
+
parseOperations() {
|
|
269
|
+
if (this.busy || !this.inputOps.value) {
|
|
270
|
+
return;
|
|
271
|
+
}
|
|
272
|
+
this._dialogService
|
|
273
|
+
.confirm('Confirm', 'Parse operations?')
|
|
274
|
+
.subscribe((yes) => {
|
|
275
|
+
if (yes) {
|
|
276
|
+
this.busy = true;
|
|
277
|
+
this.parseError = undefined;
|
|
278
|
+
this._api.parseOperations(this.inputOps.value).subscribe({
|
|
279
|
+
next: (wrapper) => {
|
|
280
|
+
const operations = [...this.operations.value];
|
|
281
|
+
operations.push(...wrapper.result);
|
|
282
|
+
this.operations.setValue(operations);
|
|
283
|
+
this.operations.markAsDirty();
|
|
284
|
+
this.operations.updateValueAndValidity();
|
|
285
|
+
this.updateViewData();
|
|
286
|
+
},
|
|
287
|
+
error: (error) => {
|
|
288
|
+
this.parseError = error.message;
|
|
289
|
+
},
|
|
290
|
+
complete: () => {
|
|
291
|
+
this.busy = false;
|
|
292
|
+
},
|
|
293
|
+
});
|
|
294
|
+
}
|
|
295
|
+
});
|
|
296
|
+
}
|
|
297
|
+
clearOperations() {
|
|
298
|
+
this._dialogService
|
|
299
|
+
.confirm('Confirm', 'Remove all operations?')
|
|
300
|
+
.subscribe((yes) => {
|
|
301
|
+
if (yes) {
|
|
302
|
+
this.resultOperationId = undefined;
|
|
303
|
+
this.operations.reset();
|
|
304
|
+
this.operations.markAsDirty();
|
|
305
|
+
this.operations.updateValueAndValidity();
|
|
306
|
+
this.updateViewData();
|
|
307
|
+
}
|
|
308
|
+
});
|
|
309
|
+
}
|
|
310
|
+
onOperationPreview(operation) {
|
|
311
|
+
this.updateViewData(operation);
|
|
312
|
+
}
|
|
313
|
+
runTo(index) {
|
|
314
|
+
const snapshot = this.getSnapshot();
|
|
315
|
+
if (!snapshot.text) {
|
|
316
|
+
return;
|
|
317
|
+
}
|
|
318
|
+
const operations = this.operations.value.slice(0, index + 1);
|
|
319
|
+
this.busy = true;
|
|
320
|
+
this._api.runOperations(snapshot.text, operations).subscribe({
|
|
321
|
+
next: (wrapper) => {
|
|
322
|
+
if (wrapper.error) {
|
|
323
|
+
this._snackbar.open(wrapper.error, 'OK');
|
|
324
|
+
return;
|
|
325
|
+
}
|
|
326
|
+
this.result = wrapper.result;
|
|
327
|
+
// TODO update view
|
|
328
|
+
// this.updateViewData(
|
|
329
|
+
// undefined,
|
|
330
|
+
// snapshot,
|
|
331
|
+
// this.operations.value[index].outputTag
|
|
332
|
+
// );
|
|
333
|
+
},
|
|
334
|
+
error: (error) => {
|
|
335
|
+
console.error(error);
|
|
336
|
+
this._snackbar.open('Error running operations', 'OK');
|
|
337
|
+
},
|
|
338
|
+
complete: () => {
|
|
339
|
+
this.busy = false;
|
|
340
|
+
},
|
|
341
|
+
});
|
|
342
|
+
}
|
|
343
|
+
onStepPick(step) {
|
|
344
|
+
if (!this.result) {
|
|
345
|
+
return;
|
|
346
|
+
}
|
|
347
|
+
this.resultOperationId = step.operation.id;
|
|
348
|
+
// get snapshot for preview by updating its text and
|
|
349
|
+
// removing operations past the last executed
|
|
350
|
+
const stepNodes = this.result.taggedNodes[step.outputTag];
|
|
351
|
+
// we always display the original base text, just updating
|
|
352
|
+
// its nodes features to those of the picked step, and adding
|
|
353
|
+
// new nodes if they are not in the original text. It is assumed
|
|
354
|
+
// that these new nodes have a manually set position.
|
|
355
|
+
const snapshot = this.getSnapshot();
|
|
356
|
+
const nodes = deepCopy(snapshot.text);
|
|
357
|
+
for (const sn of stepNodes) {
|
|
358
|
+
const i = nodes.findIndex((n) => n.id === sn.id);
|
|
359
|
+
if (i > -1) {
|
|
360
|
+
nodes[i].features = sn.features;
|
|
361
|
+
}
|
|
362
|
+
else {
|
|
363
|
+
nodes.push(sn);
|
|
364
|
+
}
|
|
365
|
+
}
|
|
366
|
+
// find max ID in snapshot.text (=original) nodes,
|
|
367
|
+
// so that any ID greater than it belongs to new nodes
|
|
368
|
+
const maxOriginalId = snapshot.text.reduce((max, n) => Math.max(max, n.id), 0);
|
|
369
|
+
// append nodes introduced before the picked step if any
|
|
370
|
+
// (those introduced by the picked step have already been added)
|
|
371
|
+
const stepIndex = this.result.steps.indexOf(step);
|
|
372
|
+
for (let i = 0; i < stepIndex; i++) {
|
|
373
|
+
const stepNodes = this.result.taggedNodes[this.result.steps[i].outputTag];
|
|
374
|
+
for (const n of stepNodes) {
|
|
375
|
+
if (n.id > maxOriginalId && !nodes.find((x) => x.id === n.id)) {
|
|
376
|
+
nodes.push(n);
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
}
|
|
380
|
+
snapshot.text = nodes;
|
|
381
|
+
const i = snapshot.operations.findIndex((o) => o.id === step.operation.id);
|
|
382
|
+
if (i > -1) {
|
|
383
|
+
snapshot.operations = snapshot.operations.slice(0, i + 1);
|
|
384
|
+
}
|
|
385
|
+
this.updateViewData(undefined, snapshot, step.outputTag);
|
|
386
|
+
}
|
|
387
|
+
onVisualEvent(event) {
|
|
388
|
+
const d = event.detail;
|
|
389
|
+
if (d.event.type === 'mouseover') {
|
|
390
|
+
const visual = d.source;
|
|
391
|
+
const sb = [];
|
|
392
|
+
sb.push('#' + visual.id);
|
|
393
|
+
sb.push(` (${visual.type})`);
|
|
394
|
+
if (visual.data?.label) {
|
|
395
|
+
sb.push(': ');
|
|
396
|
+
sb.push(visual.data.label);
|
|
397
|
+
}
|
|
398
|
+
if (visual.data?.features?.length) {
|
|
399
|
+
sb.push(visual.data.features
|
|
400
|
+
.map((f) => `${f.name}=${f.value}`)
|
|
401
|
+
.join('\n'));
|
|
402
|
+
}
|
|
403
|
+
this.visualInfo = sb.join('');
|
|
404
|
+
}
|
|
405
|
+
}
|
|
406
|
+
onHeightsChange(heights) {
|
|
407
|
+
this.lnHeights.setValue(heights || null);
|
|
408
|
+
this.lnHeights.markAsDirty();
|
|
409
|
+
this.lnHeights.updateValueAndValidity();
|
|
410
|
+
}
|
|
411
|
+
close() {
|
|
412
|
+
this.snapshotCancel.emit();
|
|
413
|
+
}
|
|
414
|
+
onSnapshotRender(event) {
|
|
415
|
+
this._renderer = event.detail.renderer;
|
|
416
|
+
}
|
|
417
|
+
toggleRulers() {
|
|
418
|
+
if (!this._renderer) {
|
|
419
|
+
return;
|
|
420
|
+
}
|
|
421
|
+
this.rulers = this._renderer.toggleRulers();
|
|
422
|
+
}
|
|
423
|
+
getSnapshot() {
|
|
424
|
+
const snapshot = {
|
|
425
|
+
size: {
|
|
426
|
+
width: this.width.value,
|
|
427
|
+
height: this.height.value,
|
|
428
|
+
},
|
|
429
|
+
style: this.style.value || undefined,
|
|
430
|
+
defs: this.defs.value || undefined,
|
|
431
|
+
text: this.baseText.value,
|
|
432
|
+
textStyle: this.textStyle.value || undefined,
|
|
433
|
+
textOptions: {
|
|
434
|
+
lineHeightOffset: this.lineHeightOffset.value,
|
|
435
|
+
minLineHeights: this.lnHeights.value || undefined,
|
|
436
|
+
charSpacingOffset: this.charSpacingOffset.value,
|
|
437
|
+
spcWidthOffset: this.spcWidthOffset.value,
|
|
438
|
+
offset: {
|
|
439
|
+
x: this.offsetX.value,
|
|
440
|
+
y: this.offsetY.value,
|
|
441
|
+
},
|
|
442
|
+
},
|
|
443
|
+
operations: this.operations.value,
|
|
444
|
+
opStyle: this.opStyle.value || undefined,
|
|
445
|
+
timelines: {}, // TODO
|
|
446
|
+
};
|
|
447
|
+
// image
|
|
448
|
+
if (this.imageUrl.value) {
|
|
449
|
+
snapshot.image = {
|
|
450
|
+
url: this.imageUrl.value,
|
|
451
|
+
opacity: this.imageOpacity.value || undefined,
|
|
452
|
+
};
|
|
453
|
+
if (this.imageX.value ||
|
|
454
|
+
this.imageY.value ||
|
|
455
|
+
this.imageWidth.value ||
|
|
456
|
+
this.imageHeight.value) {
|
|
457
|
+
snapshot.image.canvas = {
|
|
458
|
+
x: this.imageX.value,
|
|
459
|
+
y: this.imageY.value,
|
|
460
|
+
width: this.imageWidth.value,
|
|
461
|
+
height: this.imageHeight.value,
|
|
462
|
+
};
|
|
463
|
+
}
|
|
464
|
+
}
|
|
465
|
+
return snapshot;
|
|
466
|
+
}
|
|
467
|
+
save() {
|
|
468
|
+
if (this.form.invalid) {
|
|
469
|
+
return;
|
|
470
|
+
}
|
|
471
|
+
this._snapshot = this.getSnapshot();
|
|
472
|
+
this.snapshotChange.emit(this._snapshot);
|
|
473
|
+
}
|
|
474
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: SnapshotEditorComponent, deps: [{ token: i1.FormBuilder }, { token: i2.GveApiService }, { token: i3.DialogService }, { token: i4.MatSnackBar }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
475
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.4", type: SnapshotEditorComponent, isStandalone: true, selector: "gve-snapshot-editor", inputs: { snapshot: "snapshot", batchOps: "batchOps" }, outputs: { snapshotChange: "snapshotChange", snapshotCancel: "snapshotCancel" }, ngImport: i0, template: "<form [formGroup]=\"form\" (submit)=\"save()\">\r\n <mat-tab-group>\r\n <!-- text -->\r\n <mat-tab label=\"text\">\r\n <mat-expansion-panel [expanded]=\"!baseText.value\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title> base text </mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <gve-base-text-editor\r\n [text]=\"baseText.value\"\r\n (textChange)=\"onTextChange($event)\"\r\n />\r\n <fieldset>\r\n <div class=\"form-row\">\r\n <!-- offsetX -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>X offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"offsetX\"\r\n placeholder=\"X offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- offsetY -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>Y offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"offsetY\"\r\n placeholder=\"Y offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- lineHeightOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>ln h-offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"lineHeightOffset\"\r\n placeholder=\"ln h-offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- charSpacingOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>char spacing</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"charSpacingOffset\"\r\n placeholder=\"char spacing\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- spcWidthOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>spc w-offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"spcWidthOffset\"\r\n placeholder=\"spc w-offset\"\r\n />\r\n </mat-form-field>\r\n <!-- minLineHeights -->\r\n <div>\r\n <gve-ln-heights-editor\r\n [lineCount]=\"lineCount\"\r\n [heights]=\"lnHeights.value\"\r\n (heightsChange)=\"onHeightsChange($event)\"\r\n />\r\n </div>\r\n </div>\r\n <!-- textStyle -->\r\n <div>\r\n <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n <mat-label>text style</mat-label>\r\n <textarea\r\n matInput\r\n [formControl]=\"textStyle\"\r\n placeholder=\"text style\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n </fieldset>\r\n </mat-expansion-panel>\r\n </mat-tab>\r\n\r\n <!-- operations -->\r\n <mat-tab label=\"operations\">\r\n <div id=\"snapshot-container\">\r\n <div id=\"general\">\r\n <div class=\"form-row\">\r\n <!-- width -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>width</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"width\"\r\n placeholder=\"width\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- height -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"height\"\r\n placeholder=\"height\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- add op -->\r\n <div>\r\n <button\r\n type=\"button\"\r\n mat-flat-button\r\n color=\"primary\"\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 [ngClass]=\"{ selected: operation.id === resultOperationId }\">\r\n <td class=\"fit-width\">\r\n <!-- edit -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n class=\"mat-primary\"\r\n (click)=\"editOperation(index)\"\r\n matTooltip=\"Edit operation\"\r\n >\r\n <mat-icon>edit</mat-icon>\r\n </button>\r\n <!-- delete -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"warn\"\r\n (click)=\"deleteOperation(index)\"\r\n matTooltip=\"Delete operation\"\r\n >\r\n <mat-icon class=\"mat-warn\">delete</mat-icon>\r\n </button>\r\n <!-- run to -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"accent\"\r\n (click)=\"runTo(index)\"\r\n matTooltip=\"Run to this operation\"\r\n >\r\n <mat-icon class=\"mat-accent\">subscriptions</mat-icon>\r\n </button>\r\n </td>\r\n <td>{{ operation.id }}</td>\r\n <td>{{ operation.type | flatLookup : opTypeMap }}</td>\r\n <td>{{ operation.at }}</td>\r\n <td>{{ operation.run }}</td>\r\n <td>{{ operation.value }}</td>\r\n <td>{{ operation.inputTag }}</td>\r\n <td>{{ operation.outputTag }}</td>\r\n <td>{{ operation.groupId }}</td>\r\n <td>{{ operation.features?.length }}</td>\r\n </tr>\r\n }\r\n </tbody>\r\n </table>\r\n }\r\n\r\n <!-- operation editor -->\r\n <mat-expansion-panel [expanded]=\"editedOp\" [disabled]=\"!editedOp\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title>operation {{ editedOp?.id }}</mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <fieldset>\r\n <gve-chain-operation-editor\r\n [operation]=\"editedOp\"\r\n (operationCancel)=\"onOperationCancel()\"\r\n (operationChange)=\"onOperationChange($event)\"\r\n (operationPreview)=\"onOperationPreview($event)\"\r\n />\r\n </fieldset>\r\n </mat-expansion-panel>\r\n\r\n <div>\r\n <mat-progress-bar mode=\"indeterminate\" *ngIf=\"busy\" />\r\n </div>\r\n\r\n <!-- opStyle -->\r\n <div id=\"opStyle\">\r\n <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n <mat-label>operations style</mat-label>\r\n <textarea\r\n matInput\r\n [formControl]=\"opStyle\"\r\n placeholder=\"operations style\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- batch ops -->\r\n <div>\r\n <mat-expansion-panel>\r\n <mat-expansion-panel-header>\r\n <mat-panel-title>batch operations</mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <fieldset>\r\n <div id=\"batch-container\">\r\n <div id=\"batch-input\">\r\n <div>\r\n <mat-form-field class=\"full-width\">\r\n <mat-label>operations</mat-label>\r\n <textarea\r\n class=\"code\"\r\n matInput\r\n [formControl]=\"inputOps\"\r\n placeholder=\"operations\"\r\n rows=\"8\"\r\n spellcheck=\"false\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n <div class=\"form-row\">\r\n <button\r\n type=\"button\"\r\n color=\"warn\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Remove all the operations\"\r\n [disabled]=\"!operations.value.length || busy\"\r\n (click)=\"clearOperations()\"\r\n >\r\n clear\r\n </button>\r\n <button\r\n type=\"button\"\r\n color=\"primary\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Parse text and add results to the operations\"\r\n [disabled]=\"!inputOps.value || busy\"\r\n (click)=\"parseOperations()\"\r\n >\r\n batch add\r\n </button>\r\n @if (parseError) {\r\n <span class=\"error\">{{ parseError }}</span>\r\n }\r\n </div>\r\n </div>\r\n <div id=\"batch-help\">\r\n <ul>\r\n <li>\r\n <strong>replace</strong>:\r\n <code>ATxRUN<strong>=</strong>\"VALUE\"</code>\r\n </li>\r\n <li>\r\n <strong>delete</strong>:\r\n <code>ATxRUN<strong>-</strong></code>\r\n </li>\r\n <li>\r\n <strong>add-before</strong>:\r\n <code>ATxRUN<strong>+[</strong>\"VALUE\"</code>\r\n </li>\r\n <li>\r\n <strong>add-after</strong>:\r\n <code>ATxRUN<strong>+]</strong>\"VALUE\"</code>\r\n </li>\r\n <li>\r\n <strong>move-before</strong>:\r\n <code>ATxRUN<strong>>[</strong>TO</code>\r\n </li>\r\n <li>\r\n <strong>move-after</strong>:\r\n <code>ATxRUN<strong>>]</strong>TO</code>\r\n </li>\r\n <li>\r\n <strong>swap</strong>:\r\n <code>ATxRUN<strong><></strong>TOxRUN</code>\r\n </li>\r\n <li>\r\n <strong>annotate</strong>:\r\n <code>ATxRUN<strong>:</strong></code>\r\n </li>\r\n </ul>\r\n <p>For all the operations:</p>\r\n <ul>\r\n <li>\r\n prefix <code>(ITAG:OTAG)</code> to define input and/or\r\n output tags, separated by colon.\r\n </li>\r\n <li>\r\n prepend <code>@</code> to AT to use character\r\n indexes (0-N) rather than IDs.\r\n </li>\r\n <li>RUN (where applicable) defaults to 1.</li>\r\n <li>\r\n append features like <code>[NAME OPERATOR VALUE]</code>,\r\n separated by space, where:\r\n </li>\r\n <ol>\r\n <li>\r\n NAME is an arbitrary string. Prefixes:\r\n <code>*</code>=global features, <code>!</code>=remove\r\n feature (no value). Suffixes:\r\n <code>^</code>=short-lived (like\r\n <code>*version^=alpha</code>).\r\n </li>\r\n <li>\r\n OPERATOR is <code>=</code> (multiple),\r\n <code>:=</code> (single),\r\n <code>==</code> (first-single).\r\n </li>\r\n <li>\r\n VALUE is a string. If it includes spaces, wrap it in\r\n <code>\"\"</code>.\r\n </li>\r\n </ol>\r\n </ul>\r\n </div>\r\n </div>\r\n </fieldset>\r\n </mat-expansion-panel>\r\n </div>\r\n </div>\r\n\r\n <!-- result -->\r\n @if (result) {\r\n <fieldset id=\"result\">\r\n <legend>result</legend>\r\n <gve-chain-result-view\r\n [result]=\"result\"\r\n (stepPick)=\"onStepPick($event)\"\r\n />\r\n </fieldset>\r\n }\r\n </div>\r\n </mat-tab>\r\n\r\n <!-- image -->\r\n <mat-tab label=\"image\">\r\n <div id=\"image\">\r\n <div id=\"image-ctl\">\r\n <!-- url -->\r\n <mat-form-field class=\"long-text\">\r\n <mat-label>URL</mat-label>\r\n <input matInput [formControl]=\"imageUrl\" />\r\n </mat-form-field>\r\n <div class=\"form-row\">\r\n <!-- x -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>X</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"imageX\"\r\n placeholder=\"X\"\r\n />\r\n </mat-form-field>\r\n <!-- y -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>Y</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"imageY\"\r\n placeholder=\"Y\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n <div class=\"form-row\">\r\n <!-- width -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>width</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"imageWidth\"\r\n />\r\n </mat-form-field>\r\n <!-- height -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"imageHeight\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n <!-- defs -->\r\n <div>\r\n <mat-form-field class=\"long-text\">\r\n <mat-label>defs</mat-label>\r\n <textarea matInput [formControl]=\"defs\" rows=\"3\"></textarea>\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n <div id=\"image-view\">\r\n @if (imageUrl.value) {\r\n <img alt=\"background\" [src]=\"imageUrl.value\" width=\"600\" />\r\n }\r\n </div>\r\n </div>\r\n </mat-tab>\r\n </mat-tab-group>\r\n\r\n <!-- preview -->\r\n @if (snapshot) {\r\n <fieldset id=\"preview\">\r\n <legend class=\"button-row\">\r\n <span>{{ viewTitle }}</span>\r\n <button\r\n color=\"primary\"\r\n mat-icon-button\r\n type=\"button\"\r\n matTooltip=\"Refresh preview\"\r\n (click)=\"updateViewData()\"\r\n >\r\n <mat-icon class=\"mat-primary\">refresh</mat-icon>\r\n </button>\r\n </legend>\r\n <gve-snapshot-view\r\n [data]=\"viewData\"\r\n (snapshotRender)=\"onSnapshotRender($any($event))\"\r\n (visualEvent)=\"onVisualEvent($any($event))\"\r\n />\r\n <div>\r\n <mat-button-toggle\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n matTooltip=\"Toggle rulers\"\r\n [checked]=\"rulers\"\r\n (change)=\"toggleRulers()\"\r\n >\r\n <mat-icon class=\"mat-primary\">straighten</mat-icon>\r\n </mat-button-toggle>\r\n </div>\r\n @if (visualInfo) {\r\n <div id=\"visual-info\">{{ visualInfo }}</div>\r\n }\r\n </fieldset>\r\n }\r\n\r\n <!--buttons -->\r\n <div class=\"form-row-center\">\r\n <button\r\n type=\"button\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Discard changes\"\r\n (click)=\"close()\"\r\n >\r\n <mat-icon>clear</mat-icon>\r\n close\r\n </button>\r\n <button\r\n type=\"submit\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Save changes\"\r\n [disabled]=\"form.invalid || form.pristine\"\r\n >\r\n <mat-icon>check_circle</mat-icon>\r\n save\r\n </button>\r\n </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}.button-row{display:flex;align-items:center;flex-wrap:wrap}.button-row *{flex:0 0 auto}.long-text{width:100%;max-width:800px}#text-range{margin:8px;border:1px solid silver;border-radius:6px;padding:6px}mat-expansion-panel{margin:8px 0}div#visual-info{font-size:95%;color:#909090;margin:8px}div#batch-container{display:grid;gap:16px;grid-template-rows:auto;grid-template-columns:1fr auto;grid-template-areas:\"batch-input batch-help\"}div#batch-input{grid-area:batch-input}div#batch-help{grid-area:batch-help}div#batch-help strong{color:#ff8c00}#list{margin:8px 0}#opStyle{margin-top:8px}table{width:100%;border-collapse:collapse}th{color:#909090;font-weight:400;text-align:left;background-color:#e1e0e0}th,td{padding:4px;border-bottom:1px solid silver}tbody tr:nth-child(2n){background-color:#e8e8e8}td.fit-width{width:1px;white-space:nowrap}tr.selected{background-color:#c8d9eb}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}div#image{display:grid;gap:8px;grid-template-rows:auto;grid-template-columns:1fr auto;grid-template-areas:\"image-ctl image-view\"}div#image-ctl{grid-area:image-ctl}div#image-view{grid-area:image-view}@media only screen and (max-width: 959px){div#batch-container{grid-template-columns:1fr;grid-template-areas:\"batch-input\" \"batch-help\"}div#image{grid-template-columns:1fr;grid-template-areas:\"image-ctl\" \"image-view\"}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i5.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.MinValidator, selector: "input[type=number][min][formControlName],input[type=number][min][formControl],input[type=number][min][ngModel]", inputs: ["min"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i6.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i6.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatButtonToggleModule }, { kind: "component", type: i7.MatButtonToggle, selector: "mat-button-toggle", inputs: ["aria-label", "aria-labelledby", "id", "name", "value", "tabIndex", "disableRipple", "appearance", "checked", "disabled"], outputs: ["change"], exportAs: ["matButtonToggle"] }, { kind: "ngmodule", type: MatCheckboxModule }, { kind: "ngmodule", type: MatExpansionModule }, { kind: "component", type: i8.MatExpansionPanel, selector: "mat-expansion-panel", inputs: ["hideToggle", "togglePosition"], outputs: ["afterExpand", "afterCollapse"], exportAs: ["matExpansionPanel"] }, { kind: "component", type: i8.MatExpansionPanelHeader, selector: "mat-expansion-panel-header", inputs: ["expandedHeight", "collapsedHeight", "tabIndex"] }, { kind: "directive", type: i8.MatExpansionPanelTitle, selector: "mat-panel-title" }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i9.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i9.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i10.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i11.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "ngmodule", type: MatProgressBarModule }, { kind: "component", type: i12.MatProgressBar, selector: "mat-progress-bar", inputs: ["color", "value", "bufferValue", "mode"], outputs: ["animationEnd"], exportAs: ["matProgressBar"] }, { kind: "ngmodule", type: MatSnackBarModule }, { kind: "ngmodule", type: MatTabsModule }, { kind: "component", type: i13.MatTab, selector: "mat-tab", inputs: ["disabled", "label", "aria-label", "aria-labelledby", "labelClass", "bodyClass"], exportAs: ["matTab"] }, { kind: "component", type: i13.MatTabGroup, selector: "mat-tab-group", inputs: ["color", "fitInkBarToContent", "mat-stretch-tabs", "dynamicHeight", "selectedIndex", "headerPosition", "animationDuration", "contentTabIndex", "disablePagination", "disableRipple", "preserveContent", "backgroundColor"], outputs: ["selectedIndexChange", "focusChange", "animationDone", "selectedTabChange"], exportAs: ["matTabGroup"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i14.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: NgToolsModule }, { kind: "pipe", type: i15.FlatLookupPipe, name: "flatLookup" }, { kind: "component", type: BaseTextEditorComponent, selector: "gve-base-text-editor", inputs: ["text"], outputs: ["textChange"] }, { kind: "component", type: ChainOperationEditorComponent, selector: "gve-chain-operation-editor", inputs: ["operation", "snapshot", "hidePreview"], outputs: ["operationChange", "operationPreview", "operationCancel"] }, { kind: "component", type: ChainResultViewComponent, selector: "gve-chain-result-view", inputs: ["result"], outputs: ["stepPick"] }, { kind: "component", type: LnHeightsEditorComponent, selector: "gve-ln-heights-editor", inputs: ["lineCount", "heights"], outputs: ["heightsChange"] }] }); }
|
|
476
|
+
}
|
|
477
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: SnapshotEditorComponent, decorators: [{
|
|
478
|
+
type: Component,
|
|
479
|
+
args: [{ selector: 'gve-snapshot-editor', standalone: true, imports: [
|
|
480
|
+
CommonModule,
|
|
481
|
+
FormsModule,
|
|
482
|
+
ReactiveFormsModule,
|
|
483
|
+
MatButtonModule,
|
|
484
|
+
MatButtonToggleModule,
|
|
485
|
+
MatCheckboxModule,
|
|
486
|
+
MatExpansionModule,
|
|
487
|
+
MatFormFieldModule,
|
|
488
|
+
MatIconModule,
|
|
489
|
+
MatInputModule,
|
|
490
|
+
MatProgressBarModule,
|
|
491
|
+
MatSnackBarModule,
|
|
492
|
+
MatTabsModule,
|
|
493
|
+
MatTooltipModule,
|
|
494
|
+
NgToolsModule,
|
|
495
|
+
BaseTextEditorComponent,
|
|
496
|
+
BaseTextViewComponent,
|
|
497
|
+
ChainOperationEditorComponent,
|
|
498
|
+
ChainResultViewComponent,
|
|
499
|
+
LnHeightsEditorComponent,
|
|
500
|
+
], schemas: [CUSTOM_ELEMENTS_SCHEMA], template: "<form [formGroup]=\"form\" (submit)=\"save()\">\r\n <mat-tab-group>\r\n <!-- text -->\r\n <mat-tab label=\"text\">\r\n <mat-expansion-panel [expanded]=\"!baseText.value\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title> base text </mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <gve-base-text-editor\r\n [text]=\"baseText.value\"\r\n (textChange)=\"onTextChange($event)\"\r\n />\r\n <fieldset>\r\n <div class=\"form-row\">\r\n <!-- offsetX -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>X offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"offsetX\"\r\n placeholder=\"X offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- offsetY -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>Y offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"offsetY\"\r\n placeholder=\"Y offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- lineHeightOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>ln h-offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"lineHeightOffset\"\r\n placeholder=\"ln h-offset\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- charSpacingOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>char spacing</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"charSpacingOffset\"\r\n placeholder=\"char spacing\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- spcWidthOffset -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>spc w-offset</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"spcWidthOffset\"\r\n placeholder=\"spc w-offset\"\r\n />\r\n </mat-form-field>\r\n <!-- minLineHeights -->\r\n <div>\r\n <gve-ln-heights-editor\r\n [lineCount]=\"lineCount\"\r\n [heights]=\"lnHeights.value\"\r\n (heightsChange)=\"onHeightsChange($event)\"\r\n />\r\n </div>\r\n </div>\r\n <!-- textStyle -->\r\n <div>\r\n <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n <mat-label>text style</mat-label>\r\n <textarea\r\n matInput\r\n [formControl]=\"textStyle\"\r\n placeholder=\"text style\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n </fieldset>\r\n </mat-expansion-panel>\r\n </mat-tab>\r\n\r\n <!-- operations -->\r\n <mat-tab label=\"operations\">\r\n <div id=\"snapshot-container\">\r\n <div id=\"general\">\r\n <div class=\"form-row\">\r\n <!-- width -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>width</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"width\"\r\n placeholder=\"width\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- height -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"height\"\r\n placeholder=\"height\"\r\n />\r\n </mat-form-field>\r\n\r\n <!-- add op -->\r\n <div>\r\n <button\r\n type=\"button\"\r\n mat-flat-button\r\n color=\"primary\"\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 [ngClass]=\"{ selected: operation.id === resultOperationId }\">\r\n <td class=\"fit-width\">\r\n <!-- edit -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n class=\"mat-primary\"\r\n (click)=\"editOperation(index)\"\r\n matTooltip=\"Edit operation\"\r\n >\r\n <mat-icon>edit</mat-icon>\r\n </button>\r\n <!-- delete -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"warn\"\r\n (click)=\"deleteOperation(index)\"\r\n matTooltip=\"Delete operation\"\r\n >\r\n <mat-icon class=\"mat-warn\">delete</mat-icon>\r\n </button>\r\n <!-- run to -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"accent\"\r\n (click)=\"runTo(index)\"\r\n matTooltip=\"Run to this operation\"\r\n >\r\n <mat-icon class=\"mat-accent\">subscriptions</mat-icon>\r\n </button>\r\n </td>\r\n <td>{{ operation.id }}</td>\r\n <td>{{ operation.type | flatLookup : opTypeMap }}</td>\r\n <td>{{ operation.at }}</td>\r\n <td>{{ operation.run }}</td>\r\n <td>{{ operation.value }}</td>\r\n <td>{{ operation.inputTag }}</td>\r\n <td>{{ operation.outputTag }}</td>\r\n <td>{{ operation.groupId }}</td>\r\n <td>{{ operation.features?.length }}</td>\r\n </tr>\r\n }\r\n </tbody>\r\n </table>\r\n }\r\n\r\n <!-- operation editor -->\r\n <mat-expansion-panel [expanded]=\"editedOp\" [disabled]=\"!editedOp\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title>operation {{ editedOp?.id }}</mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <fieldset>\r\n <gve-chain-operation-editor\r\n [operation]=\"editedOp\"\r\n (operationCancel)=\"onOperationCancel()\"\r\n (operationChange)=\"onOperationChange($event)\"\r\n (operationPreview)=\"onOperationPreview($event)\"\r\n />\r\n </fieldset>\r\n </mat-expansion-panel>\r\n\r\n <div>\r\n <mat-progress-bar mode=\"indeterminate\" *ngIf=\"busy\" />\r\n </div>\r\n\r\n <!-- opStyle -->\r\n <div id=\"opStyle\">\r\n <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n <mat-label>operations style</mat-label>\r\n <textarea\r\n matInput\r\n [formControl]=\"opStyle\"\r\n placeholder=\"operations style\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- batch ops -->\r\n <div>\r\n <mat-expansion-panel>\r\n <mat-expansion-panel-header>\r\n <mat-panel-title>batch operations</mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <fieldset>\r\n <div id=\"batch-container\">\r\n <div id=\"batch-input\">\r\n <div>\r\n <mat-form-field class=\"full-width\">\r\n <mat-label>operations</mat-label>\r\n <textarea\r\n class=\"code\"\r\n matInput\r\n [formControl]=\"inputOps\"\r\n placeholder=\"operations\"\r\n rows=\"8\"\r\n spellcheck=\"false\"\r\n ></textarea>\r\n </mat-form-field>\r\n </div>\r\n <div class=\"form-row\">\r\n <button\r\n type=\"button\"\r\n color=\"warn\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Remove all the operations\"\r\n [disabled]=\"!operations.value.length || busy\"\r\n (click)=\"clearOperations()\"\r\n >\r\n clear\r\n </button>\r\n <button\r\n type=\"button\"\r\n color=\"primary\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Parse text and add results to the operations\"\r\n [disabled]=\"!inputOps.value || busy\"\r\n (click)=\"parseOperations()\"\r\n >\r\n batch add\r\n </button>\r\n @if (parseError) {\r\n <span class=\"error\">{{ parseError }}</span>\r\n }\r\n </div>\r\n </div>\r\n <div id=\"batch-help\">\r\n <ul>\r\n <li>\r\n <strong>replace</strong>:\r\n <code>ATxRUN<strong>=</strong>\"VALUE\"</code>\r\n </li>\r\n <li>\r\n <strong>delete</strong>:\r\n <code>ATxRUN<strong>-</strong></code>\r\n </li>\r\n <li>\r\n <strong>add-before</strong>:\r\n <code>ATxRUN<strong>+[</strong>\"VALUE\"</code>\r\n </li>\r\n <li>\r\n <strong>add-after</strong>:\r\n <code>ATxRUN<strong>+]</strong>\"VALUE\"</code>\r\n </li>\r\n <li>\r\n <strong>move-before</strong>:\r\n <code>ATxRUN<strong>>[</strong>TO</code>\r\n </li>\r\n <li>\r\n <strong>move-after</strong>:\r\n <code>ATxRUN<strong>>]</strong>TO</code>\r\n </li>\r\n <li>\r\n <strong>swap</strong>:\r\n <code>ATxRUN<strong><></strong>TOxRUN</code>\r\n </li>\r\n <li>\r\n <strong>annotate</strong>:\r\n <code>ATxRUN<strong>:</strong></code>\r\n </li>\r\n </ul>\r\n <p>For all the operations:</p>\r\n <ul>\r\n <li>\r\n prefix <code>(ITAG:OTAG)</code> to define input and/or\r\n output tags, separated by colon.\r\n </li>\r\n <li>\r\n prepend <code>@</code> to AT to use character\r\n indexes (0-N) rather than IDs.\r\n </li>\r\n <li>RUN (where applicable) defaults to 1.</li>\r\n <li>\r\n append features like <code>[NAME OPERATOR VALUE]</code>,\r\n separated by space, where:\r\n </li>\r\n <ol>\r\n <li>\r\n NAME is an arbitrary string. Prefixes:\r\n <code>*</code>=global features, <code>!</code>=remove\r\n feature (no value). Suffixes:\r\n <code>^</code>=short-lived (like\r\n <code>*version^=alpha</code>).\r\n </li>\r\n <li>\r\n OPERATOR is <code>=</code> (multiple),\r\n <code>:=</code> (single),\r\n <code>==</code> (first-single).\r\n </li>\r\n <li>\r\n VALUE is a string. If it includes spaces, wrap it in\r\n <code>\"\"</code>.\r\n </li>\r\n </ol>\r\n </ul>\r\n </div>\r\n </div>\r\n </fieldset>\r\n </mat-expansion-panel>\r\n </div>\r\n </div>\r\n\r\n <!-- result -->\r\n @if (result) {\r\n <fieldset id=\"result\">\r\n <legend>result</legend>\r\n <gve-chain-result-view\r\n [result]=\"result\"\r\n (stepPick)=\"onStepPick($event)\"\r\n />\r\n </fieldset>\r\n }\r\n </div>\r\n </mat-tab>\r\n\r\n <!-- image -->\r\n <mat-tab label=\"image\">\r\n <div id=\"image\">\r\n <div id=\"image-ctl\">\r\n <!-- url -->\r\n <mat-form-field class=\"long-text\">\r\n <mat-label>URL</mat-label>\r\n <input matInput [formControl]=\"imageUrl\" />\r\n </mat-form-field>\r\n <div class=\"form-row\">\r\n <!-- x -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>X</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"imageX\"\r\n placeholder=\"X\"\r\n />\r\n </mat-form-field>\r\n <!-- y -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>Y</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n [formControl]=\"imageY\"\r\n placeholder=\"Y\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n <div class=\"form-row\">\r\n <!-- width -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>width</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"imageWidth\"\r\n />\r\n </mat-form-field>\r\n <!-- height -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input\r\n matInput\r\n type=\"number\"\r\n min=\"0\"\r\n [formControl]=\"imageHeight\"\r\n />\r\n </mat-form-field>\r\n </div>\r\n <!-- defs -->\r\n <div>\r\n <mat-form-field class=\"long-text\">\r\n <mat-label>defs</mat-label>\r\n <textarea matInput [formControl]=\"defs\" rows=\"3\"></textarea>\r\n </mat-form-field>\r\n </div>\r\n </div>\r\n <div id=\"image-view\">\r\n @if (imageUrl.value) {\r\n <img alt=\"background\" [src]=\"imageUrl.value\" width=\"600\" />\r\n }\r\n </div>\r\n </div>\r\n </mat-tab>\r\n </mat-tab-group>\r\n\r\n <!-- preview -->\r\n @if (snapshot) {\r\n <fieldset id=\"preview\">\r\n <legend class=\"button-row\">\r\n <span>{{ viewTitle }}</span>\r\n <button\r\n color=\"primary\"\r\n mat-icon-button\r\n type=\"button\"\r\n matTooltip=\"Refresh preview\"\r\n (click)=\"updateViewData()\"\r\n >\r\n <mat-icon class=\"mat-primary\">refresh</mat-icon>\r\n </button>\r\n </legend>\r\n <gve-snapshot-view\r\n [data]=\"viewData\"\r\n (snapshotRender)=\"onSnapshotRender($any($event))\"\r\n (visualEvent)=\"onVisualEvent($any($event))\"\r\n />\r\n <div>\r\n <mat-button-toggle\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n matTooltip=\"Toggle rulers\"\r\n [checked]=\"rulers\"\r\n (change)=\"toggleRulers()\"\r\n >\r\n <mat-icon class=\"mat-primary\">straighten</mat-icon>\r\n </mat-button-toggle>\r\n </div>\r\n @if (visualInfo) {\r\n <div id=\"visual-info\">{{ visualInfo }}</div>\r\n }\r\n </fieldset>\r\n }\r\n\r\n <!--buttons -->\r\n <div class=\"form-row-center\">\r\n <button\r\n type=\"button\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Discard changes\"\r\n (click)=\"close()\"\r\n >\r\n <mat-icon>clear</mat-icon>\r\n close\r\n </button>\r\n <button\r\n type=\"submit\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Save changes\"\r\n [disabled]=\"form.invalid || form.pristine\"\r\n >\r\n <mat-icon>check_circle</mat-icon>\r\n save\r\n </button>\r\n </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}.button-row{display:flex;align-items:center;flex-wrap:wrap}.button-row *{flex:0 0 auto}.long-text{width:100%;max-width:800px}#text-range{margin:8px;border:1px solid silver;border-radius:6px;padding:6px}mat-expansion-panel{margin:8px 0}div#visual-info{font-size:95%;color:#909090;margin:8px}div#batch-container{display:grid;gap:16px;grid-template-rows:auto;grid-template-columns:1fr auto;grid-template-areas:\"batch-input batch-help\"}div#batch-input{grid-area:batch-input}div#batch-help{grid-area:batch-help}div#batch-help strong{color:#ff8c00}#list{margin:8px 0}#opStyle{margin-top:8px}table{width:100%;border-collapse:collapse}th{color:#909090;font-weight:400;text-align:left;background-color:#e1e0e0}th,td{padding:4px;border-bottom:1px solid silver}tbody tr:nth-child(2n){background-color:#e8e8e8}td.fit-width{width:1px;white-space:nowrap}tr.selected{background-color:#c8d9eb}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}div#image{display:grid;gap:8px;grid-template-rows:auto;grid-template-columns:1fr auto;grid-template-areas:\"image-ctl image-view\"}div#image-ctl{grid-area:image-ctl}div#image-view{grid-area:image-view}@media only screen and (max-width: 959px){div#batch-container{grid-template-columns:1fr;grid-template-areas:\"batch-input\" \"batch-help\"}div#image{grid-template-columns:1fr;grid-template-areas:\"image-ctl\" \"image-view\"}}\n"] }]
|
|
501
|
+
}], ctorParameters: () => [{ type: i1.FormBuilder }, { type: i2.GveApiService }, { type: i3.DialogService }, { type: i4.MatSnackBar }], propDecorators: { snapshot: [{
|
|
502
|
+
type: Input
|
|
503
|
+
}], batchOps: [{
|
|
504
|
+
type: Input
|
|
505
|
+
}], snapshotChange: [{
|
|
506
|
+
type: Output
|
|
507
|
+
}], snapshotCancel: [{
|
|
508
|
+
type: Output
|
|
509
|
+
}] } });
|
|
510
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snapshot-editor.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/gve-core/src/lib/components/snapshot-editor/snapshot-editor.component.ts","../../../../../../../projects/myrmidon/gve-core/src/lib/components/snapshot-editor/snapshot-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  CUSTOM_ELEMENTS_SCHEMA,\r\n  Component,\r\n  EventEmitter,\r\n  Input,\r\n  OnInit,\r\n  Output,\r\n} from '@angular/core';\r\nimport {\r\n  FormBuilder,\r\n  FormControl,\r\n  FormGroup,\r\n  FormsModule,\r\n  ReactiveFormsModule,\r\n  Validators,\r\n} from '@angular/forms';\r\nimport { CommonModule } from '@angular/common';\r\n\r\nimport { MatButtonModule } from '@angular/material/button';\r\nimport { MatButtonToggleModule } from '@angular/material/button-toggle';\r\nimport { MatCheckboxModule } from '@angular/material/checkbox';\r\nimport { MatExpansionModule } from '@angular/material/expansion';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatIconModule } from '@angular/material/icon';\r\nimport { MatInputModule } from '@angular/material/input';\r\nimport { MatProgressBarModule } from '@angular/material/progress-bar';\r\nimport { MatSnackBar, MatSnackBarModule } from '@angular/material/snack-bar';\r\nimport { MatTabsModule } from '@angular/material/tabs';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\n\r\nimport { 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  GveVisualEvent,\r\n  Snapshot,\r\n  SnapshotViewData,\r\n  SnapshotViewRenderEvent,\r\n  SnapshotViewService,\r\n} from '@myrmidon/gve-snapshot-view';\r\n\r\nimport {\r\n  BaseTextViewComponent,\r\n  VarBaseTextRange,\r\n} from '../base-text-view/base-text-view.component';\r\nimport { ChainOperationEditorComponent } from '../chain-operation-editor/chain-operation-editor.component';\r\nimport {\r\n  ChainOperationContextStep,\r\n  CharChainResult,\r\n  GveApiService,\r\n} from '../../services/gve-api.service';\r\nimport { ChainResultViewComponent } from '../chain-result-view/chain-result-view.component';\r\nimport { LnHeightsEditorComponent } from '../ln-heights-editor/ln-heights-editor.component';\r\nimport { BaseTextEditorComponent } from '../base-text-editor/base-text-editor.component';\r\n\r\nconst VIEW_TITLE = 'preview';\r\n\r\n/**\r\n * A component to edit a text snapshot.\r\n */\r\n@Component({\r\n  selector: 'gve-snapshot-editor',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    FormsModule,\r\n    ReactiveFormsModule,\r\n    MatButtonModule,\r\n    MatButtonToggleModule,\r\n    MatCheckboxModule,\r\n    MatExpansionModule,\r\n    MatFormFieldModule,\r\n    MatIconModule,\r\n    MatInputModule,\r\n    MatProgressBarModule,\r\n    MatSnackBarModule,\r\n    MatTabsModule,\r\n    MatTooltipModule,\r\n    NgToolsModule,\r\n    BaseTextEditorComponent,\r\n    BaseTextViewComponent,\r\n    ChainOperationEditorComponent,\r\n    ChainResultViewComponent,\r\n    LnHeightsEditorComponent,\r\n  ],\r\n  schemas: [CUSTOM_ELEMENTS_SCHEMA],\r\n  templateUrl: './snapshot-editor.component.html',\r\n  styleUrl: './snapshot-editor.component.css',\r\n})\r\nexport class SnapshotEditorComponent implements OnInit {\r\n  private _snapshot?: Snapshot;\r\n  private _renderer?: SnapshotViewService;\r\n\r\n  /**\r\n   * The snapshot to edit.\r\n   */\r\n  @Input()\r\n  public get snapshot(): Snapshot | undefined {\r\n    return this._snapshot;\r\n  }\r\n  public set snapshot(value: Snapshot | undefined | null) {\r\n    if (this._snapshot === value) {\r\n      return;\r\n    }\r\n    this._snapshot = value || undefined;\r\n    this.updateForm(this._snapshot);\r\n    this.updateViewData();\r\n  }\r\n\r\n  /**\r\n   * The batch operations text to set for the editor.\r\n   */\r\n  @Input()\r\n  public batchOps?: string;\r\n\r\n  /**\r\n   * Emitted when the user saves the edited snapshot.\r\n   */\r\n  @Output()\r\n  public readonly snapshotChange: EventEmitter<Snapshot> = new EventEmitter();\r\n\r\n  /**\r\n   * Emitted when the user cancels the snapshot editing.\r\n   */\r\n  @Output()\r\n  public readonly snapshotCancel = new EventEmitter<void>();\r\n\r\n  // general\r\n  public width: FormControl<number>;\r\n  public height: FormControl<number>;\r\n  public style: FormControl<string | null>;\r\n  // base text\r\n  public baseText: FormControl<CharNode[] | string>;\r\n  public offsetX: FormControl<number>;\r\n  public offsetY: FormControl<number>;\r\n  public lineHeightOffset: FormControl<number>;\r\n  public lnHeights: FormControl<Record<number, number> | null>;\r\n  public charSpacingOffset: FormControl<number>;\r\n  public spcWidthOffset: FormControl<number>;\r\n  public textStyle: FormControl<string | null>;\r\n  // operations\r\n  public operations: FormControl<CharChainOperation[]>;\r\n  public inputOps: FormControl<string | null>;\r\n  public opStyle: FormControl<string | null>;\r\n  public form: FormGroup;\r\n  // image and defs\r\n  public imageUrl: FormControl<string | null>;\r\n  public imageOpacity: FormControl<number>;\r\n  public imageX: FormControl<number>;\r\n  public imageY: FormControl<number>;\r\n  public imageWidth: FormControl<number>;\r\n  public imageHeight: FormControl<number>;\r\n  public defs: FormControl<string | null>;\r\n\r\n  public textRange?: VarBaseTextRange;\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: 'addBefore',\r\n    3: 'addAfter',\r\n    4: 'moveBefore',\r\n    5: 'moveAfter',\r\n    6: 'swap',\r\n    7: 'annotate',\r\n  };\r\n  public lineCount = 0;\r\n\r\n  public viewTitle = VIEW_TITLE;\r\n  public viewData?: SnapshotViewData;\r\n  public visualInfo?: string;\r\n  public rulers = true;\r\n  // run result\r\n  public result?: CharChainResult;\r\n  public resultOperationId?: string;\r\n\r\n  constructor(\r\n    formBuilder: FormBuilder,\r\n    private _api: GveApiService,\r\n    private _dialogService: DialogService,\r\n    private _snackbar: MatSnackBar\r\n  ) {\r\n    // general\r\n    this.width = new FormControl<number>(800, { nonNullable: true });\r\n    this.height = new FormControl<number>(600, { nonNullable: true });\r\n    this.style = new FormControl<string | null>(null);\r\n    // base text\r\n    this.baseText = new FormControl<CharNode[] | string>([], {\r\n      nonNullable: true,\r\n      validators: [Validators.required],\r\n    });\r\n    this.offsetX = new FormControl<number>(0, { nonNullable: true });\r\n    this.offsetY = new FormControl<number>(0, { nonNullable: true });\r\n    this.lineHeightOffset = new FormControl<number>(\r\n      DEFAULT_SVG_BASE_TEXT_OPTIONS.lineHeightOffset,\r\n      { nonNullable: true }\r\n    );\r\n    this.lnHeights = new FormControl<Record<number, number> | null>(null);\r\n    this.charSpacingOffset = new FormControl<number>(0, { nonNullable: true });\r\n    this.spcWidthOffset = new FormControl<number>(0, { nonNullable: true });\r\n    this.textStyle = new FormControl<string | null>(null);\r\n\r\n    // operations\r\n    this.operations = new FormControl<CharChainOperation[]>([], {\r\n      nonNullable: true,\r\n    });\r\n    this.inputOps = new FormControl<string | null>(null);\r\n    this.opStyle = new FormControl<string | null>(null);\r\n    // image\r\n    this.imageUrl = new FormControl<string | null>(null);\r\n    this.imageOpacity = new FormControl<number>(1, { nonNullable: true });\r\n    this.imageX = new FormControl<number>(0, { nonNullable: true });\r\n    this.imageY = new FormControl<number>(0, { nonNullable: true });\r\n    this.imageWidth = new FormControl<number>(0, { nonNullable: true });\r\n    this.imageHeight = new FormControl<number>(0, { nonNullable: true });\r\n    this.defs = new FormControl<string | null>(null);\r\n\r\n    this.form = formBuilder.group({\r\n      width: this.width,\r\n      height: this.height,\r\n      style: this.style,\r\n      baseText: this.baseText,\r\n      offsetX: this.offsetX,\r\n      offsetY: this.offsetY,\r\n      lineHeightOffset: this.lineHeightOffset,\r\n      lnHeights: this.lnHeights,\r\n      charSpacingOffset: this.charSpacingOffset,\r\n      spcWidthOffset: this.spcWidthOffset,\r\n      textStyle: this.textStyle,\r\n      operations: this.operations,\r\n      inputOps: this.inputOps,\r\n      opStyle: this.opStyle,\r\n      // image\r\n      imageUrl: this.imageUrl,\r\n      imageOpacity: this.imageOpacity,\r\n      imageX: this.imageX,\r\n      imageY: this.imageY,\r\n      imageWidth: this.imageWidth,\r\n      imageHeight: this.imageHeight,\r\n      defs: this.defs,\r\n    });\r\n  }\r\n\r\n  public ngOnInit(): void {\r\n    if (this.batchOps) {\r\n      this.inputOps.setValue(this.batchOps);\r\n      this.parseOperations();\r\n    }\r\n  }\r\n\r\n  public updateViewData(\r\n    newOperation?: CharChainOperation,\r\n    snapshot?: Snapshot,\r\n    title = VIEW_TITLE\r\n  ): void {\r\n    if (!snapshot) {\r\n      snapshot = this.getSnapshot();\r\n    }\r\n\r\n    // update or add new operation in copy\r\n    if (newOperation) {\r\n      const index = snapshot.operations.findIndex(\r\n        (op) => op.id === newOperation.id\r\n      );\r\n      if (index > -1) {\r\n        snapshot.operations.splice(index, 1, newOperation);\r\n      } else {\r\n        snapshot.operations.push(newOperation);\r\n      }\r\n    }\r\n\r\n    // update view data\r\n    this.viewTitle = title;\r\n    this.visualInfo = undefined;\r\n    this.viewData = {\r\n      snapshot: snapshot,\r\n      options: {\r\n        // debug: true,\r\n        delayedRender: true,\r\n        showRulers: true,\r\n        showGrid: true,\r\n        panZoom: true,\r\n      },\r\n    };\r\n  }\r\n\r\n  private updateForm(snapshot: Snapshot | undefined) {\r\n    if (!snapshot) {\r\n      this.form.reset();\r\n    } else {\r\n      this.width.setValue(snapshot.size.width);\r\n      this.height.setValue(snapshot.size.height);\r\n      this.style.setValue(snapshot.style || null);\r\n      this.imageUrl.setValue(snapshot.image?.url || null);\r\n      this.imageOpacity.setValue(snapshot.image?.opacity || 0);\r\n      this.imageX.setValue(snapshot.image?.canvas?.x || 0);\r\n      this.imageY.setValue(snapshot.image?.canvas?.y || 0);\r\n      this.imageWidth.setValue(snapshot.image?.canvas?.width || 0);\r\n      this.imageHeight.setValue(snapshot.image?.canvas?.height || 0);\r\n      this.defs.setValue(snapshot.defs || null);\r\n      this.baseText.setValue(snapshot.text);\r\n      this.offsetX.setValue(snapshot.textOptions?.offset?.x || 0);\r\n      this.offsetY.setValue(snapshot.textOptions?.offset?.y || 0);\r\n      this.lineHeightOffset.setValue(snapshot.textOptions?.lineHeightOffset);\r\n      this.lnHeights.setValue(snapshot.textOptions?.minLineHeights || null);\r\n      this.charSpacingOffset.setValue(snapshot.textOptions?.charSpacingOffset);\r\n      this.spcWidthOffset.setValue(snapshot.textOptions?.spcWidthOffset);\r\n      this.textStyle.setValue(snapshot.textStyle || null);\r\n      this.operations.setValue(snapshot.operations);\r\n      this.inputOps.reset();\r\n      this.opStyle.setValue(snapshot.opStyle || null);\r\n    }\r\n    this.form.markAsPristine();\r\n  }\r\n\r\n  public editNewOperation() {\r\n    this.editedOp = {\r\n      id: '',\r\n      type: 0,\r\n      at: this.textRange?.at || 0,\r\n      run: this.textRange?.run || 1,\r\n    };\r\n    this.editedOpIndex = -1;\r\n  }\r\n\r\n  public editOperation(index: number) {\r\n    this.editedOp = this.operations.value[index];\r\n    this.editedOpIndex = index;\r\n  }\r\n\r\n  private closeEditedOp() {\r\n    this.editedOp = undefined;\r\n    this.editedOpIndex = -1;\r\n  }\r\n\r\n  public onOperationCancel() {\r\n    this.closeEditedOp();\r\n  }\r\n\r\n  public onOperationChange(op: CharChainOperation) {\r\n    const operations = [...this.operations.value];\r\n\r\n    if (this.editedOpIndex > -1) {\r\n      operations.splice(this.editedOpIndex, 1, op);\r\n    } else {\r\n      operations.push(op);\r\n    }\r\n    this.operations.setValue(operations);\r\n    this.operations.markAsDirty();\r\n    this.operations.updateValueAndValidity();\r\n\r\n    this.closeEditedOp();\r\n  }\r\n\r\n  public deleteOperation(index: number) {\r\n    this._dialogService\r\n      .confirm(\r\n        'Confirm Deletion',\r\n        `Delete operation \"${this.operations.value[index].id}\"?`\r\n      )\r\n      .subscribe((yes: boolean) => {\r\n        if (yes) {\r\n          if (this.resultOperationId === this.operations.value[index].id) {\r\n            this.resultOperationId = undefined;\r\n          }\r\n          const operations = [...this.operations.value];\r\n          operations.splice(index, 1);\r\n          this.operations.setValue(operations);\r\n          this.operations.markAsDirty();\r\n          this.operations.updateValueAndValidity();\r\n        }\r\n      });\r\n  }\r\n\r\n  public onTextChange(text: CharNode[]) {\r\n    this.baseText.setValue(text);\r\n    this.baseText.updateValueAndValidity();\r\n    this.baseText.markAsDirty();\r\n    if (!text.length) {\r\n      this.lineCount = 0;\r\n    } else {\r\n      this.lineCount = text.filter((c) => c.data === '\\n').length + 1;\r\n    }\r\n    this.textRange = undefined;\r\n    this.closeEditedOp();\r\n    this.operations.reset();\r\n    this.updateViewData();\r\n  }\r\n\r\n  public onRangePick(range: VarBaseTextRange) {\r\n    this.textRange = range;\r\n  }\r\n\r\n  public parseOperations() {\r\n    if (this.busy || !this.inputOps.value) {\r\n      return;\r\n    }\r\n    this._dialogService\r\n      .confirm('Confirm', 'Parse operations?')\r\n      .subscribe((yes: boolean) => {\r\n        if (yes) {\r\n          this.busy = true;\r\n          this.parseError = undefined;\r\n\r\n          this._api.parseOperations(this.inputOps.value!).subscribe({\r\n            next: (wrapper) => {\r\n              const operations = [...this.operations.value];\r\n              operations.push(...wrapper.result!);\r\n              this.operations.setValue(operations);\r\n              this.operations.markAsDirty();\r\n              this.operations.updateValueAndValidity();\r\n              this.updateViewData();\r\n            },\r\n            error: (error) => {\r\n              this.parseError = error.message;\r\n            },\r\n            complete: () => {\r\n              this.busy = false;\r\n            },\r\n          });\r\n        }\r\n      });\r\n  }\r\n\r\n  public clearOperations(): void {\r\n    this._dialogService\r\n      .confirm('Confirm', 'Remove all operations?')\r\n      .subscribe((yes: boolean) => {\r\n        if (yes) {\r\n          this.resultOperationId = undefined;\r\n          this.operations.reset();\r\n          this.operations.markAsDirty();\r\n          this.operations.updateValueAndValidity();\r\n          this.updateViewData();\r\n        }\r\n      });\r\n  }\r\n\r\n  public onOperationPreview(operation: CharChainOperation): void {\r\n    this.updateViewData(operation);\r\n  }\r\n\r\n  public runTo(index: number): void {\r\n    const snapshot = this.getSnapshot();\r\n    if (!snapshot.text) {\r\n      return;\r\n    }\r\n    const operations = this.operations.value.slice(0, index + 1);\r\n    this.busy = true;\r\n    this._api.runOperations(snapshot.text as CharNode[], operations).subscribe({\r\n      next: (wrapper) => {\r\n        if (wrapper.error) {\r\n          this._snackbar.open(wrapper.error, 'OK');\r\n          return;\r\n        }\r\n        this.result = wrapper.result!;\r\n        // TODO update view\r\n        // this.updateViewData(\r\n        //   undefined,\r\n        //   snapshot,\r\n        //   this.operations.value[index].outputTag\r\n        // );\r\n      },\r\n      error: (error) => {\r\n        console.error(error);\r\n        this._snackbar.open('Error running operations', 'OK');\r\n      },\r\n      complete: () => {\r\n        this.busy = false;\r\n      },\r\n    });\r\n  }\r\n\r\n  public onStepPick(step: ChainOperationContextStep): void {\r\n    if (!this.result) {\r\n      return;\r\n    }\r\n    this.resultOperationId = step.operation.id;\r\n\r\n    // get snapshot for preview by updating its text and\r\n    // removing operations past the last executed\r\n    const stepNodes = this.result.taggedNodes[step.outputTag];\r\n\r\n    // we always display the original base text, just updating\r\n    // its nodes features to those of the picked step, and adding\r\n    // new nodes if they are not in the original text. It is assumed\r\n    // that these new nodes have a manually set position.\r\n    const snapshot = this.getSnapshot();\r\n    const nodes: CharNode[] = deepCopy(snapshot.text as CharNode[]);\r\n    for (const sn of stepNodes) {\r\n      const i = nodes.findIndex((n) => n.id === sn.id);\r\n      if (i > -1) {\r\n        nodes[i].features = sn.features;\r\n      } else {\r\n        nodes.push(sn);\r\n      }\r\n    }\r\n\r\n    // find max ID in snapshot.text (=original) nodes,\r\n    // so that any ID greater than it belongs to new nodes\r\n    const maxOriginalId = (snapshot.text as CharNode[]).reduce(\r\n      (max, n) => Math.max(max, n.id),\r\n      0\r\n    );\r\n\r\n    // append nodes introduced before the picked step if any\r\n    // (those introduced by the picked step have already been added)\r\n    const stepIndex = this.result.steps.indexOf(step);\r\n    for (let i = 0; i < stepIndex; i++) {\r\n      const stepNodes = this.result.taggedNodes[this.result.steps[i].outputTag];\r\n      for (const n of stepNodes) {\r\n        if (n.id > maxOriginalId && !nodes.find((x) => x.id === n.id)) {\r\n          nodes.push(n);\r\n        }\r\n      }\r\n    }\r\n\r\n    snapshot.text = nodes;\r\n    const i = snapshot.operations.findIndex((o) => o.id === step.operation.id);\r\n    if (i > -1) {\r\n      snapshot.operations = snapshot.operations.slice(0, i + 1);\r\n    }\r\n    this.updateViewData(undefined, snapshot, step.outputTag);\r\n  }\r\n\r\n  public onVisualEvent(event: CustomEvent<GveVisualEvent>): void {\r\n    const d = event.detail;\r\n    if (d.event.type === 'mouseover') {\r\n      const visual = d.source;\r\n      const sb: string[] = [];\r\n      sb.push('#' + visual.id);\r\n      sb.push(` (${visual.type})`);\r\n      if (visual.data?.label) {\r\n        sb.push(': ');\r\n        sb.push(visual.data.label);\r\n      }\r\n      if (visual.data?.features?.length) {\r\n        sb.push(\r\n          visual.data.features\r\n            .map((f: Feature) => `${f.name}=${f.value}`)\r\n            .join('\\n')\r\n        );\r\n      }\r\n      this.visualInfo = sb.join('');\r\n    }\r\n  }\r\n\r\n  public onHeightsChange(heights: Record<number, number> | undefined): void {\r\n    this.lnHeights.setValue(heights || null);\r\n    this.lnHeights.markAsDirty();\r\n    this.lnHeights.updateValueAndValidity();\r\n  }\r\n\r\n  public close(): void {\r\n    this.snapshotCancel.emit();\r\n  }\r\n\r\n  public onSnapshotRender(event: CustomEvent<SnapshotViewRenderEvent>): void {\r\n    this._renderer = (event.detail as SnapshotViewRenderEvent).renderer;\r\n  }\r\n\r\n  public toggleRulers(): void {\r\n    if (!this._renderer) {\r\n      return;\r\n    }\r\n    this.rulers = this._renderer.toggleRulers();\r\n  }\r\n\r\n  private getSnapshot(): Snapshot {\r\n    const snapshot: Snapshot = {\r\n      size: {\r\n        width: this.width.value,\r\n        height: this.height.value,\r\n      },\r\n      style: this.style.value || undefined,\r\n      defs: this.defs.value || undefined,\r\n      text: this.baseText.value!,\r\n      textStyle: this.textStyle.value || undefined,\r\n      textOptions: {\r\n        lineHeightOffset: this.lineHeightOffset.value,\r\n        minLineHeights: this.lnHeights.value || undefined,\r\n        charSpacingOffset: this.charSpacingOffset.value,\r\n        spcWidthOffset: this.spcWidthOffset.value,\r\n        offset: {\r\n          x: this.offsetX.value,\r\n          y: this.offsetY.value,\r\n        },\r\n      },\r\n      operations: this.operations.value,\r\n      opStyle: this.opStyle.value || undefined,\r\n      timelines: {}, // TODO\r\n    };\r\n\r\n    // image\r\n    if (this.imageUrl.value) {\r\n      snapshot.image = {\r\n        url: this.imageUrl.value,\r\n        opacity: this.imageOpacity.value || undefined,\r\n      };\r\n      if (\r\n        this.imageX.value ||\r\n        this.imageY.value ||\r\n        this.imageWidth.value ||\r\n        this.imageHeight.value\r\n      ) {\r\n        snapshot.image.canvas = {\r\n          x: this.imageX.value,\r\n          y: this.imageY.value,\r\n          width: this.imageWidth.value,\r\n          height: this.imageHeight.value,\r\n        };\r\n      }\r\n    }\r\n\r\n    return snapshot;\r\n  }\r\n\r\n  public save(): void {\r\n    if (this.form.invalid) {\r\n      return;\r\n    }\r\n    this._snapshot = this.getSnapshot();\r\n    this.snapshotChange.emit(this._snapshot);\r\n  }\r\n}\r\n","<form [formGroup]=\"form\" (submit)=\"save()\">\r\n  <mat-tab-group>\r\n    <!-- text -->\r\n    <mat-tab label=\"text\">\r\n      <mat-expansion-panel [expanded]=\"!baseText.value\">\r\n        <mat-expansion-panel-header>\r\n          <mat-panel-title> base text </mat-panel-title>\r\n        </mat-expansion-panel-header>\r\n        <gve-base-text-editor\r\n          [text]=\"baseText.value\"\r\n          (textChange)=\"onTextChange($event)\"\r\n        />\r\n        <fieldset>\r\n          <div class=\"form-row\">\r\n            <!-- offsetX -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>X offset</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"offsetX\"\r\n                placeholder=\"X offset\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- offsetY -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>Y offset</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"offsetY\"\r\n                placeholder=\"Y offset\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- lineHeightOffset -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>ln h-offset</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"lineHeightOffset\"\r\n                placeholder=\"ln h-offset\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- charSpacingOffset -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>char spacing</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"charSpacingOffset\"\r\n                placeholder=\"char spacing\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- spcWidthOffset -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>spc w-offset</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"spcWidthOffset\"\r\n                placeholder=\"spc w-offset\"\r\n              />\r\n            </mat-form-field>\r\n            <!-- minLineHeights -->\r\n            <div>\r\n              <gve-ln-heights-editor\r\n                [lineCount]=\"lineCount\"\r\n                [heights]=\"lnHeights.value\"\r\n                (heightsChange)=\"onHeightsChange($event)\"\r\n              />\r\n            </div>\r\n          </div>\r\n          <!-- textStyle -->\r\n          <div>\r\n            <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n              <mat-label>text style</mat-label>\r\n              <textarea\r\n                matInput\r\n                [formControl]=\"textStyle\"\r\n                placeholder=\"text style\"\r\n              ></textarea>\r\n            </mat-form-field>\r\n          </div>\r\n        </fieldset>\r\n      </mat-expansion-panel>\r\n    </mat-tab>\r\n\r\n    <!-- operations -->\r\n    <mat-tab label=\"operations\">\r\n      <div id=\"snapshot-container\">\r\n        <div id=\"general\">\r\n          <div class=\"form-row\">\r\n            <!-- width -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>width</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                min=\"0\"\r\n                [formControl]=\"width\"\r\n                placeholder=\"width\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- height -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>height</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                min=\"0\"\r\n                [formControl]=\"height\"\r\n                placeholder=\"height\"\r\n              />\r\n            </mat-form-field>\r\n\r\n            <!-- add op -->\r\n            <div>\r\n              <button\r\n                type=\"button\"\r\n                mat-flat-button\r\n                color=\"primary\"\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 [ngClass]=\"{ selected: operation.id === resultOperationId }\">\r\n                <td class=\"fit-width\">\r\n                  <!-- edit -->\r\n                  <button\r\n                    type=\"button\"\r\n                    mat-icon-button\r\n                    color=\"primary\"\r\n                    class=\"mat-primary\"\r\n                    (click)=\"editOperation(index)\"\r\n                    matTooltip=\"Edit operation\"\r\n                  >\r\n                    <mat-icon>edit</mat-icon>\r\n                  </button>\r\n                  <!-- delete -->\r\n                  <button\r\n                    type=\"button\"\r\n                    mat-icon-button\r\n                    color=\"warn\"\r\n                    (click)=\"deleteOperation(index)\"\r\n                    matTooltip=\"Delete operation\"\r\n                  >\r\n                    <mat-icon class=\"mat-warn\">delete</mat-icon>\r\n                  </button>\r\n                  <!-- run to -->\r\n                  <button\r\n                    type=\"button\"\r\n                    mat-icon-button\r\n                    color=\"accent\"\r\n                    (click)=\"runTo(index)\"\r\n                    matTooltip=\"Run to this operation\"\r\n                  >\r\n                    <mat-icon class=\"mat-accent\">subscriptions</mat-icon>\r\n                  </button>\r\n                </td>\r\n                <td>{{ operation.id }}</td>\r\n                <td>{{ operation.type | flatLookup : opTypeMap }}</td>\r\n                <td>{{ operation.at }}</td>\r\n                <td>{{ operation.run }}</td>\r\n                <td>{{ operation.value }}</td>\r\n                <td>{{ operation.inputTag }}</td>\r\n                <td>{{ operation.outputTag }}</td>\r\n                <td>{{ operation.groupId }}</td>\r\n                <td>{{ operation.features?.length }}</td>\r\n              </tr>\r\n              }\r\n            </tbody>\r\n          </table>\r\n          }\r\n\r\n          <!-- operation editor -->\r\n          <mat-expansion-panel [expanded]=\"editedOp\" [disabled]=\"!editedOp\">\r\n            <mat-expansion-panel-header>\r\n              <mat-panel-title>operation {{ editedOp?.id }}</mat-panel-title>\r\n            </mat-expansion-panel-header>\r\n            <fieldset>\r\n              <gve-chain-operation-editor\r\n                [operation]=\"editedOp\"\r\n                (operationCancel)=\"onOperationCancel()\"\r\n                (operationChange)=\"onOperationChange($event)\"\r\n                (operationPreview)=\"onOperationPreview($event)\"\r\n              />\r\n            </fieldset>\r\n          </mat-expansion-panel>\r\n\r\n          <div>\r\n            <mat-progress-bar mode=\"indeterminate\" *ngIf=\"busy\" />\r\n          </div>\r\n\r\n          <!-- opStyle -->\r\n          <div id=\"opStyle\">\r\n            <mat-form-field class=\"long-text\" appearance=\"outline\">\r\n              <mat-label>operations style</mat-label>\r\n              <textarea\r\n                matInput\r\n                [formControl]=\"opStyle\"\r\n                placeholder=\"operations style\"\r\n              ></textarea>\r\n            </mat-form-field>\r\n          </div>\r\n\r\n          <!-- batch ops -->\r\n          <div>\r\n            <mat-expansion-panel>\r\n              <mat-expansion-panel-header>\r\n                <mat-panel-title>batch operations</mat-panel-title>\r\n              </mat-expansion-panel-header>\r\n              <fieldset>\r\n                <div id=\"batch-container\">\r\n                  <div id=\"batch-input\">\r\n                    <div>\r\n                      <mat-form-field class=\"full-width\">\r\n                        <mat-label>operations</mat-label>\r\n                        <textarea\r\n                          class=\"code\"\r\n                          matInput\r\n                          [formControl]=\"inputOps\"\r\n                          placeholder=\"operations\"\r\n                          rows=\"8\"\r\n                          spellcheck=\"false\"\r\n                        ></textarea>\r\n                      </mat-form-field>\r\n                    </div>\r\n                    <div class=\"form-row\">\r\n                      <button\r\n                        type=\"button\"\r\n                        color=\"warn\"\r\n                        class=\"mat-warn\"\r\n                        mat-flat-button\r\n                        matTooltip=\"Remove all the operations\"\r\n                        [disabled]=\"!operations.value.length || busy\"\r\n                        (click)=\"clearOperations()\"\r\n                      >\r\n                        clear\r\n                      </button>\r\n                      <button\r\n                        type=\"button\"\r\n                        color=\"primary\"\r\n                        class=\"mat-primary\"\r\n                        mat-flat-button\r\n                        matTooltip=\"Parse text and add results to the operations\"\r\n                        [disabled]=\"!inputOps.value || busy\"\r\n                        (click)=\"parseOperations()\"\r\n                      >\r\n                        batch add\r\n                      </button>\r\n                      @if (parseError) {\r\n                      <span class=\"error\">{{ parseError }}</span>\r\n                      }\r\n                    </div>\r\n                  </div>\r\n                  <div id=\"batch-help\">\r\n                    <ul>\r\n                      <li>\r\n                        <strong>replace</strong>:\r\n                        <code>ATxRUN<strong>=</strong>\"VALUE\"</code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>delete</strong>:\r\n                        <code>ATxRUN<strong>-</strong></code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>add-before</strong>:\r\n                        <code>ATxRUN<strong>+[</strong>\"VALUE\"</code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>add-after</strong>:\r\n                        <code>ATxRUN<strong>+]</strong>\"VALUE\"</code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>move-before</strong>:\r\n                        <code>ATxRUN<strong>&gt;[</strong>TO</code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>move-after</strong>:\r\n                        <code>ATxRUN<strong>&gt;]</strong>TO</code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>swap</strong>:\r\n                        <code>ATxRUN<strong>&lt;&gt;</strong>TOxRUN</code>\r\n                      </li>\r\n                      <li>\r\n                        <strong>annotate</strong>:\r\n                        <code>ATxRUN<strong>:</strong></code>\r\n                      </li>\r\n                    </ul>\r\n                    <p>For all the operations:</p>\r\n                    <ul>\r\n                      <li>\r\n                        prefix <code>(ITAG:OTAG)</code> to define input and/or\r\n                        output tags, separated by colon.\r\n                      </li>\r\n                      <li>\r\n                        prepend <code>&#x40;</code> to AT to use character\r\n                        indexes (0-N) rather than IDs.\r\n                      </li>\r\n                      <li>RUN (where applicable) defaults to 1.</li>\r\n                      <li>\r\n                        append features like <code>[NAME OPERATOR VALUE]</code>,\r\n                        separated by space, where:\r\n                      </li>\r\n                      <ol>\r\n                        <li>\r\n                          NAME is an arbitrary string. Prefixes:\r\n                          <code>*</code>=global features, <code>!</code>=remove\r\n                          feature (no value). Suffixes:\r\n                          <code>^</code>=short-lived (like\r\n                          <code>*version^=alpha</code>).\r\n                        </li>\r\n                        <li>\r\n                          OPERATOR is <code>=</code> (multiple),\r\n                          <code>:=</code> (single),\r\n                          <code>==</code> (first-single).\r\n                        </li>\r\n                        <li>\r\n                          VALUE is a string. If it includes spaces, wrap it in\r\n                          <code>\"\"</code>.\r\n                        </li>\r\n                      </ol>\r\n                    </ul>\r\n                  </div>\r\n                </div>\r\n              </fieldset>\r\n            </mat-expansion-panel>\r\n          </div>\r\n        </div>\r\n\r\n        <!-- result -->\r\n        @if (result) {\r\n        <fieldset id=\"result\">\r\n          <legend>result</legend>\r\n          <gve-chain-result-view\r\n            [result]=\"result\"\r\n            (stepPick)=\"onStepPick($event)\"\r\n          />\r\n        </fieldset>\r\n        }\r\n      </div>\r\n    </mat-tab>\r\n\r\n    <!-- image -->\r\n    <mat-tab label=\"image\">\r\n      <div id=\"image\">\r\n        <div id=\"image-ctl\">\r\n          <!-- url -->\r\n          <mat-form-field class=\"long-text\">\r\n            <mat-label>URL</mat-label>\r\n            <input matInput [formControl]=\"imageUrl\" />\r\n          </mat-form-field>\r\n          <div class=\"form-row\">\r\n            <!-- x -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>X</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"imageX\"\r\n                placeholder=\"X\"\r\n              />\r\n            </mat-form-field>\r\n            <!-- y -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>Y</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                [formControl]=\"imageY\"\r\n                placeholder=\"Y\"\r\n              />\r\n            </mat-form-field>\r\n          </div>\r\n          <div class=\"form-row\">\r\n            <!-- width -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>width</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                min=\"0\"\r\n                [formControl]=\"imageWidth\"\r\n              />\r\n            </mat-form-field>\r\n            <!-- height -->\r\n            <mat-form-field class=\"input-nr\">\r\n              <mat-label>height</mat-label>\r\n              <input\r\n                matInput\r\n                type=\"number\"\r\n                min=\"0\"\r\n                [formControl]=\"imageHeight\"\r\n              />\r\n            </mat-form-field>\r\n          </div>\r\n          <!-- defs -->\r\n          <div>\r\n            <mat-form-field class=\"long-text\">\r\n              <mat-label>defs</mat-label>\r\n              <textarea matInput [formControl]=\"defs\" rows=\"3\"></textarea>\r\n            </mat-form-field>\r\n          </div>\r\n        </div>\r\n        <div id=\"image-view\">\r\n          @if (imageUrl.value) {\r\n          <img alt=\"background\" [src]=\"imageUrl.value\" width=\"600\" />\r\n          }\r\n        </div>\r\n      </div>\r\n    </mat-tab>\r\n  </mat-tab-group>\r\n\r\n  <!-- preview -->\r\n  @if (snapshot) {\r\n  <fieldset id=\"preview\">\r\n    <legend class=\"button-row\">\r\n      <span>{{ viewTitle }}</span>\r\n      <button\r\n        color=\"primary\"\r\n        mat-icon-button\r\n        type=\"button\"\r\n        matTooltip=\"Refresh preview\"\r\n        (click)=\"updateViewData()\"\r\n      >\r\n        <mat-icon class=\"mat-primary\">refresh</mat-icon>\r\n      </button>\r\n    </legend>\r\n    <gve-snapshot-view\r\n      [data]=\"viewData\"\r\n      (snapshotRender)=\"onSnapshotRender($any($event))\"\r\n      (visualEvent)=\"onVisualEvent($any($event))\"\r\n    />\r\n    <div>\r\n      <mat-button-toggle\r\n        type=\"button\"\r\n        mat-icon-button\r\n        color=\"primary\"\r\n        matTooltip=\"Toggle rulers\"\r\n        [checked]=\"rulers\"\r\n        (change)=\"toggleRulers()\"\r\n      >\r\n        <mat-icon class=\"mat-primary\">straighten</mat-icon>\r\n      </mat-button-toggle>\r\n    </div>\r\n    @if (visualInfo) {\r\n    <div id=\"visual-info\">{{ visualInfo }}</div>\r\n    }\r\n  </fieldset>\r\n  }\r\n\r\n  <!--buttons -->\r\n  <div class=\"form-row-center\">\r\n    <button\r\n      type=\"button\"\r\n      class=\"mat-warn\"\r\n      mat-flat-button\r\n      matTooltip=\"Discard changes\"\r\n      (click)=\"close()\"\r\n    >\r\n      <mat-icon>clear</mat-icon>\r\n      close\r\n    </button>\r\n    <button\r\n      type=\"submit\"\r\n      class=\"mat-primary\"\r\n      mat-flat-button\r\n      matTooltip=\"Save changes\"\r\n      [disabled]=\"form.invalid || form.pristine\"\r\n    >\r\n      <mat-icon>check_circle</mat-icon>\r\n      save\r\n    </button>\r\n  </div>\r\n</form>\r\n"]}
|