@myrmidon/gve-core 0.0.5 → 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (35) hide show
  1. package/README.md +2 -0
  2. package/fesm2022/myrmidon-gve-core.mjs +370 -259
  3. package/fesm2022/myrmidon-gve-core.mjs.map +1 -1
  4. package/lib/components/batch-operation-editor/batch-operation-editor.component.d.ts +46 -0
  5. package/lib/components/chain-result-view/chain-result-view.component.d.ts +4 -1
  6. package/lib/components/feature-editor/feature-editor.component.d.ts +1 -1
  7. package/lib/components/feature-set-editor/feature-set-editor.component.d.ts +1 -1
  8. package/lib/components/snapshot-editor/snapshot-editor.component.d.ts +11 -9
  9. package/lib/components/steps-map/steps-map.component.d.ts +2 -0
  10. package/lib/models.d.ts +8 -0
  11. package/lib/services/gve-api.service.d.ts +33 -0
  12. package/package.json +10 -12
  13. package/public-api.d.ts +1 -0
  14. package/esm2022/lib/components/animation-timeline/animation-timeline.component.mjs +0 -207
  15. package/esm2022/lib/components/animation-timeline-set/animation-timeline-set.component.mjs +0 -154
  16. package/esm2022/lib/components/animation-tween/animation-tween.component.mjs +0 -184
  17. package/esm2022/lib/components/base-text-char/base-text-char.component.mjs +0 -46
  18. package/esm2022/lib/components/base-text-editor/base-text-editor.component.mjs +0 -115
  19. package/esm2022/lib/components/base-text-view/base-text-view.component.mjs +0 -159
  20. package/esm2022/lib/components/chain-operation-editor/chain-operation-editor.component.mjs +0 -570
  21. package/esm2022/lib/components/chain-result-view/chain-result-view.component.mjs +0 -222
  22. package/esm2022/lib/components/feature-editor/feature-editor.component.mjs +0 -200
  23. package/esm2022/lib/components/feature-set-editor/feature-set-editor.component.mjs +0 -175
  24. package/esm2022/lib/components/feature-set-view/feature-set-view.component.mjs +0 -100
  25. package/esm2022/lib/components/ln-heights-editor/ln-heights-editor.component.mjs +0 -126
  26. package/esm2022/lib/components/operation-source-editor/operation-source-editor.component.mjs +0 -131
  27. package/esm2022/lib/components/simple-tree/simple-tree.component.mjs +0 -72
  28. package/esm2022/lib/components/snapshot-editor/snapshot-editor.component.mjs +0 -863
  29. package/esm2022/lib/components/steps-map/steps-map.component.mjs +0 -77
  30. package/esm2022/lib/models.mjs +0 -2
  31. package/esm2022/lib/services/gve-api.service.mjs +0 -65
  32. package/esm2022/lib/services/settings.service.mjs +0 -87
  33. package/esm2022/lib/validators/svg-validators.mjs +0 -34
  34. package/esm2022/myrmidon-gve-core.mjs +0 -5
  35. package/esm2022/public-api.mjs +0 -22
@@ -1,222 +0,0 @@
1
- import { CommonModule } from '@angular/common';
2
- import { Component, EventEmitter, Input, Output, } from '@angular/core';
3
- import { ReactiveFormsModule } from '@angular/forms';
4
- import { debounceTime, distinctUntilChanged } from 'rxjs';
5
- import { MatButtonModule } from '@angular/material/button';
6
- import { MatFormFieldModule } from '@angular/material/form-field';
7
- import { MatIconModule } from '@angular/material/icon';
8
- import { MatSelectModule } from '@angular/material/select';
9
- import { FeatureSetViewComponent } from '../feature-set-view/feature-set-view.component';
10
- import { StepsMapComponent } from '../steps-map/steps-map.component';
11
- import { BaseTextViewComponent, } from '../base-text-view/base-text-view.component';
12
- import * as i0 from "@angular/core";
13
- import * as i1 from "@angular/forms";
14
- import * as i2 from "@angular/common";
15
- import * as i3 from "@angular/material/form-field";
16
- import * as i4 from "@angular/material/select";
17
- import * as i5 from "@angular/material/core";
18
- /**
19
- * 🔑 `gve-chain-result-view`
20
- *
21
- * Component to display a chain result.
22
- * Used by the `gve-snapshot-editor` component.
23
- *
24
- * - ▶️ `result` (`CharChainResult`): the result to display.
25
- * - 🔥 `stepPick` (`ChainOperationContextStep`): emitted when a
26
- * result's step is picked by the user.
27
- */
28
- export class ChainResultViewComponent {
29
- /**
30
- * The result to display.
31
- */
32
- get result() {
33
- return this._result;
34
- }
35
- set result(value) {
36
- if (this._result === value) {
37
- return;
38
- }
39
- this._result = value || undefined;
40
- this.updateForm(this._result);
41
- // select the initial step if set
42
- this.selectInitialStep();
43
- }
44
- /**
45
- * The index of the initial step to display after the result is set.
46
- * If the index is negative, it is counted from the end of the steps.
47
- */
48
- get initialStepIndex() {
49
- return this._initialStepIndex;
50
- }
51
- set initialStepIndex(value) {
52
- if (value === undefined || value === null) {
53
- this._initialStepIndex = undefined;
54
- this.step = undefined;
55
- }
56
- else {
57
- this._initialStepIndex = value;
58
- this.selectInitialStep();
59
- }
60
- }
61
- constructor(formBuilder) {
62
- /**
63
- * Emitted when a result's step is picked by the user.
64
- */
65
- this.stepPick = new EventEmitter();
66
- this.versionTags = [];
67
- this.tags = [];
68
- this.selectionFeatures = [];
69
- this.versionTag = formBuilder.control(null);
70
- this.tag = formBuilder.control(null);
71
- }
72
- ngOnInit() {
73
- this._subs = [
74
- // whenever the staged version tag changes, update the form
75
- this.versionTag.valueChanges
76
- .pipe(distinctUntilChanged(), debounceTime(200))
77
- .subscribe((value) => {
78
- this.tag.setValue(this.getTagFromVersion(value) || null);
79
- }),
80
- // whenever the tag changes, update the step
81
- this.tag.valueChanges
82
- .pipe(distinctUntilChanged(), debounceTime(200))
83
- .subscribe((value) => {
84
- this.step = this._result?.steps.find((step) => step.outputTag === value);
85
- if (this.step) {
86
- this.stepPick.emit(this.step);
87
- }
88
- }),
89
- ];
90
- }
91
- selectInitialStep() {
92
- if (this._initialStepIndex !== undefined && this._result?.steps) {
93
- this.step =
94
- this._result.steps[this._initialStepIndex < 0
95
- ? this._result.steps.length + this._initialStepIndex
96
- : this._initialStepIndex];
97
- }
98
- }
99
- ngOnDestroy() {
100
- this._subs?.forEach((sub) => sub.unsubscribe());
101
- }
102
- updateVersionTags() {
103
- if (!this._result) {
104
- this.versionTags = [];
105
- return;
106
- }
107
- // extract version tags from result steps features named 'version'
108
- const tags = new Set();
109
- for (const step of this._result.steps) {
110
- for (const feature of step.featureSet.features) {
111
- if (feature.name === 'version') {
112
- tags.add(feature.value);
113
- }
114
- }
115
- }
116
- this.versionTags = Array.from(tags).sort();
117
- }
118
- getTagFromVersion(version) {
119
- if (!this._result || !version) {
120
- return undefined;
121
- }
122
- for (const step of this._result.steps) {
123
- for (const feature of step.featureSet.features) {
124
- if (feature.name === 'version' && feature.value === version) {
125
- return step.outputTag;
126
- }
127
- }
128
- }
129
- return undefined;
130
- }
131
- updateForm(result) {
132
- this.selectionFeatures = [];
133
- if (!result) {
134
- this.step = undefined;
135
- this.selection = undefined;
136
- this.tags = [];
137
- this.versionTags = [];
138
- }
139
- else {
140
- this.tags = result.chainTags;
141
- this.updateVersionTags();
142
- }
143
- }
144
- getNodeFeatures(id) {
145
- if (!this.step) {
146
- return [];
147
- }
148
- return (this.step.featureSet.nodeFeatures[this.step.outputTag + '_' + id] || []);
149
- }
150
- findNode(id) {
151
- if (!this.step) {
152
- return null;
153
- }
154
- for (let key of Object.keys(this._result.taggedNodes)) {
155
- const node = this._result.taggedNodes[key].find((n) => n.id === id);
156
- if (node) {
157
- return node;
158
- }
159
- }
160
- return null;
161
- }
162
- onStepCharPick(event) {
163
- const features = this.getNodeFeatures(event.char.id);
164
- const node = this.findNode(event.char.id);
165
- this.selection = `${event.char.label} (${node.sourceTag})`;
166
- this.selectionFeatures = [...features];
167
- }
168
- onStepRangePick(range) {
169
- if (range.run === 1) {
170
- // already handled by onStepCharPick
171
- return;
172
- }
173
- this.selection = range.at + '×' + range.run;
174
- const end = range.at + range.run;
175
- const features = [];
176
- const tagNodes = this._result.taggedNodes[this.tag.value];
177
- for (let node of tagNodes) {
178
- // if in range, intersect features
179
- if (node.id >= range.at && node.id < end) {
180
- const nodeFeatures = this.getNodeFeatures(node.id);
181
- if (node.id === range.at) {
182
- for (let f of nodeFeatures) {
183
- features.push(f);
184
- }
185
- }
186
- else {
187
- // features = intersection of features and charFeatures
188
- features.splice(0, features.length, ...features.filter((f) => nodeFeatures.some((cf) => cf.name === f.name)));
189
- }
190
- }
191
- }
192
- this.selectionFeatures = features;
193
- }
194
- onStepChange(step) {
195
- // setting the tag will trigger the step change
196
- this.tag.setValue(step.outputTag);
197
- this.selectionFeatures = [];
198
- }
199
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.8", ngImport: i0, type: ChainResultViewComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component }); }
200
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.8", type: ChainResultViewComponent, isStandalone: true, selector: "gve-chain-result-view", inputs: { result: "result", initialStepIndex: "initialStepIndex" }, outputs: { stepPick: "stepPick" }, ngImport: i0, template: "@if (result) {\r\n<div id=\"container\">\r\n <div id=\"bar\" class=\"form-row\">\r\n <!-- version -->\r\n <mat-form-field *ngIf=\"versionTags?.length\">\r\n <mat-label>version</mat-label>\r\n <mat-select [formControl]=\"versionTag\">\r\n <mat-option [value]=\"null\">-</mat-option>\r\n <mat-option *ngFor=\"let t of versionTags\" [value]=\"t\">{{\r\n t\r\n }}</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n <!-- tag -->\r\n <mat-form-field *ngIf=\"tags?.length\">\r\n <mat-label>tag</mat-label>\r\n <mat-select [formControl]=\"tag\">\r\n <mat-option *ngFor=\"let t of tags\" [value]=\"t\">{{ t }}</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- step -->\r\n @if (step) {\r\n <div id=\"step\">\r\n <!-- text -->\r\n <div id=\"text\">\r\n <gve-base-text-view\r\n [text]=\"result.taggedNodes[step.outputTag]\"\r\n (charPick)=\"onStepCharPick($event)\"\r\n (rangePick)=\"onStepRangePick($event)\"\r\n />\r\n </div>\r\n <!-- features -->\r\n <div id=\"feats\" class=\"form-row-top\">\r\n <div id=\"g-feats\">\r\n <div class=\"feat-header\">{{ step.outputTag }}</div>\r\n <gve-feature-set-view [features]=\"step.featureSet.features\" />\r\n </div>\r\n <div id=\"n-feats\">\r\n @if (selectionFeatures.length) {\r\n <div class=\"feat-header\">{{ selection }}</div>\r\n <gve-feature-set-view [features]=\"selectionFeatures\" />\r\n }\r\n </div>\r\n </div>\r\n </div>\r\n }\r\n\r\n <!-- steps map -->\r\n <div id=\"map\">\r\n <gve-steps-map\r\n [steps]=\"result.steps\"\r\n [selectedStep]=\"step\"\r\n (selectedStepChange)=\"onStepChange($event)\"\r\n />\r\n </div>\r\n</div>\r\n}\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}.form-row-top{display:flex;gap:8px;align-items:start;flex-wrap:wrap}div#container{display:grid;grid-template-rows:auto 1fr;grid-template-columns:3fr 1fr;grid-template-areas:\"bar map\" \"step map\";gap:8px}div#bar{grid-area:bar}div#map{grid-area:map}div#step{grid-area:step}.feat-header{background-color:#3e92cc;color:#fff;text-align:center;border:1px solid #3e92cc;border-top-left-radius:4px;border-top-right-radius:4px}@media only screen and (max-width: 959px){div#container{grid-template-columns:1fr;grid-template-areas:\"bar\" \"step\"}#map{display:none}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { 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: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i3.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i3.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatIconModule }, { kind: "ngmodule", type: MatSelectModule }, { kind: "component", type: i4.MatSelect, selector: "mat-select", inputs: ["aria-describedby", "panelClass", "disabled", "disableRipple", "tabIndex", "hideSingleSelectionIndicator", "placeholder", "required", "multiple", "disableOptionCentering", "compareWith", "value", "aria-label", "aria-labelledby", "errorStateMatcher", "typeaheadDebounceInterval", "sortComparator", "id", "panelWidth"], outputs: ["openedChange", "opened", "closed", "selectionChange", "valueChange"], exportAs: ["matSelect"] }, { kind: "component", type: i5.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { kind: "component", type: FeatureSetViewComponent, selector: "gve-feature-set-view", inputs: ["features", "featNames", "featValues", "filterThreshold"] }, { kind: "component", type: StepsMapComponent, selector: "gve-steps-map", inputs: ["steps", "selectedStep", "textFontSize"], outputs: ["selectedStepChange"] }, { kind: "component", type: BaseTextViewComponent, selector: "gve-base-text-view", inputs: ["defaultColor", "defaultBorderColor", "selectionColor", "hasLineNumber", "text"], outputs: ["charPick", "rangePick"] }] }); }
201
- }
202
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImport: i0, type: ChainResultViewComponent, decorators: [{
203
- type: Component,
204
- args: [{ selector: 'gve-chain-result-view', standalone: true, imports: [
205
- CommonModule,
206
- ReactiveFormsModule,
207
- MatButtonModule,
208
- MatFormFieldModule,
209
- MatIconModule,
210
- MatSelectModule,
211
- FeatureSetViewComponent,
212
- StepsMapComponent,
213
- BaseTextViewComponent,
214
- ], template: "@if (result) {\r\n<div id=\"container\">\r\n <div id=\"bar\" class=\"form-row\">\r\n <!-- version -->\r\n <mat-form-field *ngIf=\"versionTags?.length\">\r\n <mat-label>version</mat-label>\r\n <mat-select [formControl]=\"versionTag\">\r\n <mat-option [value]=\"null\">-</mat-option>\r\n <mat-option *ngFor=\"let t of versionTags\" [value]=\"t\">{{\r\n t\r\n }}</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n <!-- tag -->\r\n <mat-form-field *ngIf=\"tags?.length\">\r\n <mat-label>tag</mat-label>\r\n <mat-select [formControl]=\"tag\">\r\n <mat-option *ngFor=\"let t of tags\" [value]=\"t\">{{ t }}</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- step -->\r\n @if (step) {\r\n <div id=\"step\">\r\n <!-- text -->\r\n <div id=\"text\">\r\n <gve-base-text-view\r\n [text]=\"result.taggedNodes[step.outputTag]\"\r\n (charPick)=\"onStepCharPick($event)\"\r\n (rangePick)=\"onStepRangePick($event)\"\r\n />\r\n </div>\r\n <!-- features -->\r\n <div id=\"feats\" class=\"form-row-top\">\r\n <div id=\"g-feats\">\r\n <div class=\"feat-header\">{{ step.outputTag }}</div>\r\n <gve-feature-set-view [features]=\"step.featureSet.features\" />\r\n </div>\r\n <div id=\"n-feats\">\r\n @if (selectionFeatures.length) {\r\n <div class=\"feat-header\">{{ selection }}</div>\r\n <gve-feature-set-view [features]=\"selectionFeatures\" />\r\n }\r\n </div>\r\n </div>\r\n </div>\r\n }\r\n\r\n <!-- steps map -->\r\n <div id=\"map\">\r\n <gve-steps-map\r\n [steps]=\"result.steps\"\r\n [selectedStep]=\"step\"\r\n (selectedStepChange)=\"onStepChange($event)\"\r\n />\r\n </div>\r\n</div>\r\n}\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}.form-row-top{display:flex;gap:8px;align-items:start;flex-wrap:wrap}div#container{display:grid;grid-template-rows:auto 1fr;grid-template-columns:3fr 1fr;grid-template-areas:\"bar map\" \"step map\";gap:8px}div#bar{grid-area:bar}div#map{grid-area:map}div#step{grid-area:step}.feat-header{background-color:#3e92cc;color:#fff;text-align:center;border:1px solid #3e92cc;border-top-left-radius:4px;border-top-right-radius:4px}@media only screen and (max-width: 959px){div#container{grid-template-columns:1fr;grid-template-areas:\"bar\" \"step\"}#map{display:none}}\n"] }]
215
- }], ctorParameters: () => [{ type: i1.FormBuilder }], propDecorators: { result: [{
216
- type: Input
217
- }], initialStepIndex: [{
218
- type: Input
219
- }], stepPick: [{
220
- type: Output
221
- }] } });
222
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chain-result-view.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/gve-core/src/lib/components/chain-result-view/chain-result-view.component.ts","../../../../../../../projects/myrmidon/gve-core/src/lib/components/chain-result-view/chain-result-view.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport {\r\n  Component,\r\n  EventEmitter,\r\n  Input,\r\n  OnDestroy,\r\n  OnInit,\r\n  Output,\r\n} from '@angular/core';\r\nimport { FormBuilder, FormControl, ReactiveFormsModule } from '@angular/forms';\r\nimport { Subscription, debounceTime, distinctUntilChanged } from 'rxjs';\r\n\r\nimport { MatButtonModule } from '@angular/material/button';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatIconModule } from '@angular/material/icon';\r\nimport { MatSelectModule } from '@angular/material/select';\r\n\r\nimport {\r\n  CharNode,\r\n  Feature,\r\n  OperationFeature,\r\n} from '@myrmidon/gve-snapshot-view';\r\n\r\nimport { FeatureSetViewComponent } from '../feature-set-view/feature-set-view.component';\r\nimport { StepsMapComponent } from '../steps-map/steps-map.component';\r\nimport {\r\n  BaseTextViewComponent,\r\n  VarBaseTextRange,\r\n} from '../base-text-view/base-text-view.component';\r\nimport {\r\n  ChainOperationContextStep,\r\n  CharChainResult,\r\n} from '../../services/gve-api.service';\r\nimport { BaseTextCharEvent } from '../base-text-char/base-text-char.component';\r\n\r\n/**\r\n * 🔑 `gve-chain-result-view`\r\n *\r\n * Component to display a chain result.\r\n * Used by the `gve-snapshot-editor` component.\r\n *\r\n * - ▶️ `result` (`CharChainResult`): the result to display.\r\n * - 🔥 `stepPick` (`ChainOperationContextStep`): emitted when a\r\n * result's step is picked by the user.\r\n */\r\n@Component({\r\n  selector: 'gve-chain-result-view',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    ReactiveFormsModule,\r\n    MatButtonModule,\r\n    MatFormFieldModule,\r\n    MatIconModule,\r\n    MatSelectModule,\r\n    FeatureSetViewComponent,\r\n    StepsMapComponent,\r\n    BaseTextViewComponent,\r\n  ],\r\n  templateUrl: './chain-result-view.component.html',\r\n  styleUrl: './chain-result-view.component.css',\r\n})\r\nexport class ChainResultViewComponent implements OnInit, OnDestroy {\r\n  private _subs?: Subscription[];\r\n  private _result?: CharChainResult;\r\n  private _initialStepIndex?: number;\r\n\r\n  /**\r\n   * The result to display.\r\n   */\r\n  @Input()\r\n  public get result(): CharChainResult | undefined {\r\n    return this._result;\r\n  }\r\n  public set result(value: CharChainResult | undefined | null) {\r\n    if (this._result === value) {\r\n      return;\r\n    }\r\n    this._result = value || undefined;\r\n    this.updateForm(this._result);\r\n\r\n    // select the initial step if set\r\n    this.selectInitialStep();\r\n  }\r\n\r\n  /**\r\n   * The index of the initial step to display after the result is set.\r\n   * If the index is negative, it is counted from the end of the steps.\r\n   */\r\n  @Input()\r\n  public get initialStepIndex(): number | undefined {\r\n    return this._initialStepIndex;\r\n  }\r\n  public set initialStepIndex(value: number | undefined | null) {\r\n    if (value === undefined || value === null) {\r\n      this._initialStepIndex = undefined;\r\n      this.step = undefined;\r\n    } else {\r\n      this._initialStepIndex = value;\r\n      this.selectInitialStep();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Emitted when a result's step is picked by the user.\r\n   */\r\n  @Output()\r\n  public stepPick = new EventEmitter<ChainOperationContextStep>();\r\n\r\n  public versionTags: string[] = [];\r\n  public tags: string[] = [];\r\n  public versionTag: FormControl<string | null>;\r\n  public tag: FormControl<string | null>;\r\n  public step?: ChainOperationContextStep;\r\n  public selection?: string;\r\n  public selectionFeatures: OperationFeature[] = [];\r\n\r\n  constructor(formBuilder: FormBuilder) {\r\n    this.versionTag = formBuilder.control<string | null>(null);\r\n    this.tag = formBuilder.control<string | null>(null);\r\n  }\r\n\r\n  public ngOnInit(): void {\r\n    this._subs = [\r\n      // whenever the staged version tag changes, update the form\r\n      this.versionTag.valueChanges\r\n        .pipe(distinctUntilChanged(), debounceTime(200))\r\n        .subscribe((value) => {\r\n          this.tag.setValue(this.getTagFromVersion(value) || null);\r\n        }),\r\n\r\n      // whenever the tag changes, update the step\r\n      this.tag.valueChanges\r\n        .pipe(distinctUntilChanged(), debounceTime(200))\r\n        .subscribe((value) => {\r\n          this.step = this._result?.steps.find(\r\n            (step) => step.outputTag === value\r\n          );\r\n          if (this.step) {\r\n            this.stepPick.emit(this.step!);\r\n          }\r\n        }),\r\n    ];\r\n  }\r\n\r\n  private selectInitialStep(): void {\r\n    if (this._initialStepIndex !== undefined && this._result?.steps) {\r\n      this.step =\r\n        this._result.steps[\r\n          this._initialStepIndex < 0\r\n            ? this._result.steps.length + this._initialStepIndex\r\n            : this._initialStepIndex\r\n        ];\r\n    }\r\n  }\r\n\r\n  public ngOnDestroy(): void {\r\n    this._subs?.forEach((sub) => sub.unsubscribe());\r\n  }\r\n\r\n  private updateVersionTags(): void {\r\n    if (!this._result) {\r\n      this.versionTags = [];\r\n      return;\r\n    }\r\n\r\n    // extract version tags from result steps features named 'version'\r\n    const tags = new Set<string>();\r\n    for (const step of this._result.steps) {\r\n      for (const feature of step.featureSet.features) {\r\n        if (feature.name === 'version') {\r\n          tags.add(feature.value);\r\n        }\r\n      }\r\n    }\r\n    this.versionTags = Array.from(tags).sort();\r\n  }\r\n\r\n  private getTagFromVersion(version?: string | null): string | undefined {\r\n    if (!this._result || !version) {\r\n      return undefined;\r\n    }\r\n    for (const step of this._result.steps) {\r\n      for (const feature of step.featureSet.features) {\r\n        if (feature.name === 'version' && feature.value === version) {\r\n          return step.outputTag;\r\n        }\r\n      }\r\n    }\r\n    return undefined;\r\n  }\r\n\r\n  private updateForm(result?: CharChainResult): void {\r\n    this.selectionFeatures = [];\r\n\r\n    if (!result) {\r\n      this.step = undefined;\r\n      this.selection = undefined;\r\n      this.tags = [];\r\n      this.versionTags = [];\r\n    } else {\r\n      this.tags = result.chainTags;\r\n      this.updateVersionTags();\r\n    }\r\n  }\r\n\r\n  private getNodeFeatures(id: number): Feature[] {\r\n    if (!this.step) {\r\n      return [];\r\n    }\r\n    return (\r\n      this.step.featureSet.nodeFeatures[this.step.outputTag + '_' + id] || []\r\n    );\r\n  }\r\n\r\n  private findNode(id: number): CharNode | null {\r\n    if (!this.step) {\r\n      return null;\r\n    }\r\n    for (let key of Object.keys(this._result!.taggedNodes)) {\r\n      const node = this._result!.taggedNodes[key].find((n) => n.id === id);\r\n      if (node) {\r\n        return node;\r\n      }\r\n    }\r\n    return null;\r\n  }\r\n\r\n  public onStepCharPick(event: BaseTextCharEvent): void {\r\n    const features = this.getNodeFeatures(event.char.id);\r\n    const node = this.findNode(event.char.id)!;\r\n    this.selection = `${event.char.label} (${node.sourceTag})`;\r\n    this.selectionFeatures = [...features];\r\n  }\r\n\r\n  public onStepRangePick(range: VarBaseTextRange): void {\r\n    if (range.run === 1) {\r\n      // already handled by onStepCharPick\r\n      return;\r\n    }\r\n    this.selection = range.at + '×' + range.run;\r\n    const end = range.at + range.run;\r\n\r\n    const features: Feature[] = [];\r\n    const tagNodes = this._result!.taggedNodes[this.tag.value!];\r\n\r\n    for (let node of tagNodes) {\r\n      // if in range, intersect features\r\n      if (node.id >= range.at && node.id < end) {\r\n        const nodeFeatures = this.getNodeFeatures(node.id);\r\n\r\n        if (node.id === range.at) {\r\n          for (let f of nodeFeatures) {\r\n            features.push(f);\r\n          }\r\n        } else {\r\n          // features = intersection of features and charFeatures\r\n          features.splice(\r\n            0,\r\n            features.length,\r\n            ...features.filter((f) =>\r\n              nodeFeatures.some((cf) => cf.name === f.name)\r\n            )\r\n          );\r\n        }\r\n      }\r\n    }\r\n\r\n    this.selectionFeatures = features;\r\n  }\r\n\r\n  public onStepChange(step: ChainOperationContextStep): void {\r\n    // setting the tag will trigger the step change\r\n    this.tag.setValue(step.outputTag);\r\n    this.selectionFeatures = [];\r\n  }\r\n}\r\n","@if (result) {\r\n<div id=\"container\">\r\n  <div id=\"bar\" class=\"form-row\">\r\n    <!-- version -->\r\n    <mat-form-field *ngIf=\"versionTags?.length\">\r\n      <mat-label>version</mat-label>\r\n      <mat-select [formControl]=\"versionTag\">\r\n        <mat-option [value]=\"null\">-</mat-option>\r\n        <mat-option *ngFor=\"let t of versionTags\" [value]=\"t\">{{\r\n          t\r\n        }}</mat-option>\r\n      </mat-select>\r\n    </mat-form-field>\r\n    <!-- tag -->\r\n    <mat-form-field *ngIf=\"tags?.length\">\r\n      <mat-label>tag</mat-label>\r\n      <mat-select [formControl]=\"tag\">\r\n        <mat-option *ngFor=\"let t of tags\" [value]=\"t\">{{ t }}</mat-option>\r\n      </mat-select>\r\n    </mat-form-field>\r\n  </div>\r\n\r\n  <!-- step -->\r\n  @if (step) {\r\n  <div id=\"step\">\r\n    <!-- text -->\r\n    <div id=\"text\">\r\n      <gve-base-text-view\r\n        [text]=\"result.taggedNodes[step.outputTag]\"\r\n        (charPick)=\"onStepCharPick($event)\"\r\n        (rangePick)=\"onStepRangePick($event)\"\r\n      />\r\n    </div>\r\n    <!-- features -->\r\n    <div id=\"feats\" class=\"form-row-top\">\r\n      <div id=\"g-feats\">\r\n        <div class=\"feat-header\">{{ step.outputTag }}</div>\r\n        <gve-feature-set-view [features]=\"step.featureSet.features\" />\r\n      </div>\r\n      <div id=\"n-feats\">\r\n        @if (selectionFeatures.length) {\r\n        <div class=\"feat-header\">{{ selection }}</div>\r\n        <gve-feature-set-view [features]=\"selectionFeatures\" />\r\n        }\r\n      </div>\r\n    </div>\r\n  </div>\r\n  }\r\n\r\n  <!-- steps map -->\r\n  <div id=\"map\">\r\n    <gve-steps-map\r\n      [steps]=\"result.steps\"\r\n      [selectedStep]=\"step\"\r\n      (selectedStepChange)=\"onStepChange($event)\"\r\n    />\r\n  </div>\r\n</div>\r\n}\r\n"]}
@@ -1,200 +0,0 @@
1
- import { Component, EventEmitter, Input, Output, ViewChild, } from '@angular/core';
2
- import { ReactiveFormsModule, Validators, } from '@angular/forms';
3
- import { CommonModule } from '@angular/common';
4
- import { debounceTime, distinctUntilChanged } from 'rxjs';
5
- // material
6
- import { MatButtonModule } from '@angular/material/button';
7
- import { MatCheckboxModule } from '@angular/material/checkbox';
8
- import { MatFormFieldModule } from '@angular/material/form-field';
9
- import { MatIconModule } from '@angular/material/icon';
10
- import { MatInputModule } from '@angular/material/input';
11
- import { MatSelectModule } from '@angular/material/select';
12
- import { MatTooltipModule } from '@angular/material/tooltip';
13
- import { FeatureSetPolicy, } from '@myrmidon/gve-snapshot-view';
14
- import * as i0 from "@angular/core";
15
- import * as i1 from "@angular/forms";
16
- import * as i2 from "@angular/common";
17
- import * as i3 from "@angular/material/button";
18
- import * as i4 from "@angular/material/checkbox";
19
- import * as i5 from "@angular/material/form-field";
20
- import * as i6 from "@angular/material/icon";
21
- import * as i7 from "@angular/material/input";
22
- import * as i8 from "@angular/material/select";
23
- import * as i9 from "@angular/material/core";
24
- import * as i10 from "@angular/material/tooltip";
25
- /**
26
- * 🔑 `gve-feature-editor`
27
- *
28
- * Component for editing a single feature, whose model is a name=value
29
- * pair, plus a set policy value which defines the desired behavior
30
- * when adding that feature to a set.
31
- * Used by `gve-feature-set-editor`.
32
- *
33
- * - ▶️ `feature` (`Feature`): the feature to edit.
34
- * - ▶️ `featNames` (`LabeledId[]`): the list of feature names to display
35
- * in the _name_ selection. This is used when you have a closed list of
36
- * features. Each item in the list is an object with _id_ and _label_
37
- * properties.
38
- * - ▶️ `featValues` (`FeatureMap`): the feature values map. When
39
- * specified and the user selects a feature name present in the map keys,
40
- * the corresponding values will be used to populate the value selection.
41
- * - 🔥 `featureChange` (`Feature`): emitted when feature has changed.
42
- * - 🔥 `featureCancel`: emitted when the user cancels the feature editing.
43
- */
44
- export class FeatureEditorComponent {
45
- /**
46
- * The feature values map. When specified and the user selects a feature
47
- * name present in the map keys, the corresponding values will be used
48
- * to populate the value selection.
49
- */
50
- get featValues() {
51
- return this._map;
52
- }
53
- set featValues(value) {
54
- this._map = value || undefined;
55
- this.nameIds = this.getLabeledIdsFor(this.name.value);
56
- }
57
- /**
58
- * The feature to edit.
59
- */
60
- get feature() {
61
- return this._feature;
62
- }
63
- set feature(value) {
64
- if (this._feature === value) {
65
- return;
66
- }
67
- this._feature = value || undefined;
68
- this.updateForm(this._feature);
69
- setTimeout(() => {
70
- this.nameControl?.nativeElement?.focus();
71
- }, 500);
72
- }
73
- constructor(formBuilder) {
74
- /**
75
- * True if the feature is a variant operation feature, which has
76
- * additional properties like negation, global, and short-lived.
77
- */
78
- this.isVar = false;
79
- /**
80
- * Event emitted when the user cancels the feature editing.
81
- */
82
- this.featureCancel = new EventEmitter();
83
- /**
84
- * Event emitted when the user saves the edited feature.
85
- */
86
- this.featureChange = new EventEmitter();
87
- this.name = formBuilder.control('', {
88
- validators: [Validators.required, Validators.maxLength(50)],
89
- nonNullable: true,
90
- });
91
- this.value = formBuilder.control('', {
92
- validators: [Validators.maxLength(5000)],
93
- nonNullable: true,
94
- });
95
- this.setPolicy = formBuilder.control(FeatureSetPolicy.multiple, {
96
- nonNullable: true,
97
- });
98
- this.isNegated = formBuilder.control(false, { nonNullable: true });
99
- this.isGlobal = formBuilder.control(false, { nonNullable: true });
100
- this.isShortLived = formBuilder.control(false, { nonNullable: true });
101
- this.form = formBuilder.group({
102
- name: this.name,
103
- value: this.value,
104
- setPolicy: this.setPolicy,
105
- isNegated: this.isNegated,
106
- isGlobal: this.isGlobal,
107
- isShortLived: this.isShortLived,
108
- });
109
- }
110
- ngOnInit() {
111
- // whenever the name changes, update the value selection
112
- this._sub = this.name.valueChanges
113
- .pipe(debounceTime(300), distinctUntilChanged())
114
- .subscribe((name) => {
115
- if (!this._frozen && this.featNames?.length) {
116
- this.value.reset();
117
- this.nameIds = this.getLabeledIdsFor(name);
118
- }
119
- });
120
- }
121
- ngOnDestroy() {
122
- this._sub?.unsubscribe();
123
- }
124
- getLabeledIdsFor(id) {
125
- if (!id) {
126
- return undefined;
127
- }
128
- return this.featValues ? this.featValues[id] : undefined;
129
- }
130
- updateForm(feature) {
131
- if (!feature) {
132
- this.form.reset();
133
- }
134
- else {
135
- this._frozen = true;
136
- this.name.setValue(feature.name);
137
- this.value.setValue(feature.value);
138
- this.setPolicy.setValue(feature.setPolicy || FeatureSetPolicy.multiple);
139
- this.isNegated.setValue(feature.isNegated || false);
140
- this.isGlobal.setValue(feature.isGlobal || false);
141
- this.isShortLived.setValue(feature.isShortLived || false);
142
- this._frozen = false;
143
- }
144
- }
145
- cancel() {
146
- this.featureCancel.emit();
147
- }
148
- save() {
149
- if (!this.form.valid) {
150
- return;
151
- }
152
- this.feature = this.isVar
153
- ? {
154
- name: this.name.value.trim(),
155
- value: this.value.value.trim(),
156
- setPolicy: this.setPolicy.value,
157
- isNegated: this.isNegated.value,
158
- isGlobal: this.isGlobal.value,
159
- isShortLived: this.isShortLived.value,
160
- }
161
- : {
162
- name: this.name.value.trim(),
163
- value: this.value.value.trim(),
164
- setPolicy: this.setPolicy.value,
165
- };
166
- this.featureChange.emit(this.feature);
167
- }
168
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.8", ngImport: i0, type: FeatureEditorComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component }); }
169
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.8", type: FeatureEditorComponent, isStandalone: true, selector: "gve-feature-editor", inputs: { featNames: "featNames", featValues: "featValues", feature: "feature", isVar: "isVar" }, outputs: { featureCancel: "featureCancel", featureChange: "featureChange" }, viewQueries: [{ propertyName: "nameControl", first: true, predicate: ["nameCtl"], descendants: true }], ngImport: i0, template: "<form [formGroup]=\"form\" (submit)=\"save()\">\r\n <div class=\"form-row\">\r\n @if (featNames?.length) {\r\n <!-- name (bound) -->\r\n <mat-form-field>\r\n <mat-label>name</mat-label>\r\n <mat-select #nameCtl [formControl]=\"name\">\r\n <mat-option *ngFor=\"let i of featNames\" [value]=\"i.id\">{{\r\n i.label\r\n }}</mat-option>\r\n </mat-select>\r\n @if ($any(name).errors?.required && (name.dirty || name.touched)) {\r\n <mat-error>name required</mat-error>\r\n }\r\n </mat-form-field>\r\n } @else {\r\n <!-- name (free) -->\r\n <mat-form-field>\r\n <mat-label>name</mat-label>\r\n <input #nameCtl matInput [formControl]=\"name\" />\r\n <mat-error\r\n *ngIf=\"$any(name).errors?.required && (name.dirty || name.touched)\"\r\n >name required</mat-error\r\n >\r\n @if ($any(name).errors?.maxLength && (name.dirty || name.touched)) {\r\n <mat-error>name too long</mat-error>\r\n }\r\n </mat-form-field>\r\n }\r\n\r\n <!-- value (bound) -->\r\n @if (nameIds?.length) {\r\n <mat-form-field>\r\n <mat-label>value</mat-label>\r\n <mat-select [formControl]=\"value\">\r\n <mat-option *ngFor=\"let e of nameIds\" [value]=\"e.id\">{{\r\n e.label\r\n }}</mat-option>\r\n </mat-select>\r\n @if ($any(value).errors?.required && (value.dirty || value.touched)) {\r\n <mat-error>value required</mat-error>\r\n }\r\n </mat-form-field>\r\n } @else {\r\n <!-- value (free) -->\r\n <mat-form-field>\r\n <mat-label>value</mat-label>\r\n <input matInput [formControl]=\"value\" />\r\n @if ($any(value).errors?.required && (value.dirty || value.touched)) {\r\n <mat-error>value required</mat-error>\r\n } @if ($any(value).errors?.maxLength && (value.dirty || value.touched)) {\r\n <mat-error>value too long</mat-error>\r\n }\r\n </mat-form-field>\r\n }\r\n\r\n <!-- setPolicy -->\r\n <mat-form-field>\r\n <mat-label>set policy</mat-label>\r\n <mat-select [formControl]=\"setPolicy\">\r\n <mat-option [value]=\"0\">multiple</mat-option>\r\n <mat-option [value]=\"1\">single</mat-option>\r\n <mat-option [value]=\"2\">single first</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n @if (isVar) {\r\n <div class=\"form-row\">\r\n <!-- isNegated -->\r\n <mat-checkbox [formControl]=\"isNegated\">negated</mat-checkbox>\r\n\r\n <!-- isShortLived -->\r\n <mat-checkbox [formControl]=\"isShortLived\">short-lived</mat-checkbox>\r\n\r\n <!-- isGlobal -->\r\n <mat-checkbox [formControl]=\"isGlobal\">global</mat-checkbox>\r\n </div>\r\n }\r\n\r\n <!-- buttons -->\r\n <div class=\"form-row\">\r\n <button\r\n type=\"button\"\r\n color=\"warn\"\r\n mat-icon-button\r\n matTooltip=\"Discard changes\"\r\n (click)=\"cancel()\"\r\n >\r\n <mat-icon class=\"mat-warn\">clear</mat-icon>\r\n </button>\r\n <button\r\n type=\"submit\"\r\n color=\"primary\"\r\n mat-icon-button\r\n matTooltip=\"Accept changes\"\r\n [disabled]=\"form.invalid || form.pristine\"\r\n >\r\n <mat-icon class=\"mat-primary\">check_circle</mat-icon>\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 *{flex:0 0 auto}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { 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.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { 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: i3.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatCheckboxModule }, { kind: "component", type: i4.MatCheckbox, selector: "mat-checkbox", inputs: ["aria-label", "aria-labelledby", "aria-describedby", "id", "required", "labelPosition", "name", "value", "disableRipple", "tabIndex", "color", "disabledInteractive", "checked", "disabled", "indeterminate"], outputs: ["change", "indeterminateChange"], exportAs: ["matCheckbox"] }, { 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: "directive", type: i5.MatError, selector: "mat-error, [matError]", inputs: ["id"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i6.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i7.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: MatSelectModule }, { kind: "component", type: i8.MatSelect, selector: "mat-select", inputs: ["aria-describedby", "panelClass", "disabled", "disableRipple", "tabIndex", "hideSingleSelectionIndicator", "placeholder", "required", "multiple", "disableOptionCentering", "compareWith", "value", "aria-label", "aria-labelledby", "errorStateMatcher", "typeaheadDebounceInterval", "sortComparator", "id", "panelWidth"], outputs: ["openedChange", "opened", "closed", "selectionChange", "valueChange"], exportAs: ["matSelect"] }, { kind: "component", type: i9.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i10.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }] }); }
170
- }
171
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.8", ngImport: i0, type: FeatureEditorComponent, decorators: [{
172
- type: Component,
173
- args: [{ selector: 'gve-feature-editor', standalone: true, imports: [
174
- CommonModule,
175
- ReactiveFormsModule,
176
- MatButtonModule,
177
- MatCheckboxModule,
178
- MatFormFieldModule,
179
- MatIconModule,
180
- MatInputModule,
181
- MatSelectModule,
182
- MatTooltipModule,
183
- ], template: "<form [formGroup]=\"form\" (submit)=\"save()\">\r\n <div class=\"form-row\">\r\n @if (featNames?.length) {\r\n <!-- name (bound) -->\r\n <mat-form-field>\r\n <mat-label>name</mat-label>\r\n <mat-select #nameCtl [formControl]=\"name\">\r\n <mat-option *ngFor=\"let i of featNames\" [value]=\"i.id\">{{\r\n i.label\r\n }}</mat-option>\r\n </mat-select>\r\n @if ($any(name).errors?.required && (name.dirty || name.touched)) {\r\n <mat-error>name required</mat-error>\r\n }\r\n </mat-form-field>\r\n } @else {\r\n <!-- name (free) -->\r\n <mat-form-field>\r\n <mat-label>name</mat-label>\r\n <input #nameCtl matInput [formControl]=\"name\" />\r\n <mat-error\r\n *ngIf=\"$any(name).errors?.required && (name.dirty || name.touched)\"\r\n >name required</mat-error\r\n >\r\n @if ($any(name).errors?.maxLength && (name.dirty || name.touched)) {\r\n <mat-error>name too long</mat-error>\r\n }\r\n </mat-form-field>\r\n }\r\n\r\n <!-- value (bound) -->\r\n @if (nameIds?.length) {\r\n <mat-form-field>\r\n <mat-label>value</mat-label>\r\n <mat-select [formControl]=\"value\">\r\n <mat-option *ngFor=\"let e of nameIds\" [value]=\"e.id\">{{\r\n e.label\r\n }}</mat-option>\r\n </mat-select>\r\n @if ($any(value).errors?.required && (value.dirty || value.touched)) {\r\n <mat-error>value required</mat-error>\r\n }\r\n </mat-form-field>\r\n } @else {\r\n <!-- value (free) -->\r\n <mat-form-field>\r\n <mat-label>value</mat-label>\r\n <input matInput [formControl]=\"value\" />\r\n @if ($any(value).errors?.required && (value.dirty || value.touched)) {\r\n <mat-error>value required</mat-error>\r\n } @if ($any(value).errors?.maxLength && (value.dirty || value.touched)) {\r\n <mat-error>value too long</mat-error>\r\n }\r\n </mat-form-field>\r\n }\r\n\r\n <!-- setPolicy -->\r\n <mat-form-field>\r\n <mat-label>set policy</mat-label>\r\n <mat-select [formControl]=\"setPolicy\">\r\n <mat-option [value]=\"0\">multiple</mat-option>\r\n <mat-option [value]=\"1\">single</mat-option>\r\n <mat-option [value]=\"2\">single first</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n @if (isVar) {\r\n <div class=\"form-row\">\r\n <!-- isNegated -->\r\n <mat-checkbox [formControl]=\"isNegated\">negated</mat-checkbox>\r\n\r\n <!-- isShortLived -->\r\n <mat-checkbox [formControl]=\"isShortLived\">short-lived</mat-checkbox>\r\n\r\n <!-- isGlobal -->\r\n <mat-checkbox [formControl]=\"isGlobal\">global</mat-checkbox>\r\n </div>\r\n }\r\n\r\n <!-- buttons -->\r\n <div class=\"form-row\">\r\n <button\r\n type=\"button\"\r\n color=\"warn\"\r\n mat-icon-button\r\n matTooltip=\"Discard changes\"\r\n (click)=\"cancel()\"\r\n >\r\n <mat-icon class=\"mat-warn\">clear</mat-icon>\r\n </button>\r\n <button\r\n type=\"submit\"\r\n color=\"primary\"\r\n mat-icon-button\r\n matTooltip=\"Accept changes\"\r\n [disabled]=\"form.invalid || form.pristine\"\r\n >\r\n <mat-icon class=\"mat-primary\">check_circle</mat-icon>\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 *{flex:0 0 auto}\n"] }]
184
- }], ctorParameters: () => [{ type: i1.FormBuilder }], propDecorators: { nameControl: [{
185
- type: ViewChild,
186
- args: ['nameCtl']
187
- }], featNames: [{
188
- type: Input
189
- }], featValues: [{
190
- type: Input
191
- }], feature: [{
192
- type: Input
193
- }], isVar: [{
194
- type: Input
195
- }], featureCancel: [{
196
- type: Output
197
- }], featureChange: [{
198
- type: Output
199
- }] } });
200
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"feature-editor.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/gve-core/src/lib/components/feature-editor/feature-editor.component.ts","../../../../../../../projects/myrmidon/gve-core/src/lib/components/feature-editor/feature-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  ElementRef,\r\n  EventEmitter,\r\n  Input,\r\n  OnDestroy,\r\n  OnInit,\r\n  Output,\r\n  ViewChild,\r\n} from '@angular/core';\r\nimport {\r\n  FormBuilder,\r\n  FormControl,\r\n  FormGroup,\r\n  ReactiveFormsModule,\r\n  Validators,\r\n} from '@angular/forms';\r\nimport { CommonModule } from '@angular/common';\r\nimport { Subscription, debounceTime, distinctUntilChanged } from 'rxjs';\r\n\r\n// material\r\nimport { MatButtonModule } from '@angular/material/button';\r\nimport { MatCheckboxModule } from '@angular/material/checkbox';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatIconModule } from '@angular/material/icon';\r\nimport { MatInputModule } from '@angular/material/input';\r\nimport { MatSelectModule } from '@angular/material/select';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\n\r\nimport {\r\n  Feature,\r\n  FeatureSetPolicy,\r\n  LabeledId,\r\n  OperationFeature,\r\n} from '@myrmidon/gve-snapshot-view';\r\n\r\n/**\r\n * Features map. This is used for features from closed sets,\r\n * and lists the possible values for each feature.\r\n * The key is the feature name, and the value is an array of\r\n * labeled IDs representing the allowed values for that feature.\r\n */\r\nexport interface FeatureMap {\r\n  [key: string]: LabeledId[];\r\n}\r\n\r\n/**\r\n * 🔑 `gve-feature-editor`\r\n *\r\n * Component for editing a single feature, whose model is a name=value\r\n * pair, plus a set policy value which defines the desired behavior\r\n * when adding that feature to a set.\r\n * Used by `gve-feature-set-editor`.\r\n *\r\n * - ▶️ `feature` (`Feature`): the feature to edit.\r\n * - ▶️ `featNames` (`LabeledId[]`): the list of feature names to display\r\n * in the _name_ selection. This is used when you have a closed list of\r\n * features. Each item in the list is an object with _id_ and _label_\r\n * properties.\r\n * - ▶️ `featValues` (`FeatureMap`): the feature values map. When\r\n * specified and the user selects a feature name present in the map keys,\r\n * the corresponding values will be used to populate the value selection.\r\n * - 🔥 `featureChange` (`Feature`): emitted when feature has changed.\r\n * - 🔥 `featureCancel`: emitted when the user cancels the feature editing.\r\n */\r\n@Component({\r\n  selector: 'gve-feature-editor',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    ReactiveFormsModule,\r\n    MatButtonModule,\r\n    MatCheckboxModule,\r\n    MatFormFieldModule,\r\n    MatIconModule,\r\n    MatInputModule,\r\n    MatSelectModule,\r\n    MatTooltipModule,\r\n  ],\r\n  templateUrl: './feature-editor.component.html',\r\n  styleUrl: './feature-editor.component.css',\r\n})\r\nexport class FeatureEditorComponent implements OnInit, OnDestroy {\r\n  private _sub?: Subscription;\r\n  private _feature?: OperationFeature;\r\n  private _map?: FeatureMap;\r\n  private _frozen?: boolean;\r\n\r\n  @ViewChild('nameCtl') nameControl?: ElementRef;\r\n\r\n  /**\r\n   * The list of feature names to display in the name selection.\r\n   * This is used when you have a closed list of features.\r\n   */\r\n  @Input()\r\n  public featNames: LabeledId[] | undefined;\r\n\r\n  /**\r\n   * The feature values map. When specified and the user selects a feature\r\n   * name present in the map keys, the corresponding values will be used\r\n   * to populate the value selection.\r\n   */\r\n  @Input()\r\n  public get featValues(): FeatureMap | undefined {\r\n    return this._map;\r\n  }\r\n  public set featValues(value: FeatureMap | undefined | null) {\r\n    this._map = value || undefined;\r\n    this.nameIds = this.getLabeledIdsFor(this.name.value);\r\n  }\r\n\r\n  /**\r\n   * The feature to edit.\r\n   */\r\n  @Input()\r\n  public get feature(): Feature | OperationFeature | undefined {\r\n    return this._feature;\r\n  }\r\n  public set feature(value: Feature | OperationFeature | undefined | null) {\r\n    if (this._feature === value) {\r\n      return;\r\n    }\r\n    this._feature = value || undefined;\r\n\r\n    this.updateForm(this._feature);\r\n    setTimeout(() => {\r\n      this.nameControl?.nativeElement?.focus();\r\n    }, 500);\r\n  }\r\n\r\n  /**\r\n   * True if the feature is a variant operation feature, which has\r\n   * additional properties like negation, global, and short-lived.\r\n   */\r\n  @Input()\r\n  public isVar = false;\r\n\r\n  /**\r\n   * Event emitted when the user cancels the feature editing.\r\n   */\r\n  @Output()\r\n  public featureCancel = new EventEmitter<void>();\r\n\r\n  /**\r\n   * Event emitted when the user saves the edited feature.\r\n   */\r\n  @Output()\r\n  public featureChange = new EventEmitter<Feature | OperationFeature>();\r\n\r\n  public name: FormControl<string>;\r\n  public value: FormControl<string>;\r\n  public setPolicy: FormControl<FeatureSetPolicy>;\r\n  // var feature controls\r\n  public isNegated: FormControl<boolean>;\r\n  public isGlobal: FormControl<boolean>;\r\n  public isShortLived: FormControl<boolean>;\r\n\r\n  public form: FormGroup;\r\n\r\n  public nameIds?: LabeledId[];\r\n\r\n  constructor(formBuilder: FormBuilder) {\r\n    this.name = formBuilder.control<string>('', {\r\n      validators: [Validators.required, Validators.maxLength(50)],\r\n      nonNullable: true,\r\n    });\r\n    this.value = formBuilder.control('', {\r\n      validators: [Validators.maxLength(5000)],\r\n      nonNullable: true,\r\n    });\r\n    this.setPolicy = formBuilder.control(FeatureSetPolicy.multiple, {\r\n      nonNullable: true,\r\n    });\r\n    this.isNegated = formBuilder.control(false, { nonNullable: true });\r\n    this.isGlobal = formBuilder.control(false, { nonNullable: true });\r\n    this.isShortLived = formBuilder.control(false, { nonNullable: true });\r\n    this.form = formBuilder.group({\r\n      name: this.name,\r\n      value: this.value,\r\n      setPolicy: this.setPolicy,\r\n      isNegated: this.isNegated,\r\n      isGlobal: this.isGlobal,\r\n      isShortLived: this.isShortLived,\r\n    });\r\n  }\r\n\r\n  public ngOnInit(): void {\r\n    // whenever the name changes, update the value selection\r\n    this._sub = this.name.valueChanges\r\n      .pipe(debounceTime(300), distinctUntilChanged())\r\n      .subscribe((name) => {\r\n        if (!this._frozen && this.featNames?.length) {\r\n          this.value.reset();\r\n          this.nameIds = this.getLabeledIdsFor(name);\r\n        }\r\n      });\r\n  }\r\n\r\n  public ngOnDestroy(): void {\r\n    this._sub?.unsubscribe();\r\n  }\r\n\r\n  private getLabeledIdsFor(id?: string): LabeledId[] | undefined {\r\n    if (!id) {\r\n      return undefined;\r\n    }\r\n    return this.featValues ? this.featValues[id] : undefined;\r\n  }\r\n\r\n  private updateForm(feature?: OperationFeature): void {\r\n    if (!feature) {\r\n      this.form.reset();\r\n    } else {\r\n      this._frozen = true;\r\n      this.name.setValue(feature.name);\r\n      this.value.setValue(feature.value);\r\n      this.setPolicy.setValue(feature.setPolicy || FeatureSetPolicy.multiple);\r\n      this.isNegated.setValue(feature.isNegated || false);\r\n      this.isGlobal.setValue(feature.isGlobal || false);\r\n      this.isShortLived.setValue(feature.isShortLived || false);\r\n      this._frozen = false;\r\n    }\r\n  }\r\n\r\n  public cancel(): void {\r\n    this.featureCancel.emit();\r\n  }\r\n\r\n  public save(): void {\r\n    if (!this.form.valid) {\r\n      return;\r\n    }\r\n    this.feature = this.isVar\r\n      ? {\r\n          name: this.name.value.trim(),\r\n          value: this.value.value.trim(),\r\n          setPolicy: this.setPolicy.value,\r\n          isNegated: this.isNegated.value,\r\n          isGlobal: this.isGlobal.value,\r\n          isShortLived: this.isShortLived.value,\r\n        }\r\n      : {\r\n          name: this.name.value.trim(),\r\n          value: this.value.value.trim(),\r\n          setPolicy: this.setPolicy.value,\r\n        };\r\n    this.featureChange.emit(this.feature);\r\n  }\r\n}\r\n","<form [formGroup]=\"form\" (submit)=\"save()\">\r\n  <div class=\"form-row\">\r\n    @if (featNames?.length) {\r\n    <!-- name (bound) -->\r\n    <mat-form-field>\r\n      <mat-label>name</mat-label>\r\n      <mat-select #nameCtl [formControl]=\"name\">\r\n        <mat-option *ngFor=\"let i of featNames\" [value]=\"i.id\">{{\r\n          i.label\r\n        }}</mat-option>\r\n      </mat-select>\r\n      @if ($any(name).errors?.required && (name.dirty || name.touched)) {\r\n      <mat-error>name required</mat-error>\r\n      }\r\n    </mat-form-field>\r\n    } @else {\r\n    <!-- name (free) -->\r\n    <mat-form-field>\r\n      <mat-label>name</mat-label>\r\n      <input #nameCtl matInput [formControl]=\"name\" />\r\n      <mat-error\r\n        *ngIf=\"$any(name).errors?.required && (name.dirty || name.touched)\"\r\n        >name required</mat-error\r\n      >\r\n      @if ($any(name).errors?.maxLength && (name.dirty || name.touched)) {\r\n      <mat-error>name too long</mat-error>\r\n      }\r\n    </mat-form-field>\r\n    }\r\n\r\n    <!-- value (bound) -->\r\n    @if (nameIds?.length) {\r\n    <mat-form-field>\r\n      <mat-label>value</mat-label>\r\n      <mat-select [formControl]=\"value\">\r\n        <mat-option *ngFor=\"let e of nameIds\" [value]=\"e.id\">{{\r\n          e.label\r\n        }}</mat-option>\r\n      </mat-select>\r\n      @if ($any(value).errors?.required && (value.dirty || value.touched)) {\r\n      <mat-error>value required</mat-error>\r\n      }\r\n    </mat-form-field>\r\n    } @else {\r\n    <!-- value (free) -->\r\n    <mat-form-field>\r\n      <mat-label>value</mat-label>\r\n      <input matInput [formControl]=\"value\" />\r\n      @if ($any(value).errors?.required && (value.dirty || value.touched)) {\r\n      <mat-error>value required</mat-error>\r\n      } @if ($any(value).errors?.maxLength && (value.dirty || value.touched)) {\r\n      <mat-error>value too long</mat-error>\r\n      }\r\n    </mat-form-field>\r\n    }\r\n\r\n    <!-- setPolicy -->\r\n    <mat-form-field>\r\n      <mat-label>set policy</mat-label>\r\n      <mat-select [formControl]=\"setPolicy\">\r\n        <mat-option [value]=\"0\">multiple</mat-option>\r\n        <mat-option [value]=\"1\">single</mat-option>\r\n        <mat-option [value]=\"2\">single first</mat-option>\r\n      </mat-select>\r\n    </mat-form-field>\r\n  </div>\r\n\r\n  @if (isVar) {\r\n  <div class=\"form-row\">\r\n    <!-- isNegated -->\r\n    <mat-checkbox [formControl]=\"isNegated\">negated</mat-checkbox>\r\n\r\n    <!-- isShortLived -->\r\n    <mat-checkbox [formControl]=\"isShortLived\">short-lived</mat-checkbox>\r\n\r\n    <!-- isGlobal -->\r\n    <mat-checkbox [formControl]=\"isGlobal\">global</mat-checkbox>\r\n  </div>\r\n  }\r\n\r\n  <!-- buttons -->\r\n  <div class=\"form-row\">\r\n    <button\r\n      type=\"button\"\r\n      color=\"warn\"\r\n      mat-icon-button\r\n      matTooltip=\"Discard changes\"\r\n      (click)=\"cancel()\"\r\n    >\r\n      <mat-icon class=\"mat-warn\">clear</mat-icon>\r\n    </button>\r\n    <button\r\n      type=\"submit\"\r\n      color=\"primary\"\r\n      mat-icon-button\r\n      matTooltip=\"Accept changes\"\r\n      [disabled]=\"form.invalid || form.pristine\"\r\n    >\r\n      <mat-icon class=\"mat-primary\">check_circle</mat-icon>\r\n    </button>\r\n  </div>\r\n</form>\r\n"]}