@myrmidon/gve-core 0.0.6 → 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/fesm2022/myrmidon-gve-core.mjs +134 -136
- package/fesm2022/myrmidon-gve-core.mjs.map +1 -1
- package/lib/components/feature-editor/feature-editor.component.d.ts +1 -1
- package/package.json +6 -8
- package/esm2022/lib/components/animation-timeline/animation-timeline.component.mjs +0 -207
- package/esm2022/lib/components/animation-timeline-set/animation-timeline-set.component.mjs +0 -154
- package/esm2022/lib/components/animation-tween/animation-tween.component.mjs +0 -184
- package/esm2022/lib/components/base-text-char/base-text-char.component.mjs +0 -46
- package/esm2022/lib/components/base-text-editor/base-text-editor.component.mjs +0 -115
- package/esm2022/lib/components/base-text-view/base-text-view.component.mjs +0 -159
- package/esm2022/lib/components/batch-operation-editor/batch-operation-editor.component.mjs +0 -111
- package/esm2022/lib/components/chain-operation-editor/chain-operation-editor.component.mjs +0 -570
- package/esm2022/lib/components/chain-result-view/chain-result-view.component.mjs +0 -225
- package/esm2022/lib/components/feature-editor/feature-editor.component.mjs +0 -200
- package/esm2022/lib/components/feature-set-editor/feature-set-editor.component.mjs +0 -175
- package/esm2022/lib/components/feature-set-view/feature-set-view.component.mjs +0 -100
- package/esm2022/lib/components/ln-heights-editor/ln-heights-editor.component.mjs +0 -126
- package/esm2022/lib/components/operation-source-editor/operation-source-editor.component.mjs +0 -131
- package/esm2022/lib/components/simple-tree/simple-tree.component.mjs +0 -72
- package/esm2022/lib/components/snapshot-editor/snapshot-editor.component.mjs +0 -870
- package/esm2022/lib/components/steps-map/steps-map.component.mjs +0 -83
- package/esm2022/lib/models.mjs +0 -2
- package/esm2022/lib/services/gve-api.service.mjs +0 -65
- package/esm2022/lib/services/settings.service.mjs +0 -87
- package/esm2022/lib/validators/svg-validators.mjs +0 -34
- package/esm2022/myrmidon-gve-core.mjs +0 -5
- package/esm2022/public-api.mjs +0 -23
|
@@ -1,225 +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. This provides a version picker
|
|
22
|
-
* by staged version or simple step, and shows the selected step's text
|
|
23
|
-
* and features, plus a steps map. User can pick a step from both the
|
|
24
|
-
* pickers or the map.
|
|
25
|
-
* Used by the `gve-snapshot-editor` component.
|
|
26
|
-
*
|
|
27
|
-
* - ▶️ `result` (`CharChainResult`): the result to display.
|
|
28
|
-
* - 🔥 `stepPick` (`ChainOperationContextStep`): emitted when a
|
|
29
|
-
* result's step is picked by the user.
|
|
30
|
-
*/
|
|
31
|
-
export class ChainResultViewComponent {
|
|
32
|
-
/**
|
|
33
|
-
* The result to display.
|
|
34
|
-
*/
|
|
35
|
-
get result() {
|
|
36
|
-
return this._result;
|
|
37
|
-
}
|
|
38
|
-
set result(value) {
|
|
39
|
-
if (this._result === value) {
|
|
40
|
-
return;
|
|
41
|
-
}
|
|
42
|
-
this._result = value || undefined;
|
|
43
|
-
this.updateForm(this._result);
|
|
44
|
-
// select the initial step if set
|
|
45
|
-
this.selectInitialStep();
|
|
46
|
-
}
|
|
47
|
-
/**
|
|
48
|
-
* The index of the initial step to display after the result is set.
|
|
49
|
-
* If the index is negative, it is counted from the end of the steps.
|
|
50
|
-
*/
|
|
51
|
-
get initialStepIndex() {
|
|
52
|
-
return this._initialStepIndex;
|
|
53
|
-
}
|
|
54
|
-
set initialStepIndex(value) {
|
|
55
|
-
if (value === undefined || value === null) {
|
|
56
|
-
this._initialStepIndex = undefined;
|
|
57
|
-
this.step = undefined;
|
|
58
|
-
}
|
|
59
|
-
else {
|
|
60
|
-
this._initialStepIndex = value;
|
|
61
|
-
this.selectInitialStep();
|
|
62
|
-
}
|
|
63
|
-
}
|
|
64
|
-
constructor(formBuilder) {
|
|
65
|
-
/**
|
|
66
|
-
* Emitted when a result's step is picked by the user.
|
|
67
|
-
*/
|
|
68
|
-
this.stepPick = new EventEmitter();
|
|
69
|
-
this.versionTags = [];
|
|
70
|
-
this.tags = [];
|
|
71
|
-
this.selectionFeatures = [];
|
|
72
|
-
this.versionTag = formBuilder.control(null);
|
|
73
|
-
this.tag = formBuilder.control(null);
|
|
74
|
-
}
|
|
75
|
-
ngOnInit() {
|
|
76
|
-
this._subs = [
|
|
77
|
-
// whenever the staged version tag changes, update the form
|
|
78
|
-
this.versionTag.valueChanges
|
|
79
|
-
.pipe(distinctUntilChanged(), debounceTime(200))
|
|
80
|
-
.subscribe((value) => {
|
|
81
|
-
this.tag.setValue(this.getTagFromVersion(value) || null);
|
|
82
|
-
}),
|
|
83
|
-
// whenever the tag changes, update the step
|
|
84
|
-
this.tag.valueChanges
|
|
85
|
-
.pipe(distinctUntilChanged(), debounceTime(200))
|
|
86
|
-
.subscribe((value) => {
|
|
87
|
-
this.step = this._result?.steps.find((step) => step.outputTag === value);
|
|
88
|
-
if (this.step) {
|
|
89
|
-
this.stepPick.emit(this.step);
|
|
90
|
-
}
|
|
91
|
-
}),
|
|
92
|
-
];
|
|
93
|
-
}
|
|
94
|
-
selectInitialStep() {
|
|
95
|
-
if (this._initialStepIndex !== undefined && this._result?.steps) {
|
|
96
|
-
this.step =
|
|
97
|
-
this._result.steps[this._initialStepIndex < 0
|
|
98
|
-
? this._result.steps.length + this._initialStepIndex
|
|
99
|
-
: this._initialStepIndex];
|
|
100
|
-
}
|
|
101
|
-
}
|
|
102
|
-
ngOnDestroy() {
|
|
103
|
-
this._subs?.forEach((sub) => sub.unsubscribe());
|
|
104
|
-
}
|
|
105
|
-
updateVersionTags() {
|
|
106
|
-
if (!this._result) {
|
|
107
|
-
this.versionTags = [];
|
|
108
|
-
return;
|
|
109
|
-
}
|
|
110
|
-
// extract version tags from result steps features named 'version'
|
|
111
|
-
const tags = new Set();
|
|
112
|
-
for (const step of this._result.steps) {
|
|
113
|
-
for (const feature of step.featureSet.features) {
|
|
114
|
-
if (feature.name === 'version') {
|
|
115
|
-
tags.add(feature.value);
|
|
116
|
-
}
|
|
117
|
-
}
|
|
118
|
-
}
|
|
119
|
-
this.versionTags = Array.from(tags).sort();
|
|
120
|
-
}
|
|
121
|
-
getTagFromVersion(version) {
|
|
122
|
-
if (!this._result || !version) {
|
|
123
|
-
return undefined;
|
|
124
|
-
}
|
|
125
|
-
for (const step of this._result.steps) {
|
|
126
|
-
for (const feature of step.featureSet.features) {
|
|
127
|
-
if (feature.name === 'version' && feature.value === version) {
|
|
128
|
-
return step.outputTag;
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
}
|
|
132
|
-
return undefined;
|
|
133
|
-
}
|
|
134
|
-
updateForm(result) {
|
|
135
|
-
this.selectionFeatures = [];
|
|
136
|
-
if (!result) {
|
|
137
|
-
this.step = undefined;
|
|
138
|
-
this.selection = undefined;
|
|
139
|
-
this.tags = [];
|
|
140
|
-
this.versionTags = [];
|
|
141
|
-
}
|
|
142
|
-
else {
|
|
143
|
-
this.tags = result.chainTags;
|
|
144
|
-
this.updateVersionTags();
|
|
145
|
-
}
|
|
146
|
-
}
|
|
147
|
-
getNodeFeatures(id) {
|
|
148
|
-
if (!this.step) {
|
|
149
|
-
return [];
|
|
150
|
-
}
|
|
151
|
-
return (this.step.featureSet.nodeFeatures[this.step.outputTag + '_' + id] || []);
|
|
152
|
-
}
|
|
153
|
-
findNode(id) {
|
|
154
|
-
if (!this.step) {
|
|
155
|
-
return null;
|
|
156
|
-
}
|
|
157
|
-
for (let key of Object.keys(this._result.taggedNodes)) {
|
|
158
|
-
const node = this._result.taggedNodes[key].find((n) => n.id === id);
|
|
159
|
-
if (node) {
|
|
160
|
-
return node;
|
|
161
|
-
}
|
|
162
|
-
}
|
|
163
|
-
return null;
|
|
164
|
-
}
|
|
165
|
-
onStepCharPick(event) {
|
|
166
|
-
const features = this.getNodeFeatures(event.char.id);
|
|
167
|
-
const node = this.findNode(event.char.id);
|
|
168
|
-
this.selection = `${event.char.label} (${node.sourceTag})`;
|
|
169
|
-
this.selectionFeatures = [...features];
|
|
170
|
-
}
|
|
171
|
-
onStepRangePick(range) {
|
|
172
|
-
if (range.run === 1) {
|
|
173
|
-
// already handled by onStepCharPick
|
|
174
|
-
return;
|
|
175
|
-
}
|
|
176
|
-
this.selection = range.at + '×' + range.run;
|
|
177
|
-
const end = range.at + range.run;
|
|
178
|
-
const features = [];
|
|
179
|
-
const tagNodes = this._result.taggedNodes[this.tag.value];
|
|
180
|
-
for (let node of tagNodes) {
|
|
181
|
-
// if in range, intersect features
|
|
182
|
-
if (node.id >= range.at && node.id < end) {
|
|
183
|
-
const nodeFeatures = this.getNodeFeatures(node.id);
|
|
184
|
-
if (node.id === range.at) {
|
|
185
|
-
for (let f of nodeFeatures) {
|
|
186
|
-
features.push(f);
|
|
187
|
-
}
|
|
188
|
-
}
|
|
189
|
-
else {
|
|
190
|
-
// features = intersection of features and charFeatures
|
|
191
|
-
features.splice(0, features.length, ...features.filter((f) => nodeFeatures.some((cf) => cf.name === f.name)));
|
|
192
|
-
}
|
|
193
|
-
}
|
|
194
|
-
}
|
|
195
|
-
this.selectionFeatures = features;
|
|
196
|
-
}
|
|
197
|
-
onStepChange(step) {
|
|
198
|
-
// setting the tag will trigger the step change
|
|
199
|
-
this.tag.setValue(step.outputTag);
|
|
200
|
-
this.selectionFeatures = [];
|
|
201
|
-
}
|
|
202
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: ChainResultViewComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
203
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.12", 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"] }] }); }
|
|
204
|
-
}
|
|
205
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: ChainResultViewComponent, decorators: [{
|
|
206
|
-
type: Component,
|
|
207
|
-
args: [{ selector: 'gve-chain-result-view', standalone: true, imports: [
|
|
208
|
-
CommonModule,
|
|
209
|
-
ReactiveFormsModule,
|
|
210
|
-
MatButtonModule,
|
|
211
|
-
MatFormFieldModule,
|
|
212
|
-
MatIconModule,
|
|
213
|
-
MatSelectModule,
|
|
214
|
-
FeatureSetViewComponent,
|
|
215
|
-
StepsMapComponent,
|
|
216
|
-
BaseTextViewComponent,
|
|
217
|
-
], 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"] }]
|
|
218
|
-
}], ctorParameters: () => [{ type: i1.FormBuilder }], propDecorators: { result: [{
|
|
219
|
-
type: Input
|
|
220
|
-
}], initialStepIndex: [{
|
|
221
|
-
type: Input
|
|
222
|
-
}], stepPick: [{
|
|
223
|
-
type: Output
|
|
224
|
-
}] } });
|
|
225
|
-
//# 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. This provides a version picker\r\n * by staged version or simple step, and shows the selected step's text\r\n * and features, plus a steps map. User can pick a step from both the\r\n * pickers or the map.\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.12", 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.12", 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.12", 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"]}
|