@myrmidon/gve-core 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +83 -0
- package/esm2022/lib/components/animation-timeline/animation-timeline.component.mjs +176 -0
- package/esm2022/lib/components/animation-tween/animation-tween.component.mjs +161 -0
- package/esm2022/lib/components/animation-vars/animation-vars.component.mjs +141 -0
- package/esm2022/lib/components/base-text-char/base-text-char.component.mjs +37 -0
- package/esm2022/lib/components/base-text-editor/base-text-editor.component.mjs +111 -0
- package/esm2022/lib/components/base-text-view/base-text-view.component.mjs +125 -0
- package/esm2022/lib/components/chain-operation-editor/chain-operation-editor.component.mjs +512 -0
- package/esm2022/lib/components/chain-result-view/chain-result-view.component.mjs +190 -0
- package/esm2022/lib/components/feature-editor/feature-editor.component.mjs +185 -0
- package/esm2022/lib/components/feature-set-editor/feature-set-editor.component.mjs +152 -0
- package/esm2022/lib/components/feature-set-view/feature-set-view.component.mjs +80 -0
- package/esm2022/lib/components/ln-heights-editor/ln-heights-editor.component.mjs +95 -0
- package/esm2022/lib/components/operation-source-editor/operation-source-editor.component.mjs +106 -0
- package/esm2022/lib/components/simple-tree/simple-tree.component.mjs +56 -0
- package/esm2022/lib/components/snapshot-editor/snapshot-editor.component.mjs +510 -0
- package/esm2022/lib/components/steps-map/steps-map.component.mjs +55 -0
- package/esm2022/lib/models.mjs +2 -0
- package/esm2022/lib/services/gve-api.service.mjs +52 -0
- package/esm2022/lib/services/settings.service.mjs +86 -0
- package/esm2022/lib/validators/svg-validators.mjs +28 -0
- package/esm2022/myrmidon-gve-core.mjs +5 -0
- package/esm2022/public-api.mjs +23 -0
- package/fesm2022/myrmidon-gve-core.mjs +2592 -0
- package/fesm2022/myrmidon-gve-core.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/components/animation-timeline/animation-timeline.component.d.ts +42 -0
- package/lib/components/animation-tween/animation-tween.component.d.ts +47 -0
- package/lib/components/animation-vars/animation-vars.component.d.ts +30 -0
- package/lib/components/base-text-char/base-text-char.component.d.ts +29 -0
- package/lib/components/base-text-editor/base-text-editor.component.d.ts +40 -0
- package/lib/components/base-text-view/base-text-view.component.d.ts +29 -0
- package/lib/components/chain-operation-editor/chain-operation-editor.component.d.ts +102 -0
- package/lib/components/chain-result-view/chain-result-view.component.d.ts +44 -0
- package/lib/components/feature-editor/feature-editor.component.d.ts +72 -0
- package/lib/components/feature-set-editor/feature-set-editor.component.d.ts +55 -0
- package/lib/components/feature-set-view/feature-set-view.component.d.ts +36 -0
- package/lib/components/ln-heights-editor/ln-heights-editor.component.d.ts +22 -0
- package/lib/components/operation-source-editor/operation-source-editor.component.d.ts +24 -0
- package/lib/components/simple-tree/simple-tree.component.d.ts +16 -0
- package/lib/components/snapshot-editor/snapshot-editor.component.d.ts +105 -0
- package/lib/components/steps-map/steps-map.component.d.ts +29 -0
- package/lib/models.d.ts +7 -0
- package/lib/services/gve-api.service.d.ts +57 -0
- package/lib/services/settings.service.d.ts +53 -0
- package/lib/validators/svg-validators.d.ts +4 -0
- package/package.json +45 -0
- package/public-api.d.ts +19 -0
|
@@ -0,0 +1,190 @@
|
|
|
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
|
+
* Component to display a chain result.
|
|
20
|
+
*/
|
|
21
|
+
export class ChainResultViewComponent {
|
|
22
|
+
/**
|
|
23
|
+
* The result to display.
|
|
24
|
+
*/
|
|
25
|
+
get result() {
|
|
26
|
+
return this._result;
|
|
27
|
+
}
|
|
28
|
+
set result(value) {
|
|
29
|
+
if (this._result === value) {
|
|
30
|
+
return;
|
|
31
|
+
}
|
|
32
|
+
this._result = value || undefined;
|
|
33
|
+
this.updateForm(this._result);
|
|
34
|
+
// select last step by default
|
|
35
|
+
if (this._result?.steps?.length) {
|
|
36
|
+
this._stepPickFrozen = true;
|
|
37
|
+
this.step = this._result.steps[this._result.steps.length - 1];
|
|
38
|
+
this.tag.setValue(this.step.outputTag);
|
|
39
|
+
this._stepPickFrozen = false;
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
constructor(formBuilder) {
|
|
43
|
+
/**
|
|
44
|
+
* Emits when a result's step is picked.
|
|
45
|
+
*/
|
|
46
|
+
this.stepPick = new EventEmitter();
|
|
47
|
+
this.versionTags = [];
|
|
48
|
+
this.tags = [];
|
|
49
|
+
this.selectionFeatures = [];
|
|
50
|
+
this.versionTag = formBuilder.control(null);
|
|
51
|
+
this.tag = formBuilder.control(null);
|
|
52
|
+
}
|
|
53
|
+
ngOnInit() {
|
|
54
|
+
this._subs = [
|
|
55
|
+
this.versionTag.valueChanges
|
|
56
|
+
.pipe(distinctUntilChanged(), debounceTime(200))
|
|
57
|
+
.subscribe((value) => {
|
|
58
|
+
this.tag.setValue(this.getTagFromVersion(value) || null);
|
|
59
|
+
}),
|
|
60
|
+
this.tag.valueChanges
|
|
61
|
+
.pipe(distinctUntilChanged(), debounceTime(200))
|
|
62
|
+
.subscribe((value) => {
|
|
63
|
+
this.step = this._result?.steps.find((step) => step.outputTag === value);
|
|
64
|
+
if (this.step && !this._stepPickFrozen) {
|
|
65
|
+
this.stepPick.emit(this.step);
|
|
66
|
+
}
|
|
67
|
+
}),
|
|
68
|
+
];
|
|
69
|
+
}
|
|
70
|
+
ngOnDestroy() {
|
|
71
|
+
this._subs?.forEach((sub) => sub.unsubscribe());
|
|
72
|
+
}
|
|
73
|
+
updateVersionTags() {
|
|
74
|
+
if (!this._result) {
|
|
75
|
+
this.versionTags = [];
|
|
76
|
+
return;
|
|
77
|
+
}
|
|
78
|
+
// extract version tags from result steps features named 'version'
|
|
79
|
+
const tags = new Set();
|
|
80
|
+
for (const step of this._result.steps) {
|
|
81
|
+
for (const feature of step.featureSet.features) {
|
|
82
|
+
if (feature.name === 'version') {
|
|
83
|
+
tags.add(feature.value);
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
this.versionTags = Array.from(tags).sort();
|
|
88
|
+
}
|
|
89
|
+
getTagFromVersion(version) {
|
|
90
|
+
if (!this._result || !version) {
|
|
91
|
+
return undefined;
|
|
92
|
+
}
|
|
93
|
+
for (const step of this._result.steps) {
|
|
94
|
+
for (const feature of step.featureSet.features) {
|
|
95
|
+
if (feature.name === 'version' && feature.value === version) {
|
|
96
|
+
return step.outputTag;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
return undefined;
|
|
101
|
+
}
|
|
102
|
+
updateForm(result) {
|
|
103
|
+
this.selectionFeatures = [];
|
|
104
|
+
if (!result) {
|
|
105
|
+
this.step = undefined;
|
|
106
|
+
this.selection = undefined;
|
|
107
|
+
this.tags = [];
|
|
108
|
+
this.versionTags = [];
|
|
109
|
+
}
|
|
110
|
+
else {
|
|
111
|
+
this.tags = result.chainTags;
|
|
112
|
+
this.updateVersionTags();
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
getNodeFeatures(id) {
|
|
116
|
+
if (!this.step) {
|
|
117
|
+
return [];
|
|
118
|
+
}
|
|
119
|
+
return (this.step.featureSet.nodeFeatures[this.step.outputTag + '_' + id] || []);
|
|
120
|
+
}
|
|
121
|
+
findNode(id) {
|
|
122
|
+
if (!this.step) {
|
|
123
|
+
return null;
|
|
124
|
+
}
|
|
125
|
+
for (let key of Object.keys(this._result.taggedNodes)) {
|
|
126
|
+
const node = this._result.taggedNodes[key].find((n) => n.id === id);
|
|
127
|
+
if (node) {
|
|
128
|
+
return node;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
return null;
|
|
132
|
+
}
|
|
133
|
+
onStepCharPick(event) {
|
|
134
|
+
const features = this.getNodeFeatures(event.char.id);
|
|
135
|
+
const node = this.findNode(event.char.id);
|
|
136
|
+
this.selection = `${event.char.label} (${node.sourceTag})`;
|
|
137
|
+
this.selectionFeatures = [...features];
|
|
138
|
+
}
|
|
139
|
+
onStepRangePick(range) {
|
|
140
|
+
if (range.run === 1) {
|
|
141
|
+
// already handled by onStepCharPick
|
|
142
|
+
return;
|
|
143
|
+
}
|
|
144
|
+
this.selection = range.at + '×' + range.run;
|
|
145
|
+
const end = range.at + range.run;
|
|
146
|
+
const features = [];
|
|
147
|
+
const tagNodes = this._result.taggedNodes[this.tag.value];
|
|
148
|
+
for (let node of tagNodes) {
|
|
149
|
+
// if in range, intersect features
|
|
150
|
+
if (node.id >= range.at && node.id < end) {
|
|
151
|
+
const nodeFeatures = this.getNodeFeatures(node.id);
|
|
152
|
+
if (node.id === range.at) {
|
|
153
|
+
for (let f of nodeFeatures) {
|
|
154
|
+
features.push(f);
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
else {
|
|
158
|
+
// features = intersection of features and charFeatures
|
|
159
|
+
features.splice(0, features.length, ...features.filter((f) => nodeFeatures.some((cf) => cf.name === f.name)));
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
this.selectionFeatures = features;
|
|
164
|
+
}
|
|
165
|
+
onStepChange(step) {
|
|
166
|
+
this.tag.setValue(step.outputTag);
|
|
167
|
+
this.selectionFeatures = [];
|
|
168
|
+
}
|
|
169
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: ChainResultViewComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
170
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.4", type: ChainResultViewComponent, isStandalone: true, selector: "gve-chain-result-view", inputs: { result: "result" }, 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", "minFilterCount"] }, { 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"] }] }); }
|
|
171
|
+
}
|
|
172
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: ChainResultViewComponent, decorators: [{
|
|
173
|
+
type: Component,
|
|
174
|
+
args: [{ selector: 'gve-chain-result-view', standalone: true, imports: [
|
|
175
|
+
CommonModule,
|
|
176
|
+
ReactiveFormsModule,
|
|
177
|
+
MatButtonModule,
|
|
178
|
+
MatFormFieldModule,
|
|
179
|
+
MatIconModule,
|
|
180
|
+
MatSelectModule,
|
|
181
|
+
FeatureSetViewComponent,
|
|
182
|
+
StepsMapComponent,
|
|
183
|
+
BaseTextViewComponent,
|
|
184
|
+
], 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"] }]
|
|
185
|
+
}], ctorParameters: () => [{ type: i1.FormBuilder }], propDecorators: { result: [{
|
|
186
|
+
type: Input
|
|
187
|
+
}], stepPick: [{
|
|
188
|
+
type: Output
|
|
189
|
+
}] } });
|
|
190
|
+
//# 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 * Component to display a chain result.\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 _stepPickFrozen?: boolean;\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 last step by default\r\n    if (this._result?.steps?.length) {\r\n      this._stepPickFrozen = true;\r\n      this.step = this._result.steps[this._result.steps.length - 1];\r\n      this.tag.setValue(this.step.outputTag);\r\n      this._stepPickFrozen = false;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Emits when a result's step is picked.\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      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      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 && !this._stepPickFrozen) {\r\n            this.stepPick.emit(this.step!);\r\n          }\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    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"]}
|
|
@@ -0,0 +1,185 @@
|
|
|
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
|
+
* Feature editor. This edits a single feature, either free or from
|
|
27
|
+
* a closed set. In the latter case you must define the set via
|
|
28
|
+
* the `labeledIds` input. Also, you can provide a map of features
|
|
29
|
+
* to values via the `map` input. This is useful when you want to
|
|
30
|
+
* provide a set of features and values to the user, and you want
|
|
31
|
+
* to provide human-readable labels for the values.
|
|
32
|
+
*/
|
|
33
|
+
export class FeatureEditorComponent {
|
|
34
|
+
/**
|
|
35
|
+
* The feature values map. When specified and the user selects a feature
|
|
36
|
+
* name present in the map keys, the corresponding values will be used
|
|
37
|
+
* to populate the value selection.
|
|
38
|
+
*/
|
|
39
|
+
get featValues() {
|
|
40
|
+
return this._map;
|
|
41
|
+
}
|
|
42
|
+
set featValues(value) {
|
|
43
|
+
this._map = value || undefined;
|
|
44
|
+
this.nameIds = this.getLabeledIdsFor(this.name.value);
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* The feature to edit.
|
|
48
|
+
*/
|
|
49
|
+
get feature() {
|
|
50
|
+
return this._feature;
|
|
51
|
+
}
|
|
52
|
+
set feature(value) {
|
|
53
|
+
if (this._feature === value) {
|
|
54
|
+
return;
|
|
55
|
+
}
|
|
56
|
+
this._feature = value || undefined;
|
|
57
|
+
this.updateForm(this._feature);
|
|
58
|
+
setTimeout(() => {
|
|
59
|
+
this.nameControl?.nativeElement?.focus();
|
|
60
|
+
}, 500);
|
|
61
|
+
}
|
|
62
|
+
constructor(formBuilder) {
|
|
63
|
+
/**
|
|
64
|
+
* True if the feature is a variant operation feature, which has
|
|
65
|
+
* additional properties like negation, global, and short-lived.
|
|
66
|
+
*/
|
|
67
|
+
this.isVar = false;
|
|
68
|
+
/**
|
|
69
|
+
* Event emitted when the user cancels the feature editing.
|
|
70
|
+
*/
|
|
71
|
+
this.featureCancel = new EventEmitter();
|
|
72
|
+
/**
|
|
73
|
+
* Event emitted when the user saves the edited feature.
|
|
74
|
+
*/
|
|
75
|
+
this.featureChange = new EventEmitter();
|
|
76
|
+
this.name = formBuilder.control('', {
|
|
77
|
+
validators: [Validators.required, Validators.maxLength(50)],
|
|
78
|
+
nonNullable: true,
|
|
79
|
+
});
|
|
80
|
+
this.value = formBuilder.control('', {
|
|
81
|
+
validators: [Validators.maxLength(5000)],
|
|
82
|
+
nonNullable: true,
|
|
83
|
+
});
|
|
84
|
+
this.setPolicy = formBuilder.control(FeatureSetPolicy.multiple, {
|
|
85
|
+
nonNullable: true,
|
|
86
|
+
});
|
|
87
|
+
this.isNegated = formBuilder.control(false, { nonNullable: true });
|
|
88
|
+
this.isGlobal = formBuilder.control(false, { nonNullable: true });
|
|
89
|
+
this.isShortLived = formBuilder.control(false, { nonNullable: true });
|
|
90
|
+
this.form = formBuilder.group({
|
|
91
|
+
name: this.name,
|
|
92
|
+
value: this.value,
|
|
93
|
+
setPolicy: this.setPolicy,
|
|
94
|
+
isNegated: this.isNegated,
|
|
95
|
+
isGlobal: this.isGlobal,
|
|
96
|
+
isShortLived: this.isShortLived,
|
|
97
|
+
});
|
|
98
|
+
}
|
|
99
|
+
ngOnInit() {
|
|
100
|
+
// whenever the name changes, update the value selection
|
|
101
|
+
this._sub = this.name.valueChanges
|
|
102
|
+
.pipe(debounceTime(300), distinctUntilChanged())
|
|
103
|
+
.subscribe((name) => {
|
|
104
|
+
this.value.reset();
|
|
105
|
+
this.nameIds = this.getLabeledIdsFor(name);
|
|
106
|
+
});
|
|
107
|
+
}
|
|
108
|
+
ngOnDestroy() {
|
|
109
|
+
this._sub?.unsubscribe();
|
|
110
|
+
}
|
|
111
|
+
getLabeledIdsFor(id) {
|
|
112
|
+
if (!id) {
|
|
113
|
+
return undefined;
|
|
114
|
+
}
|
|
115
|
+
return this.featValues ? this.featValues[id] : undefined;
|
|
116
|
+
}
|
|
117
|
+
updateForm(feature) {
|
|
118
|
+
if (!feature) {
|
|
119
|
+
this.form.reset();
|
|
120
|
+
}
|
|
121
|
+
else {
|
|
122
|
+
this.name.setValue(feature.name);
|
|
123
|
+
this.value.setValue(feature.value);
|
|
124
|
+
this.setPolicy.setValue(feature.setPolicy || FeatureSetPolicy.multiple);
|
|
125
|
+
this.isNegated.setValue(feature.isNegated || false);
|
|
126
|
+
this.isGlobal.setValue(feature.isGlobal || false);
|
|
127
|
+
this.isShortLived.setValue(feature.isShortLived || false);
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
cancel() {
|
|
131
|
+
this.featureCancel.emit();
|
|
132
|
+
}
|
|
133
|
+
save() {
|
|
134
|
+
if (!this.form.valid) {
|
|
135
|
+
return;
|
|
136
|
+
}
|
|
137
|
+
this.feature = this.isVar
|
|
138
|
+
? {
|
|
139
|
+
name: this.name.value.trim(),
|
|
140
|
+
value: this.value.value.trim(),
|
|
141
|
+
setPolicy: this.setPolicy.value,
|
|
142
|
+
isNegated: this.isNegated.value,
|
|
143
|
+
isGlobal: this.isGlobal.value,
|
|
144
|
+
isShortLived: this.isShortLived.value,
|
|
145
|
+
}
|
|
146
|
+
: {
|
|
147
|
+
name: this.name.value.trim(),
|
|
148
|
+
value: this.value.value.trim(),
|
|
149
|
+
setPolicy: this.setPolicy.value,
|
|
150
|
+
};
|
|
151
|
+
this.featureChange.emit(this.feature);
|
|
152
|
+
}
|
|
153
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: FeatureEditorComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
154
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.4", 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", "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"] }] }); }
|
|
155
|
+
}
|
|
156
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: FeatureEditorComponent, decorators: [{
|
|
157
|
+
type: Component,
|
|
158
|
+
args: [{ selector: 'gve-feature-editor', standalone: true, imports: [
|
|
159
|
+
CommonModule,
|
|
160
|
+
ReactiveFormsModule,
|
|
161
|
+
MatButtonModule,
|
|
162
|
+
MatCheckboxModule,
|
|
163
|
+
MatFormFieldModule,
|
|
164
|
+
MatIconModule,
|
|
165
|
+
MatInputModule,
|
|
166
|
+
MatSelectModule,
|
|
167
|
+
MatTooltipModule,
|
|
168
|
+
], 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"] }]
|
|
169
|
+
}], ctorParameters: () => [{ type: i1.FormBuilder }], propDecorators: { nameControl: [{
|
|
170
|
+
type: ViewChild,
|
|
171
|
+
args: ['nameCtl']
|
|
172
|
+
}], featNames: [{
|
|
173
|
+
type: Input
|
|
174
|
+
}], featValues: [{
|
|
175
|
+
type: Input
|
|
176
|
+
}], feature: [{
|
|
177
|
+
type: Input
|
|
178
|
+
}], isVar: [{
|
|
179
|
+
type: Input
|
|
180
|
+
}], featureCancel: [{
|
|
181
|
+
type: Output
|
|
182
|
+
}], featureChange: [{
|
|
183
|
+
type: Output
|
|
184
|
+
}] } });
|
|
185
|
+
//# 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 */\r\nexport interface FeatureMap {\r\n  [key: string]: LabeledId[];\r\n}\r\n\r\n/**\r\n * Feature editor. This edits a single feature, either free or from\r\n * a closed set. In the latter case you must define the set via\r\n * the `labeledIds` input. Also, you can provide a map of features\r\n * to values via the `map` input. This is useful when you want to\r\n * provide a set of features and values to the user, and you want\r\n * to provide human-readable labels for the values.\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\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        this.value.reset();\r\n        this.nameIds = this.getLabeledIdsFor(name);\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.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    }\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"]}
|