@myrmidon/gve-core 0.0.6 → 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (27) hide show
  1. package/fesm2022/myrmidon-gve-core.mjs +134 -136
  2. package/fesm2022/myrmidon-gve-core.mjs.map +1 -1
  3. package/lib/components/feature-editor/feature-editor.component.d.ts +1 -1
  4. package/package.json +6 -8
  5. package/esm2022/lib/components/animation-timeline/animation-timeline.component.mjs +0 -207
  6. package/esm2022/lib/components/animation-timeline-set/animation-timeline-set.component.mjs +0 -154
  7. package/esm2022/lib/components/animation-tween/animation-tween.component.mjs +0 -184
  8. package/esm2022/lib/components/base-text-char/base-text-char.component.mjs +0 -46
  9. package/esm2022/lib/components/base-text-editor/base-text-editor.component.mjs +0 -115
  10. package/esm2022/lib/components/base-text-view/base-text-view.component.mjs +0 -159
  11. package/esm2022/lib/components/batch-operation-editor/batch-operation-editor.component.mjs +0 -111
  12. package/esm2022/lib/components/chain-operation-editor/chain-operation-editor.component.mjs +0 -570
  13. package/esm2022/lib/components/chain-result-view/chain-result-view.component.mjs +0 -225
  14. package/esm2022/lib/components/feature-editor/feature-editor.component.mjs +0 -200
  15. package/esm2022/lib/components/feature-set-editor/feature-set-editor.component.mjs +0 -175
  16. package/esm2022/lib/components/feature-set-view/feature-set-view.component.mjs +0 -100
  17. package/esm2022/lib/components/ln-heights-editor/ln-heights-editor.component.mjs +0 -126
  18. package/esm2022/lib/components/operation-source-editor/operation-source-editor.component.mjs +0 -131
  19. package/esm2022/lib/components/simple-tree/simple-tree.component.mjs +0 -72
  20. package/esm2022/lib/components/snapshot-editor/snapshot-editor.component.mjs +0 -870
  21. package/esm2022/lib/components/steps-map/steps-map.component.mjs +0 -83
  22. package/esm2022/lib/models.mjs +0 -2
  23. package/esm2022/lib/services/gve-api.service.mjs +0 -65
  24. package/esm2022/lib/services/settings.service.mjs +0 -87
  25. package/esm2022/lib/validators/svg-validators.mjs +0 -34
  26. package/esm2022/myrmidon-gve-core.mjs +0 -5
  27. package/esm2022/public-api.mjs +0 -23
@@ -1,159 +0,0 @@
1
- import { CommonModule } from '@angular/common';
2
- import { Component, EventEmitter, Input, Output } from '@angular/core';
3
- import { BaseTextCharComponent, } from '../base-text-char/base-text-char.component';
4
- import { SnapshotViewService } from '@myrmidon/gve-snapshot-view';
5
- import * as i0 from "@angular/core";
6
- /**
7
- * 🔑 `gve-base-text-view`
8
- *
9
- * A component to display a selectable base text.
10
- * Used by the chain result view component and the base text editor component.
11
- *
12
- * - ▶️ `defaultColor` (`string`): the default color for the text.
13
- * - ▶️ `defaultBorderColor` (`string`): the default border color for the text.
14
- * - ▶️ `selectionColor` (`string`): the color for the selected text.
15
- * - ▶️ `hasLineNumber` (`boolean`): true if line numbers should be displayed next to each line.
16
- * - ▶️ `text` (`string` | `CharNode[]`): the text to display.
17
- * - 🔥 `charPick` (`BaseTextCharEvent`): emitted when a character is picked.
18
- * - 🔥 `rangePick` (`VarBaseTextRange`): emitted when a range is picked.
19
- */
20
- export class BaseTextViewComponent {
21
- constructor() {
22
- /**
23
- * The default color for the text.
24
- */
25
- this.defaultColor = '#DBDBDB';
26
- /**
27
- * The default border color for the text.
28
- */
29
- this.defaultBorderColor = '#DBDBDB';
30
- /**
31
- * The color for the selected text.
32
- */
33
- this.selectionColor = '#3E92CC';
34
- /**
35
- * True if line numbers should be displayed next to each line.
36
- */
37
- this.hasLineNumber = false;
38
- /**
39
- * Emitted when a character is picked.
40
- */
41
- this.charPick = new EventEmitter();
42
- /**
43
- * Emitted when a range is picked. This is preceded by a character pick event.
44
- * The range is defined by the starting character and the number of characters.
45
- * The range is inclusive.
46
- */
47
- this.rangePick = new EventEmitter();
48
- this.lines = [];
49
- }
50
- /**
51
- * The text to display.
52
- */
53
- get text() {
54
- return this._text;
55
- }
56
- set text(value) {
57
- this._text = value || undefined;
58
- this.buildLines();
59
- }
60
- buildLines() {
61
- if (!this._text) {
62
- this.lines = [];
63
- return;
64
- }
65
- const newLines = [];
66
- newLines.push([]);
67
- if (Array.isArray(this._text)) {
68
- const nodes = this._text;
69
- for (let i = 0; i < nodes.length; i++) {
70
- const char = {
71
- id: nodes[i].id,
72
- value: nodes[i].data,
73
- label: nodes[i].label,
74
- color: this.defaultColor,
75
- borderColor: this.defaultBorderColor,
76
- emSize: 1.5,
77
- };
78
- newLines[newLines.length - 1].push(char);
79
- if (char.value === '\n') {
80
- newLines.push([]);
81
- }
82
- }
83
- }
84
- else {
85
- for (let i = 0; i < this._text.length; i++) {
86
- const char = {
87
- id: i + 1,
88
- value: this._text[i],
89
- label: SnapshotViewService.translateSpecialChar(this._text[i]),
90
- color: this.defaultColor,
91
- borderColor: this.defaultBorderColor,
92
- emSize: 1.5,
93
- };
94
- newLines[newLines.length - 1].push(char);
95
- if (char.value === '\n') {
96
- newLines.push([]);
97
- }
98
- }
99
- }
100
- this.lines = newLines;
101
- }
102
- resetColors() {
103
- for (const line of this.lines) {
104
- for (const c of line) {
105
- c.color = this.defaultColor;
106
- c.borderColor = this.defaultBorderColor;
107
- }
108
- }
109
- }
110
- onCharPick(event) {
111
- this.resetColors();
112
- if (this._lastSelectedChar &&
113
- this._lastSelectedChar !== event.char &&
114
- event.event.ctrlKey) {
115
- const minId = Math.min(this._lastSelectedChar.id, event.char.id);
116
- const maxId = Math.max(this._lastSelectedChar.id, event.char.id);
117
- for (const line of this.lines) {
118
- for (const c of line) {
119
- if (c.id >= minId && c.id <= maxId) {
120
- c.color = this.selectionColor;
121
- c.borderColor = this.selectionColor;
122
- }
123
- }
124
- }
125
- this.charPick.emit(event);
126
- this.rangePick.emit({ at: minId, run: maxId - minId + 1 });
127
- return;
128
- }
129
- else {
130
- // select current char
131
- event.char.color = this.selectionColor;
132
- event.char.borderColor = this.selectionColor;
133
- this.charPick.emit(event);
134
- this.rangePick.emit({ at: event.char.id, run: 1 });
135
- }
136
- this._lastSelectedChar = event.char;
137
- }
138
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: BaseTextViewComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
139
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.12", type: BaseTextViewComponent, isStandalone: true, selector: "gve-base-text-view", inputs: { defaultColor: "defaultColor", defaultBorderColor: "defaultBorderColor", selectionColor: "selectionColor", hasLineNumber: "hasLineNumber", text: "text" }, outputs: { charPick: "charPick", rangePick: "rangePick" }, ngImport: i0, template: "<div id=\"text\">\r\n @for (line of lines; track $index) {\r\n <div class=\"line\">\r\n @if (hasLineNumber) {\r\n <div class=\"nr\">\r\n {{ lines.indexOf(line) + 1 }}\r\n </div>\r\n } @for (c of line; track c.id) {\r\n <gve-base-text-char [char]=\"c\" (charPick)=\"onCharPick($event)\" />\r\n }\r\n </div>\r\n }\r\n</div>\r\n", styles: [".line{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.line *{flex:0 0 auto}.nr{font-size:.8em;font-weight:700;color:silver;margin:0 4px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: BaseTextCharComponent, selector: "gve-base-text-char", inputs: ["char"], outputs: ["charPick"] }] }); }
140
- }
141
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: BaseTextViewComponent, decorators: [{
142
- type: Component,
143
- args: [{ selector: 'gve-base-text-view', standalone: true, imports: [CommonModule, BaseTextCharComponent], template: "<div id=\"text\">\r\n @for (line of lines; track $index) {\r\n <div class=\"line\">\r\n @if (hasLineNumber) {\r\n <div class=\"nr\">\r\n {{ lines.indexOf(line) + 1 }}\r\n </div>\r\n } @for (c of line; track c.id) {\r\n <gve-base-text-char [char]=\"c\" (charPick)=\"onCharPick($event)\" />\r\n }\r\n </div>\r\n }\r\n</div>\r\n", styles: [".line{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.line *{flex:0 0 auto}.nr{font-size:.8em;font-weight:700;color:silver;margin:0 4px}\n"] }]
144
- }], propDecorators: { defaultColor: [{
145
- type: Input
146
- }], defaultBorderColor: [{
147
- type: Input
148
- }], selectionColor: [{
149
- type: Input
150
- }], hasLineNumber: [{
151
- type: Input
152
- }], text: [{
153
- type: Input
154
- }], charPick: [{
155
- type: Output
156
- }], rangePick: [{
157
- type: Output
158
- }] } });
159
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-text-view.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/gve-core/src/lib/components/base-text-view/base-text-view.component.ts","../../../../../../../projects/myrmidon/gve-core/src/lib/components/base-text-view/base-text-view.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { Component, EventEmitter, Input, Output } from '@angular/core';\r\n\r\nimport {\r\n  BaseTextChar,\r\n  BaseTextCharComponent,\r\n  BaseTextCharEvent,\r\n} from '../base-text-char/base-text-char.component';\r\nimport { CharNode, SnapshotViewService } from '@myrmidon/gve-snapshot-view';\r\n\r\n/**\r\n * A range in the base text.\r\n */\r\nexport interface VarBaseTextRange {\r\n  at: number;\r\n  run: number;\r\n}\r\n\r\n/**\r\n * 🔑 `gve-base-text-view`\r\n *\r\n * A component to display a selectable base text.\r\n * Used by the chain result view component and the base text editor component.\r\n *\r\n * - ▶️ `defaultColor` (`string`): the default color for the text.\r\n * - ▶️ `defaultBorderColor` (`string`): the default border color for the text.\r\n * - ▶️ `selectionColor` (`string`): the color for the selected text.\r\n * - ▶️ `hasLineNumber` (`boolean`): true if line numbers should be displayed next to each line.\r\n * - ▶️ `text` (`string` | `CharNode[]`): the text to display.\r\n * - 🔥 `charPick` (`BaseTextCharEvent`): emitted when a character is picked.\r\n * - 🔥 `rangePick` (`VarBaseTextRange`): emitted when a range is picked.\r\n */\r\n@Component({\r\n  selector: 'gve-base-text-view',\r\n  standalone: true,\r\n  imports: [CommonModule, BaseTextCharComponent],\r\n  templateUrl: './base-text-view.component.html',\r\n  styleUrl: './base-text-view.component.css',\r\n})\r\nexport class BaseTextViewComponent {\r\n  private _text?: string | CharNode[];\r\n  private _lastSelectedChar?: BaseTextChar;\r\n\r\n  /**\r\n   * The default color for the text.\r\n   */\r\n  @Input()\r\n  public defaultColor = '#DBDBDB';\r\n\r\n  /**\r\n   * The default border color for the text.\r\n   */\r\n  @Input()\r\n  public defaultBorderColor = '#DBDBDB';\r\n\r\n  /**\r\n   * The color for the selected text.\r\n   */\r\n  @Input()\r\n  public selectionColor = '#3E92CC';\r\n\r\n  /**\r\n   * True if line numbers should be displayed next to each line.\r\n   */\r\n  @Input()\r\n  public hasLineNumber = false;\r\n\r\n  /**\r\n   * The text to display.\r\n   */\r\n  @Input()\r\n  public get text(): string | CharNode[] | undefined {\r\n    return this._text;\r\n  }\r\n  public set text(value: string | CharNode[] | undefined | null) {\r\n    this._text = value || undefined;\r\n    this.buildLines();\r\n  }\r\n\r\n  /**\r\n   * Emitted when a character is picked.\r\n   */\r\n  @Output()\r\n  public charPick: EventEmitter<BaseTextCharEvent> =\r\n    new EventEmitter<BaseTextCharEvent>();\r\n\r\n  /**\r\n   * Emitted when a range is picked. This is preceded by a character pick event.\r\n   * The range is defined by the starting character and the number of characters.\r\n   * The range is inclusive.\r\n   */\r\n  @Output()\r\n  public rangePick: EventEmitter<VarBaseTextRange> =\r\n    new EventEmitter<VarBaseTextRange>();\r\n\r\n  public lines: BaseTextChar[][] = [];\r\n\r\n  private buildLines(): void {\r\n    if (!this._text) {\r\n      this.lines = [];\r\n      return;\r\n    }\r\n\r\n    const newLines: BaseTextChar[][] = [];\r\n    newLines.push([]);\r\n\r\n    if (Array.isArray(this._text)) {\r\n      const nodes = this._text as CharNode[];\r\n      for (let i = 0; i < nodes.length; i++) {\r\n        const char: BaseTextChar = {\r\n          id: nodes[i].id,\r\n          value: nodes[i].data,\r\n          label: nodes[i].label,\r\n          color: this.defaultColor,\r\n          borderColor: this.defaultBorderColor,\r\n          emSize: 1.5,\r\n        };\r\n        newLines[newLines.length - 1].push(char);\r\n\r\n        if (char.value === '\\n') {\r\n          newLines.push([]);\r\n        }\r\n      }\r\n    } else {\r\n      for (let i = 0; i < this._text.length; i++) {\r\n        const char: BaseTextChar = {\r\n          id: i + 1,\r\n          value: this._text[i],\r\n          label: SnapshotViewService.translateSpecialChar(this._text[i]),\r\n          color: this.defaultColor,\r\n          borderColor: this.defaultBorderColor,\r\n          emSize: 1.5,\r\n        };\r\n        newLines[newLines.length - 1].push(char);\r\n\r\n        if (char.value === '\\n') {\r\n          newLines.push([]);\r\n        }\r\n      }\r\n    }\r\n\r\n    this.lines = newLines;\r\n  }\r\n\r\n  private resetColors(): void {\r\n    for (const line of this.lines) {\r\n      for (const c of line) {\r\n        c.color = this.defaultColor;\r\n        c.borderColor = this.defaultBorderColor;\r\n      }\r\n    }\r\n  }\r\n\r\n  public onCharPick(event: BaseTextCharEvent): void {\r\n    this.resetColors();\r\n\r\n    if (\r\n      this._lastSelectedChar &&\r\n      this._lastSelectedChar !== event.char &&\r\n      event.event.ctrlKey\r\n    ) {\r\n      const minId = Math.min(this._lastSelectedChar.id, event.char.id);\r\n      const maxId = Math.max(this._lastSelectedChar.id, event.char.id);\r\n\r\n      for (const line of this.lines) {\r\n        for (const c of line) {\r\n          if (c.id >= minId && c.id <= maxId) {\r\n            c.color = this.selectionColor;\r\n            c.borderColor = this.selectionColor;\r\n          }\r\n        }\r\n      }\r\n\r\n      this.charPick.emit(event);\r\n      this.rangePick.emit({ at: minId, run: maxId - minId + 1 });\r\n      return;\r\n    } else {\r\n      // select current char\r\n      event.char.color = this.selectionColor;\r\n      event.char.borderColor = this.selectionColor;\r\n      this.charPick.emit(event);\r\n      this.rangePick.emit({ at: event.char.id, run: 1 });\r\n    }\r\n\r\n    this._lastSelectedChar = event.char;\r\n  }\r\n}\r\n","<div id=\"text\">\r\n  @for (line of lines; track $index) {\r\n  <div class=\"line\">\r\n    @if (hasLineNumber) {\r\n    <div class=\"nr\">\r\n      {{ lines.indexOf(line) + 1 }}\r\n    </div>\r\n    } @for (c of line; track c.id) {\r\n    <gve-base-text-char [char]=\"c\" (charPick)=\"onCharPick($event)\" />\r\n    }\r\n  </div>\r\n  }\r\n</div>\r\n"]}
@@ -1,111 +0,0 @@
1
- import { Component, EventEmitter, Inject, Input, Optional, Output, } from '@angular/core';
2
- import { ReactiveFormsModule, Validators, } from '@angular/forms';
3
- import { MatButtonModule } from '@angular/material/button';
4
- import { MAT_DIALOG_DATA } from '@angular/material/dialog';
5
- import { MatFormFieldModule } from '@angular/material/form-field';
6
- import { MatIconModule } from '@angular/material/icon';
7
- import { MatInputModule } from '@angular/material/input';
8
- import * as i0 from "@angular/core";
9
- import * as i1 from "@angular/forms";
10
- import * as i2 from "../../services/gve-api.service";
11
- import * as i3 from "@angular/material/dialog";
12
- import * as i4 from "@angular/material/button";
13
- import * as i5 from "@angular/material/form-field";
14
- import * as i6 from "@angular/material/input";
15
- /**
16
- * 🔑 `gve-batch-operation-editor`
17
- *
18
- * A component to edit a batch of operations. This component can be used both in a dialog
19
- * and as a standalone component.
20
- *
21
- * - ▶️ `preset` (`string`): the optional preset text to parse.
22
- * - 🔥 `operationsChange` (`CharChainOperation[]`): event emitted when operations change.
23
- */
24
- export class BatchOperationEditorComponent {
25
- /**
26
- * The preset text to parse if any. Usually you start with a blank
27
- * text, but sometimes you might want to pre-set it.
28
- */
29
- get preset() {
30
- return this._preset;
31
- }
32
- set preset(value) {
33
- if (this._preset === value) {
34
- return;
35
- }
36
- this._preset = value || undefined;
37
- this.parseOperations();
38
- }
39
- constructor(formBuilder, _api,
40
- // this component can be used as a dialog
41
- dialogRef, data) {
42
- this._api = _api;
43
- this.dialogRef = dialogRef;
44
- this.data = data;
45
- /**
46
- * Emitted when operations change.
47
- */
48
- this.operationsChange = new EventEmitter();
49
- this.text = formBuilder.control(null, [
50
- Validators.required,
51
- Validators.maxLength(2000),
52
- ]);
53
- this.form = formBuilder.group({
54
- inputOps: this.text,
55
- });
56
- }
57
- ngOnInit() {
58
- if (this.data?.payload?.preset) {
59
- this.preset = this.data.payload.preset;
60
- this.parseOperations();
61
- }
62
- }
63
- parseOperations() {
64
- if (this.busy || !this.text.value) {
65
- return;
66
- }
67
- this.busy = true;
68
- this.parseError = undefined;
69
- this._api.parseOperations(this.text.value).subscribe({
70
- next: (wrapper) => {
71
- this.operationsChange.emit(wrapper.result);
72
- if (!this.preset) {
73
- this.dialogRef?.close(wrapper.result);
74
- }
75
- },
76
- error: (error) => {
77
- this.parseError = error.message;
78
- },
79
- complete: () => {
80
- this.busy = false;
81
- },
82
- });
83
- }
84
- close() {
85
- this.dialogRef?.close();
86
- }
87
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: BatchOperationEditorComponent, deps: [{ token: i1.FormBuilder }, { token: i2.GveApiService }, { token: i3.MatDialogRef, optional: true }, { token: MAT_DIALOG_DATA, optional: true }], target: i0.ɵɵFactoryTarget.Component }); }
88
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.12", type: BatchOperationEditorComponent, isStandalone: true, selector: "gve-batch-operation-editor", inputs: { preset: "preset" }, outputs: { operationsChange: "operationsChange" }, ngImport: i0, template: "<div [style.padding]=\"dialogRef ? '8px' : '0'\">\r\n @if (dialogRef) {\r\n <div id=\"heading\">\r\n <h2>Add Operations</h2>\r\n </div>\r\n }\r\n <fieldset>\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]=\"text\"\r\n rows=\"8\"\r\n spellcheck=\"false\"\r\n ></textarea>\r\n </mat-form-field>\r\n @if (parseError) {\r\n <div class=\"error\">{{ parseError }}</div>\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 output tags,\r\n separated by colon.\r\n </li>\r\n <li>\r\n prepend <code>&#x40;</code> to AT to use character indexes (0-N)\r\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>, separated by\r\n space, where:\r\n </li>\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 feature (no\r\n value). Suffixes: <code>^</code>=short-lived (like\r\n <code>*version^=alpha</code>).\r\n </li>\r\n <li>\r\n OPERATOR is <code>=</code> (multiple), <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 </li>\r\n </ul>\r\n </div>\r\n <div class=\"form-row-center\">\r\n @if (dialogRef) {\r\n <button\r\n type=\"button\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Close dialog\"\r\n (click)=\"close()\"\r\n >\r\n close\r\n </button>\r\n }\r\n <button\r\n type=\"button\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Parse text into operations\"\r\n [disabled]=\"!text.value || busy\"\r\n (click)=\"parseOperations()\"\r\n >\r\n batch add\r\n </button>\r\n </div>\r\n </fieldset>\r\n</div>\r\n", styles: [".full-width{width:100%}.error{color:red}.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}div#heading{margin:8px;text-align:center}div#batch-help strong{color:#ff8c00}\n"], dependencies: [{ kind: "ngmodule", type: ReactiveFormsModule }, { 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.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i4.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i5.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i5.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatIconModule }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i6.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"] }] }); }
89
- }
90
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: BatchOperationEditorComponent, decorators: [{
91
- type: Component,
92
- args: [{ selector: 'gve-batch-operation-editor', standalone: true, imports: [
93
- ReactiveFormsModule,
94
- MatButtonModule,
95
- MatFormFieldModule,
96
- MatIconModule,
97
- MatInputModule,
98
- ], template: "<div [style.padding]=\"dialogRef ? '8px' : '0'\">\r\n @if (dialogRef) {\r\n <div id=\"heading\">\r\n <h2>Add Operations</h2>\r\n </div>\r\n }\r\n <fieldset>\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]=\"text\"\r\n rows=\"8\"\r\n spellcheck=\"false\"\r\n ></textarea>\r\n </mat-form-field>\r\n @if (parseError) {\r\n <div class=\"error\">{{ parseError }}</div>\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 output tags,\r\n separated by colon.\r\n </li>\r\n <li>\r\n prepend <code>&#x40;</code> to AT to use character indexes (0-N)\r\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>, separated by\r\n space, where:\r\n </li>\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 feature (no\r\n value). Suffixes: <code>^</code>=short-lived (like\r\n <code>*version^=alpha</code>).\r\n </li>\r\n <li>\r\n OPERATOR is <code>=</code> (multiple), <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 </li>\r\n </ul>\r\n </div>\r\n <div class=\"form-row-center\">\r\n @if (dialogRef) {\r\n <button\r\n type=\"button\"\r\n class=\"mat-warn\"\r\n mat-flat-button\r\n matTooltip=\"Close dialog\"\r\n (click)=\"close()\"\r\n >\r\n close\r\n </button>\r\n }\r\n <button\r\n type=\"button\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Parse text into operations\"\r\n [disabled]=\"!text.value || busy\"\r\n (click)=\"parseOperations()\"\r\n >\r\n batch add\r\n </button>\r\n </div>\r\n </fieldset>\r\n</div>\r\n", styles: [".full-width{width:100%}.error{color:red}.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}div#heading{margin:8px;text-align:center}div#batch-help strong{color:#ff8c00}\n"] }]
99
- }], ctorParameters: () => [{ type: i1.FormBuilder }, { type: i2.GveApiService }, { type: i3.MatDialogRef, decorators: [{
100
- type: Optional
101
- }] }, { type: undefined, decorators: [{
102
- type: Optional
103
- }, {
104
- type: Inject,
105
- args: [MAT_DIALOG_DATA]
106
- }] }], propDecorators: { preset: [{
107
- type: Input
108
- }], operationsChange: [{
109
- type: Output
110
- }] } });
111
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"batch-operation-editor.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/gve-core/src/lib/components/batch-operation-editor/batch-operation-editor.component.ts","../../../../../../../projects/myrmidon/gve-core/src/lib/components/batch-operation-editor/batch-operation-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  EventEmitter,\r\n  Inject,\r\n  Input,\r\n  OnInit,\r\n  Optional,\r\n  Output,\r\n} from '@angular/core';\r\nimport {\r\n  FormBuilder,\r\n  FormControl,\r\n  FormGroup,\r\n  ReactiveFormsModule,\r\n  Validators,\r\n} from '@angular/forms';\r\n\r\nimport { MatButtonModule } from '@angular/material/button';\r\nimport { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatIconModule } from '@angular/material/icon';\r\nimport { MatInputModule } from '@angular/material/input';\r\n\r\nimport { CharChainOperation } from '@myrmidon/gve-snapshot-view';\r\n\r\nimport { PayloadMatDialogConfig } from '../../models';\r\nimport { GveApiService } from '../../services/gve-api.service';\r\n\r\n/**\r\n * 🔑 `gve-batch-operation-editor`\r\n *\r\n * A component to edit a batch of operations. This component can be used both in a dialog\r\n * and as a standalone component.\r\n *\r\n * - ▶️ `preset` (`string`): the optional preset text to parse.\r\n * - 🔥 `operationsChange` (`CharChainOperation[]`): event emitted when operations change.\r\n */\r\n@Component({\r\n  selector: 'gve-batch-operation-editor',\r\n  standalone: true,\r\n  imports: [\r\n    ReactiveFormsModule,\r\n    MatButtonModule,\r\n    MatFormFieldModule,\r\n    MatIconModule,\r\n    MatInputModule,\r\n  ],\r\n  templateUrl: './batch-operation-editor.component.html',\r\n  styleUrl: './batch-operation-editor.component.css',\r\n})\r\nexport class BatchOperationEditorComponent implements OnInit {\r\n  private _preset?: string;\r\n\r\n  public busy?: boolean;\r\n  public parseError?: string;\r\n  public text: FormControl<string | null>;\r\n  public form: FormGroup;\r\n\r\n  /**\r\n   * The preset text to parse if any. Usually you start with a blank\r\n   * text, but sometimes you might want to pre-set it.\r\n   */\r\n  @Input()\r\n  public get preset(): string | undefined | null {\r\n    return this._preset;\r\n  }\r\n  public set preset(value: string | undefined | null) {\r\n    if (this._preset === value) {\r\n      return;\r\n    }\r\n    this._preset = value || undefined;\r\n    this.parseOperations();\r\n  }\r\n\r\n  /**\r\n   * Emitted when operations change.\r\n   */\r\n  @Output()\r\n  public readonly operationsChange: EventEmitter<CharChainOperation[]> =\r\n    new EventEmitter();\r\n\r\n  constructor(\r\n    formBuilder: FormBuilder,\r\n    private _api: GveApiService,\r\n    // this component can be used as a dialog\r\n    @Optional()\r\n    public dialogRef?: MatDialogRef<BatchOperationEditorComponent>,\r\n    @Optional()\r\n    @Inject(MAT_DIALOG_DATA)\r\n    public data?: PayloadMatDialogConfig<{ preset?: string }>\r\n  ) {\r\n    this.text = formBuilder.control<string | null>(null, [\r\n      Validators.required,\r\n      Validators.maxLength(2000),\r\n    ]);\r\n    this.form = formBuilder.group({\r\n      inputOps: this.text,\r\n    });\r\n  }\r\n\r\n  public ngOnInit() {\r\n    if (this.data?.payload?.preset) {\r\n      this.preset = this.data.payload.preset;\r\n      this.parseOperations();\r\n    }\r\n  }\r\n\r\n  public parseOperations() {\r\n    if (this.busy || !this.text.value) {\r\n      return;\r\n    }\r\n    this.busy = true;\r\n    this.parseError = undefined;\r\n\r\n    this._api.parseOperations(this.text.value!).subscribe({\r\n      next: (wrapper) => {\r\n        this.operationsChange.emit(wrapper.result);\r\n        if (!this.preset) {\r\n          this.dialogRef?.close(wrapper.result);\r\n        }\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  public close(): void {\r\n    this.dialogRef?.close();\r\n  }\r\n}\r\n","<div [style.padding]=\"dialogRef ? '8px' : '0'\">\r\n  @if (dialogRef) {\r\n  <div id=\"heading\">\r\n    <h2>Add Operations</h2>\r\n  </div>\r\n  }\r\n  <fieldset>\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]=\"text\"\r\n            rows=\"8\"\r\n            spellcheck=\"false\"\r\n          ></textarea>\r\n        </mat-form-field>\r\n        @if (parseError) {\r\n        <div class=\"error\">{{ parseError }}</div>\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 output tags,\r\n          separated by colon.\r\n        </li>\r\n        <li>\r\n          prepend <code>&#x40;</code> to AT to use character indexes (0-N)\r\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>, separated by\r\n          space, where:\r\n        </li>\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 feature (no\r\n              value). Suffixes: <code>^</code>=short-lived (like\r\n              <code>*version^=alpha</code>).\r\n            </li>\r\n            <li>\r\n              OPERATOR is <code>=</code> (multiple), <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        </li>\r\n      </ul>\r\n    </div>\r\n    <div class=\"form-row-center\">\r\n      @if (dialogRef) {\r\n      <button\r\n        type=\"button\"\r\n        class=\"mat-warn\"\r\n        mat-flat-button\r\n        matTooltip=\"Close dialog\"\r\n        (click)=\"close()\"\r\n      >\r\n        close\r\n      </button>\r\n      }\r\n      <button\r\n        type=\"button\"\r\n        class=\"mat-primary\"\r\n        mat-flat-button\r\n        matTooltip=\"Parse text into operations\"\r\n        [disabled]=\"!text.value || busy\"\r\n        (click)=\"parseOperations()\"\r\n      >\r\n        batch add\r\n      </button>\r\n    </div>\r\n  </fieldset>\r\n</div>\r\n"]}