@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,175 +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 { MatButtonModule } from '@angular/material/button';
|
|
5
|
-
import { MatExpansionModule } from '@angular/material/expansion';
|
|
6
|
-
import { MatFormFieldModule } from '@angular/material/form-field';
|
|
7
|
-
import { MatIconModule } from '@angular/material/icon';
|
|
8
|
-
import { MatInputModule } from '@angular/material/input';
|
|
9
|
-
import { MatSelectModule } from '@angular/material/select';
|
|
10
|
-
import { MatTooltipModule } from '@angular/material/tooltip';
|
|
11
|
-
import { debounceTime, distinctUntilChanged } from 'rxjs';
|
|
12
|
-
import { NgToolsModule } from '@myrmidon/ng-tools';
|
|
13
|
-
import { FeatureSetPolicy, } from '@myrmidon/gve-snapshot-view';
|
|
14
|
-
import { FeatureEditorComponent, } from '../feature-editor/feature-editor.component';
|
|
15
|
-
import * as i0 from "@angular/core";
|
|
16
|
-
import * as i1 from "@angular/forms";
|
|
17
|
-
import * as i2 from "@angular/material/button";
|
|
18
|
-
import * as i3 from "@angular/material/expansion";
|
|
19
|
-
import * as i4 from "@angular/material/form-field";
|
|
20
|
-
import * as i5 from "@angular/material/icon";
|
|
21
|
-
import * as i6 from "@angular/material/input";
|
|
22
|
-
import * as i7 from "@angular/material/tooltip";
|
|
23
|
-
import * as i8 from "@myrmidon/ng-tools";
|
|
24
|
-
/**
|
|
25
|
-
* 🔑 `gve-feature-set-editor`
|
|
26
|
-
*
|
|
27
|
-
* Component for editing a set of features. It shows a list of features,
|
|
28
|
-
* allowing to edit each of them or adding new ones. Optionally you can
|
|
29
|
-
* specify a list of feature names and values to use in the selection.
|
|
30
|
-
* Used by base text editor and chain operation editor components.
|
|
31
|
-
*
|
|
32
|
-
* - ▶️ `featNames` (`LabeledId[]`): the list of feature names to display
|
|
33
|
-
* in the _name_ selection. This is used when you have a closed list of features.
|
|
34
|
-
* - ▶️ `featValues` (`FeatureMap`): the feature values map. When specified
|
|
35
|
-
* and the user selects a feature name present in the map keys, the corresponding
|
|
36
|
-
* values will be used to populate the value selection.
|
|
37
|
-
* - ▶️ `filterThreshold` (`number`): the threshold at which the features filter
|
|
38
|
-
* should become visible. If set to 0, the filter is always visible; if set to -1,
|
|
39
|
-
* it is always invisible; otherwise, it gets visible when the number of features
|
|
40
|
-
* is greater than the threshold. Default is 5.
|
|
41
|
-
* - ▶️ `features` (`Feature[]`): the features to edit.
|
|
42
|
-
* - ▶️ `isVar`: true if the feature is a variant operation feature, which
|
|
43
|
-
* has additional properties like negation, global, and short-lived.
|
|
44
|
-
* - 🔥 `featuresChange` (`Feature[]`): emitted when features have changed.
|
|
45
|
-
*/
|
|
46
|
-
export class FeatureSetEditorComponent {
|
|
47
|
-
/**
|
|
48
|
-
* The features to edit.
|
|
49
|
-
*/
|
|
50
|
-
get features() {
|
|
51
|
-
return this._features;
|
|
52
|
-
}
|
|
53
|
-
set features(value) {
|
|
54
|
-
if (this._features === value) {
|
|
55
|
-
return;
|
|
56
|
-
}
|
|
57
|
-
this._features = value;
|
|
58
|
-
this.applyFeatureFilter(this.filter.value);
|
|
59
|
-
}
|
|
60
|
-
constructor(formBuilder) {
|
|
61
|
-
this.POLICIES = ['M', 'S', 'SF'];
|
|
62
|
-
/**
|
|
63
|
-
* True if the features are variable features.
|
|
64
|
-
*/
|
|
65
|
-
this.isVar = false;
|
|
66
|
-
/**
|
|
67
|
-
* The threshold at which the features filter should become visible.
|
|
68
|
-
* If set to 0, the filter is always visible; if set to -1, it is always
|
|
69
|
-
* invisible; otherwise, it gets visible when the number of features
|
|
70
|
-
* is greater than the threshold. Default is 5.
|
|
71
|
-
*/
|
|
72
|
-
this.filterThreshold = 5;
|
|
73
|
-
/**
|
|
74
|
-
* Emitted when the features change.
|
|
75
|
-
*/
|
|
76
|
-
this.featuresChange = new EventEmitter();
|
|
77
|
-
this._features = [];
|
|
78
|
-
this.filteredFeatures = [];
|
|
79
|
-
this.filter = formBuilder.control(null);
|
|
80
|
-
this._editedFeatureIndex = -1;
|
|
81
|
-
}
|
|
82
|
-
applyFeatureFilter(name) {
|
|
83
|
-
this.filteredFeatures = this.features.filter((feature) => {
|
|
84
|
-
return !name || feature.name.toLowerCase().includes(name.toLowerCase());
|
|
85
|
-
});
|
|
86
|
-
}
|
|
87
|
-
ngOnInit() {
|
|
88
|
-
this._sub = this.filter.valueChanges
|
|
89
|
-
.pipe(distinctUntilChanged(), debounceTime(300))
|
|
90
|
-
.subscribe((value) => {
|
|
91
|
-
this.applyFeatureFilter(value);
|
|
92
|
-
});
|
|
93
|
-
}
|
|
94
|
-
ngOnDestroy() {
|
|
95
|
-
this._sub?.unsubscribe();
|
|
96
|
-
}
|
|
97
|
-
addFeature() {
|
|
98
|
-
this.editedFeature = {
|
|
99
|
-
name: '',
|
|
100
|
-
value: '',
|
|
101
|
-
setPolicy: FeatureSetPolicy.multiple,
|
|
102
|
-
};
|
|
103
|
-
this._editedFeatureIndex = -1;
|
|
104
|
-
}
|
|
105
|
-
editFeature(feature) {
|
|
106
|
-
this.editedFeature = feature;
|
|
107
|
-
this._editedFeatureIndex = this.features.indexOf(feature);
|
|
108
|
-
}
|
|
109
|
-
deleteFeature(feature) {
|
|
110
|
-
const index = this.features.indexOf(feature);
|
|
111
|
-
if (index < 0) {
|
|
112
|
-
return;
|
|
113
|
-
}
|
|
114
|
-
const features = [...this.features];
|
|
115
|
-
features.splice(index, 1);
|
|
116
|
-
this.features = features;
|
|
117
|
-
this.featuresChange.emit(this.features);
|
|
118
|
-
}
|
|
119
|
-
onFeatureChange(feature) {
|
|
120
|
-
const features = [...this.features];
|
|
121
|
-
for (let i = features.length - 1; i >= 0; i--) {
|
|
122
|
-
if (features[i].name === feature.name) {
|
|
123
|
-
if (feature.setPolicy === FeatureSetPolicy.single ||
|
|
124
|
-
features[i].setPolicy === FeatureSetPolicy.single) {
|
|
125
|
-
features.splice(i, 1);
|
|
126
|
-
}
|
|
127
|
-
}
|
|
128
|
-
}
|
|
129
|
-
if (this._editedFeatureIndex === -1) {
|
|
130
|
-
features.push(feature);
|
|
131
|
-
}
|
|
132
|
-
else {
|
|
133
|
-
features.splice(this._editedFeatureIndex, 1, feature);
|
|
134
|
-
}
|
|
135
|
-
this.features = features;
|
|
136
|
-
this.editedFeature = undefined;
|
|
137
|
-
this._editedFeatureIndex = -1;
|
|
138
|
-
this.featuresChange.emit(this.features);
|
|
139
|
-
}
|
|
140
|
-
onFeatureCancel() {
|
|
141
|
-
this.editedFeature = undefined;
|
|
142
|
-
this._editedFeatureIndex = -1;
|
|
143
|
-
}
|
|
144
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: FeatureSetEditorComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
145
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.12", type: FeatureSetEditorComponent, isStandalone: true, selector: "gve-feature-set-editor", inputs: { isVar: "isVar", featNames: "featNames", featValues: "featValues", filterThreshold: "filterThreshold", features: "features" }, outputs: { featuresChange: "featuresChange" }, ngImport: i0, template: "<div>\r\n <div class=\"form-row\">\r\n <!-- filter -->\r\n @if (filterThreshold === 0 || features.length > filterThreshold) {\r\n <mat-form-field>\r\n <input matInput placeholder=\"filter\" [formControl]=\"filter\" />\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n matSuffix\r\n (click)=\"filter.reset()\"\r\n [disabled]=\"!filter.value\"\r\n [attr.aria-label]=\"'Reset filter'\"\r\n >\r\n <mat-icon color=\"warn\" class=\"mat-warn\">cancel</mat-icon>\r\n </button>\r\n </mat-form-field>\r\n }\r\n\r\n <!-- add feature button -->\r\n <button\r\n type=\"button\"\r\n color=\"primary\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Add a new feature\"\r\n (click)=\"addFeature()\"\r\n [class.in-row-button]=\"\r\n filterThreshold === 0 || features.length > filterThreshold\r\n \"\r\n >\r\n <mat-icon>add</mat-icon>\r\n feature\r\n </button>\r\n </div>\r\n\r\n <!-- list -->\r\n @if (features.length) {\r\n <table>\r\n <thead>\r\n <tr>\r\n <th></th>\r\n <th>feature</th>\r\n <th>value</th>\r\n <th>policy</th>\r\n @if (isVar) {\r\n <th>negated</th>\r\n <th>global</th>\r\n <th>short-lived</th>\r\n }\r\n </tr>\r\n </thead>\r\n <tbody>\r\n @for (feature of filteredFeatures; track feature) {\r\n <tr>\r\n <td class=\"fit-width\">\r\n <!-- edit -->\r\n <button\r\n type=\"button\"\r\n color=\"primary\"\r\n mat-icon-button\r\n matTooltip=\"Edit this feature\"\r\n (click)=\"editFeature(feature)\"\r\n >\r\n <mat-icon class=\"mat-primary\">edit</mat-icon>\r\n </button>\r\n <!-- delete -->\r\n <button\r\n type=\"button\"\r\n color=\"warn\"\r\n mat-icon-button\r\n matTooltip=\"Delete this feature\"\r\n (click)=\"deleteFeature(feature)\"\r\n >\r\n <mat-icon class=\"mat-warn\">delete</mat-icon>\r\n </button>\r\n </td>\r\n <td>\r\n @if (featNames?.length) {\r\n <span>{{\r\n feature.name | flatLookup : featNames : \"id\" : \"label\"\r\n }}</span>\r\n } @else {\r\n <span>{{ feature.name }}</span>\r\n }\r\n </td>\r\n <td>\r\n @if (featValues) {\r\n <span>{{\r\n feature.value | flatLookup : featValues[feature.name]\r\n }}</span>\r\n } @else {\r\n <span>{{ feature.value }}</span>\r\n }\r\n </td>\r\n <td>\r\n <span>{{ POLICIES[feature.setPolicy] }}</span>\r\n </td>\r\n @if (isVar) {\r\n <td>\r\n <span>{{ $any(feature).isNegated ? \"yes\" : \"no\" }}</span>\r\n </td>\r\n <td>\r\n <span>{{ $any(feature).isGlobal ? \"yes\" : \"no\" }}</span>\r\n </td>\r\n <td>\r\n <span>{{ $any(feature).isShortLived ? \"yes\" : \"no\" }}</span>\r\n </td>\r\n }\r\n </tr>\r\n }\r\n </tbody>\r\n </table>\r\n }\r\n\r\n <!-- editor -->\r\n <mat-expansion-panel [disabled]=\"!editedFeature\" [expanded]=\"editedFeature\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title>\r\n <span>{{ editedFeature?.name || \"feature\" }}</span>\r\n </mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <gve-feature-editor\r\n [isVar]=\"isVar\"\r\n [featNames]=\"featNames\"\r\n [featValues]=\"featValues\"\r\n [feature]=\"editedFeature\"\r\n (featureChange)=\"onFeatureChange($event)\"\r\n (featureCancel)=\"onFeatureCancel()\"\r\n />\r\n </mat-expansion-panel>\r\n</div>\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}.in-row-button{margin-top:-16px}table{width:100%;border-collapse:collapse;margin-top:8px}th{color:#909090;font-weight:400;text-align:left;background-color:#e1e0e0}th,td{padding:4px;border-bottom:1px solid silver}tbody tr:nth-child(2n){background-color:#e8e8e8}td.fit-width{width:1px;white-space:nowrap}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i2.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i2.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatExpansionModule }, { kind: "component", type: i3.MatExpansionPanel, selector: "mat-expansion-panel", inputs: ["hideToggle", "togglePosition"], outputs: ["afterExpand", "afterCollapse"], exportAs: ["matExpansionPanel"] }, { kind: "component", type: i3.MatExpansionPanelHeader, selector: "mat-expansion-panel-header", inputs: ["expandedHeight", "collapsedHeight", "tabIndex"] }, { kind: "directive", type: i3.MatExpansionPanelTitle, selector: "mat-panel-title" }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i4.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i4.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i5.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i6.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "ngmodule", type: MatSelectModule }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i7.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: NgToolsModule }, { kind: "pipe", type: i8.FlatLookupPipe, name: "flatLookup" }, { kind: "component", type: FeatureEditorComponent, selector: "gve-feature-editor", inputs: ["featNames", "featValues", "feature", "isVar"], outputs: ["featureCancel", "featureChange"] }] }); }
|
|
146
|
-
}
|
|
147
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: FeatureSetEditorComponent, decorators: [{
|
|
148
|
-
type: Component,
|
|
149
|
-
args: [{ selector: 'gve-feature-set-editor', standalone: true, imports: [
|
|
150
|
-
CommonModule,
|
|
151
|
-
ReactiveFormsModule,
|
|
152
|
-
MatButtonModule,
|
|
153
|
-
MatExpansionModule,
|
|
154
|
-
MatFormFieldModule,
|
|
155
|
-
MatIconModule,
|
|
156
|
-
MatInputModule,
|
|
157
|
-
MatSelectModule,
|
|
158
|
-
MatTooltipModule,
|
|
159
|
-
NgToolsModule,
|
|
160
|
-
FeatureEditorComponent,
|
|
161
|
-
], template: "<div>\r\n <div class=\"form-row\">\r\n <!-- filter -->\r\n @if (filterThreshold === 0 || features.length > filterThreshold) {\r\n <mat-form-field>\r\n <input matInput placeholder=\"filter\" [formControl]=\"filter\" />\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n matSuffix\r\n (click)=\"filter.reset()\"\r\n [disabled]=\"!filter.value\"\r\n [attr.aria-label]=\"'Reset filter'\"\r\n >\r\n <mat-icon color=\"warn\" class=\"mat-warn\">cancel</mat-icon>\r\n </button>\r\n </mat-form-field>\r\n }\r\n\r\n <!-- add feature button -->\r\n <button\r\n type=\"button\"\r\n color=\"primary\"\r\n class=\"mat-primary\"\r\n mat-flat-button\r\n matTooltip=\"Add a new feature\"\r\n (click)=\"addFeature()\"\r\n [class.in-row-button]=\"\r\n filterThreshold === 0 || features.length > filterThreshold\r\n \"\r\n >\r\n <mat-icon>add</mat-icon>\r\n feature\r\n </button>\r\n </div>\r\n\r\n <!-- list -->\r\n @if (features.length) {\r\n <table>\r\n <thead>\r\n <tr>\r\n <th></th>\r\n <th>feature</th>\r\n <th>value</th>\r\n <th>policy</th>\r\n @if (isVar) {\r\n <th>negated</th>\r\n <th>global</th>\r\n <th>short-lived</th>\r\n }\r\n </tr>\r\n </thead>\r\n <tbody>\r\n @for (feature of filteredFeatures; track feature) {\r\n <tr>\r\n <td class=\"fit-width\">\r\n <!-- edit -->\r\n <button\r\n type=\"button\"\r\n color=\"primary\"\r\n mat-icon-button\r\n matTooltip=\"Edit this feature\"\r\n (click)=\"editFeature(feature)\"\r\n >\r\n <mat-icon class=\"mat-primary\">edit</mat-icon>\r\n </button>\r\n <!-- delete -->\r\n <button\r\n type=\"button\"\r\n color=\"warn\"\r\n mat-icon-button\r\n matTooltip=\"Delete this feature\"\r\n (click)=\"deleteFeature(feature)\"\r\n >\r\n <mat-icon class=\"mat-warn\">delete</mat-icon>\r\n </button>\r\n </td>\r\n <td>\r\n @if (featNames?.length) {\r\n <span>{{\r\n feature.name | flatLookup : featNames : \"id\" : \"label\"\r\n }}</span>\r\n } @else {\r\n <span>{{ feature.name }}</span>\r\n }\r\n </td>\r\n <td>\r\n @if (featValues) {\r\n <span>{{\r\n feature.value | flatLookup : featValues[feature.name]\r\n }}</span>\r\n } @else {\r\n <span>{{ feature.value }}</span>\r\n }\r\n </td>\r\n <td>\r\n <span>{{ POLICIES[feature.setPolicy] }}</span>\r\n </td>\r\n @if (isVar) {\r\n <td>\r\n <span>{{ $any(feature).isNegated ? \"yes\" : \"no\" }}</span>\r\n </td>\r\n <td>\r\n <span>{{ $any(feature).isGlobal ? \"yes\" : \"no\" }}</span>\r\n </td>\r\n <td>\r\n <span>{{ $any(feature).isShortLived ? \"yes\" : \"no\" }}</span>\r\n </td>\r\n }\r\n </tr>\r\n }\r\n </tbody>\r\n </table>\r\n }\r\n\r\n <!-- editor -->\r\n <mat-expansion-panel [disabled]=\"!editedFeature\" [expanded]=\"editedFeature\">\r\n <mat-expansion-panel-header>\r\n <mat-panel-title>\r\n <span>{{ editedFeature?.name || \"feature\" }}</span>\r\n </mat-panel-title>\r\n </mat-expansion-panel-header>\r\n <gve-feature-editor\r\n [isVar]=\"isVar\"\r\n [featNames]=\"featNames\"\r\n [featValues]=\"featValues\"\r\n [feature]=\"editedFeature\"\r\n (featureChange)=\"onFeatureChange($event)\"\r\n (featureCancel)=\"onFeatureCancel()\"\r\n />\r\n </mat-expansion-panel>\r\n</div>\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}.in-row-button{margin-top:-16px}table{width:100%;border-collapse:collapse;margin-top:8px}th{color:#909090;font-weight:400;text-align:left;background-color:#e1e0e0}th,td{padding:4px;border-bottom:1px solid silver}tbody tr:nth-child(2n){background-color:#e8e8e8}td.fit-width{width:1px;white-space:nowrap}\n"] }]
|
|
162
|
-
}], ctorParameters: () => [{ type: i1.FormBuilder }], propDecorators: { isVar: [{
|
|
163
|
-
type: Input
|
|
164
|
-
}], featNames: [{
|
|
165
|
-
type: Input
|
|
166
|
-
}], featValues: [{
|
|
167
|
-
type: Input
|
|
168
|
-
}], filterThreshold: [{
|
|
169
|
-
type: Input
|
|
170
|
-
}], features: [{
|
|
171
|
-
type: Input
|
|
172
|
-
}], featuresChange: [{
|
|
173
|
-
type: Output
|
|
174
|
-
}] } });
|
|
175
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"feature-set-editor.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/gve-core/src/lib/components/feature-set-editor/feature-set-editor.component.ts","../../../../../../../projects/myrmidon/gve-core/src/lib/components/feature-set-editor/feature-set-editor.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\n\r\nimport { MatButtonModule } from '@angular/material/button';\r\nimport { MatExpansionModule } from '@angular/material/expansion';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatIconModule } from '@angular/material/icon';\r\nimport { MatInputModule } from '@angular/material/input';\r\nimport { MatSelectModule } from '@angular/material/select';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\n\r\nimport { Subscription, debounceTime, distinctUntilChanged } from 'rxjs';\r\n\r\nimport { NgToolsModule } from '@myrmidon/ng-tools';\r\n\r\nimport {\r\n  FeatureSetPolicy,\r\n  LabeledId,\r\n  Feature,\r\n  OperationFeature,\r\n} from '@myrmidon/gve-snapshot-view';\r\nimport {\r\n  FeatureEditorComponent,\r\n  FeatureMap,\r\n} from '../feature-editor/feature-editor.component';\r\n\r\n/**\r\n * 🔑 `gve-feature-set-editor`\r\n *\r\n * Component for editing a set of features. It shows a list of features,\r\n * allowing to edit each of them or adding new ones. Optionally you can\r\n * specify a list of feature names and values to use in the selection.\r\n * Used by base text editor and chain operation editor components.\r\n *\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 features.\r\n * - ▶️ `featValues` (`FeatureMap`): the feature values map. When specified\r\n * and the user selects a feature name present in the map keys, the corresponding\r\n * values will be used to populate the value selection.\r\n * - ▶️ `filterThreshold` (`number`): the threshold at which the features filter\r\n * should become visible. If set to 0, the filter is always visible; if set to -1,\r\n * it is always invisible; otherwise, it gets visible when the number of features\r\n * is greater than the threshold. Default is 5.\r\n * - ▶️ `features` (`Feature[]`): the features to edit.\r\n * - ▶️ `isVar`: true if the feature is a variant operation feature, which\r\n * has additional properties like negation, global, and short-lived.\r\n * - 🔥 `featuresChange` (`Feature[]`): emitted when features have changed.\r\n */\r\n@Component({\r\n  selector: 'gve-feature-set-editor',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    ReactiveFormsModule,\r\n    MatButtonModule,\r\n    MatExpansionModule,\r\n    MatFormFieldModule,\r\n    MatIconModule,\r\n    MatInputModule,\r\n    MatSelectModule,\r\n    MatTooltipModule,\r\n    NgToolsModule,\r\n    FeatureEditorComponent,\r\n  ],\r\n  templateUrl: './feature-set-editor.component.html',\r\n  styleUrl: './feature-set-editor.component.css',\r\n})\r\nexport class FeatureSetEditorComponent implements OnInit, OnDestroy {\r\n  private _sub?: Subscription;\r\n  private _features: Feature[] | OperationFeature[];\r\n  private _editedFeatureIndex: number;\r\n\r\n  public POLICIES = ['M', 'S', 'SF'];\r\n\r\n  /**\r\n   * True if the features are variable features.\r\n   */\r\n  @Input()\r\n  public isVar = false;\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 featValues: FeatureMap | undefined;\r\n\r\n  /**\r\n   * The threshold at which the features filter should become visible.\r\n   * If set to 0, the filter is always visible; if set to -1, it is always\r\n   * invisible; otherwise, it gets visible when the number of features\r\n   * is greater than the threshold. Default is 5.\r\n   */\r\n  @Input()\r\n  public filterThreshold = 5;\r\n\r\n  /**\r\n   * The features to edit.\r\n   */\r\n  @Input()\r\n  public get features(): Feature[] | OperationFeature[] {\r\n    return this._features;\r\n  }\r\n  public set features(value: Feature[] | OperationFeature[]) {\r\n    if (this._features === value) {\r\n      return;\r\n    }\r\n    this._features = value;\r\n    this.applyFeatureFilter(this.filter.value);\r\n  }\r\n\r\n  /**\r\n   * Emitted when the features change.\r\n   */\r\n  @Output()\r\n  public featuresChange = new EventEmitter<Feature[] | OperationFeature[]>();\r\n\r\n  public editedFeature?: Feature | OperationFeature;\r\n\r\n  // filter\r\n  public filter: FormControl<string | null>;\r\n  public filteredFeatures: Feature[] | OperationFeature[];\r\n\r\n  constructor(formBuilder: FormBuilder) {\r\n    this._features = [];\r\n    this.filteredFeatures = [];\r\n    this.filter = formBuilder.control<string | null>(null);\r\n    this._editedFeatureIndex = -1;\r\n  }\r\n\r\n  private applyFeatureFilter(name?: string | null): void {\r\n    this.filteredFeatures = this.features.filter((feature) => {\r\n      return !name || feature.name.toLowerCase().includes(name.toLowerCase());\r\n    });\r\n  }\r\n\r\n  public ngOnInit(): void {\r\n    this._sub = this.filter.valueChanges\r\n      .pipe(distinctUntilChanged(), debounceTime(300))\r\n      .subscribe((value) => {\r\n        this.applyFeatureFilter(value);\r\n      });\r\n  }\r\n\r\n  public ngOnDestroy(): void {\r\n    this._sub?.unsubscribe();\r\n  }\r\n\r\n  public addFeature(): void {\r\n    this.editedFeature = {\r\n      name: '',\r\n      value: '',\r\n      setPolicy: FeatureSetPolicy.multiple,\r\n    };\r\n    this._editedFeatureIndex = -1;\r\n  }\r\n\r\n  public editFeature(feature: Feature | OperationFeature): void {\r\n    this.editedFeature = feature;\r\n    this._editedFeatureIndex = this.features.indexOf(feature);\r\n  }\r\n\r\n  public deleteFeature(feature: Feature | OperationFeature): void {\r\n    const index = this.features.indexOf(feature);\r\n    if (index < 0) {\r\n      return;\r\n    }\r\n    const features = [...this.features];\r\n    features.splice(index, 1);\r\n    this.features = features;\r\n    this.featuresChange.emit(this.features);\r\n  }\r\n\r\n  public onFeatureChange(feature: Feature | OperationFeature): void {\r\n    const features = [...this.features];\r\n\r\n    for (let i = features.length - 1; i >= 0; i--) {\r\n      if (features[i].name === feature.name) {\r\n        if (\r\n          feature.setPolicy === FeatureSetPolicy.single ||\r\n          features[i].setPolicy === FeatureSetPolicy.single\r\n        ) {\r\n          features.splice(i, 1);\r\n        }\r\n      }\r\n    }\r\n\r\n    if (this._editedFeatureIndex === -1) {\r\n      features.push(feature);\r\n    } else {\r\n      features.splice(this._editedFeatureIndex, 1, feature);\r\n    }\r\n\r\n    this.features = features;\r\n    this.editedFeature = undefined;\r\n    this._editedFeatureIndex = -1;\r\n    this.featuresChange.emit(this.features);\r\n  }\r\n\r\n  public onFeatureCancel(): void {\r\n    this.editedFeature = undefined;\r\n    this._editedFeatureIndex = -1;\r\n  }\r\n}\r\n","<div>\r\n  <div class=\"form-row\">\r\n    <!-- filter -->\r\n    @if (filterThreshold === 0 || features.length > filterThreshold) {\r\n    <mat-form-field>\r\n      <input matInput placeholder=\"filter\" [formControl]=\"filter\" />\r\n      <button\r\n        type=\"button\"\r\n        mat-icon-button\r\n        matSuffix\r\n        (click)=\"filter.reset()\"\r\n        [disabled]=\"!filter.value\"\r\n        [attr.aria-label]=\"'Reset filter'\"\r\n      >\r\n        <mat-icon color=\"warn\" class=\"mat-warn\">cancel</mat-icon>\r\n      </button>\r\n    </mat-form-field>\r\n    }\r\n\r\n    <!-- add feature button -->\r\n    <button\r\n      type=\"button\"\r\n      color=\"primary\"\r\n      class=\"mat-primary\"\r\n      mat-flat-button\r\n      matTooltip=\"Add a new feature\"\r\n      (click)=\"addFeature()\"\r\n      [class.in-row-button]=\"\r\n        filterThreshold === 0 || features.length > filterThreshold\r\n      \"\r\n    >\r\n      <mat-icon>add</mat-icon>\r\n      feature\r\n    </button>\r\n  </div>\r\n\r\n  <!-- list -->\r\n  @if (features.length) {\r\n  <table>\r\n    <thead>\r\n      <tr>\r\n        <th></th>\r\n        <th>feature</th>\r\n        <th>value</th>\r\n        <th>policy</th>\r\n        @if (isVar) {\r\n        <th>negated</th>\r\n        <th>global</th>\r\n        <th>short-lived</th>\r\n        }\r\n      </tr>\r\n    </thead>\r\n    <tbody>\r\n      @for (feature of filteredFeatures; track feature) {\r\n      <tr>\r\n        <td class=\"fit-width\">\r\n          <!-- edit -->\r\n          <button\r\n            type=\"button\"\r\n            color=\"primary\"\r\n            mat-icon-button\r\n            matTooltip=\"Edit this feature\"\r\n            (click)=\"editFeature(feature)\"\r\n          >\r\n            <mat-icon class=\"mat-primary\">edit</mat-icon>\r\n          </button>\r\n          <!-- delete -->\r\n          <button\r\n            type=\"button\"\r\n            color=\"warn\"\r\n            mat-icon-button\r\n            matTooltip=\"Delete this feature\"\r\n            (click)=\"deleteFeature(feature)\"\r\n          >\r\n            <mat-icon class=\"mat-warn\">delete</mat-icon>\r\n          </button>\r\n        </td>\r\n        <td>\r\n          @if (featNames?.length) {\r\n          <span>{{\r\n            feature.name | flatLookup : featNames : \"id\" : \"label\"\r\n          }}</span>\r\n          } @else {\r\n          <span>{{ feature.name }}</span>\r\n          }\r\n        </td>\r\n        <td>\r\n          @if (featValues) {\r\n          <span>{{\r\n            feature.value | flatLookup : featValues[feature.name]\r\n          }}</span>\r\n          } @else {\r\n          <span>{{ feature.value }}</span>\r\n          }\r\n        </td>\r\n        <td>\r\n          <span>{{ POLICIES[feature.setPolicy] }}</span>\r\n        </td>\r\n        @if (isVar) {\r\n        <td>\r\n          <span>{{ $any(feature).isNegated ? \"yes\" : \"no\" }}</span>\r\n        </td>\r\n        <td>\r\n          <span>{{ $any(feature).isGlobal ? \"yes\" : \"no\" }}</span>\r\n        </td>\r\n        <td>\r\n          <span>{{ $any(feature).isShortLived ? \"yes\" : \"no\" }}</span>\r\n        </td>\r\n        }\r\n      </tr>\r\n      }\r\n    </tbody>\r\n  </table>\r\n  }\r\n\r\n  <!-- editor -->\r\n  <mat-expansion-panel [disabled]=\"!editedFeature\" [expanded]=\"editedFeature\">\r\n    <mat-expansion-panel-header>\r\n      <mat-panel-title>\r\n        <span>{{ editedFeature?.name || \"feature\" }}</span>\r\n      </mat-panel-title>\r\n    </mat-expansion-panel-header>\r\n    <gve-feature-editor\r\n      [isVar]=\"isVar\"\r\n      [featNames]=\"featNames\"\r\n      [featValues]=\"featValues\"\r\n      [feature]=\"editedFeature\"\r\n      (featureChange)=\"onFeatureChange($event)\"\r\n      (featureCancel)=\"onFeatureCancel()\"\r\n    />\r\n  </mat-expansion-panel>\r\n</div>\r\n"]}
|
|
@@ -1,100 +0,0 @@
|
|
|
1
|
-
import { Component, Input } from '@angular/core';
|
|
2
|
-
import { ReactiveFormsModule } from '@angular/forms';
|
|
3
|
-
import { CommonModule } from '@angular/common';
|
|
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 { MatInputModule } from '@angular/material/input';
|
|
9
|
-
import { NgToolsModule } from '@myrmidon/ng-tools';
|
|
10
|
-
import * as i0 from "@angular/core";
|
|
11
|
-
import * as i1 from "@angular/forms";
|
|
12
|
-
import * as i2 from "@angular/material/button";
|
|
13
|
-
import * as i3 from "@angular/material/form-field";
|
|
14
|
-
import * as i4 from "@angular/material/icon";
|
|
15
|
-
import * as i5 from "@angular/material/input";
|
|
16
|
-
import * as i6 from "@myrmidon/ng-tools";
|
|
17
|
-
/**
|
|
18
|
-
* 🔑 `gve-feature-set-view`
|
|
19
|
-
*
|
|
20
|
-
* A component to display a list of features.
|
|
21
|
-
* Used by the chain result view component.
|
|
22
|
-
*
|
|
23
|
-
* - ▶️ `features` (`Feature[]`): the features to display.
|
|
24
|
-
* - ▶️ `featNames` (`LabeledId[]`): the list of feature names to display
|
|
25
|
-
* in the _name_ selection. This is used when you have a closed list of features.
|
|
26
|
-
* - ▶️ `featValues` (`FeatureMap`): the feature values map. When specified
|
|
27
|
-
* and the user selects a feature name present in the map keys, the corresponding
|
|
28
|
-
* values will be used to populate the value selection.
|
|
29
|
-
* - ▶️ `filterThreshold` (`number`): the threshold at which the features filter
|
|
30
|
-
* should become visible. If set to 0, the filter is always visible; if set to -1,
|
|
31
|
-
* it is always invisible; otherwise, it gets visible when the number of features
|
|
32
|
-
* is greater than the threshold. Default is 5.
|
|
33
|
-
*/
|
|
34
|
-
export class FeatureSetViewComponent {
|
|
35
|
-
/**
|
|
36
|
-
* The features.
|
|
37
|
-
*/
|
|
38
|
-
get features() {
|
|
39
|
-
return this._features;
|
|
40
|
-
}
|
|
41
|
-
set features(value) {
|
|
42
|
-
if (this._features === value) {
|
|
43
|
-
return;
|
|
44
|
-
}
|
|
45
|
-
this._features = value || undefined;
|
|
46
|
-
this.filter.reset();
|
|
47
|
-
this.applyFeatureFilter();
|
|
48
|
-
}
|
|
49
|
-
constructor(formBuilder) {
|
|
50
|
-
this._features = [];
|
|
51
|
-
/**
|
|
52
|
-
* The threshold at which the features filter should become visible.
|
|
53
|
-
* If set to 0, the filter is always visible; if set to -1, it is always
|
|
54
|
-
* invisible; otherwise, it gets visible when the number of features
|
|
55
|
-
* is greater than the threshold. Default is 5.
|
|
56
|
-
*/
|
|
57
|
-
this.filterThreshold = 5;
|
|
58
|
-
this.filteredFeatures = [];
|
|
59
|
-
this.filter = formBuilder.control(null);
|
|
60
|
-
}
|
|
61
|
-
applyFeatureFilter(name) {
|
|
62
|
-
this.filteredFeatures = this._features.filter((feature) => {
|
|
63
|
-
return !name || feature.name.toLowerCase().includes(name.toLowerCase());
|
|
64
|
-
});
|
|
65
|
-
}
|
|
66
|
-
ngOnInit() {
|
|
67
|
-
this._sub = this.filter.valueChanges
|
|
68
|
-
.pipe(distinctUntilChanged(), debounceTime(300))
|
|
69
|
-
.subscribe((value) => {
|
|
70
|
-
this.applyFeatureFilter(value);
|
|
71
|
-
});
|
|
72
|
-
this.applyFeatureFilter();
|
|
73
|
-
}
|
|
74
|
-
ngOnDestroy() {
|
|
75
|
-
this._sub?.unsubscribe();
|
|
76
|
-
}
|
|
77
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: FeatureSetViewComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
78
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.12", type: FeatureSetViewComponent, isStandalone: true, selector: "gve-feature-set-view", inputs: { features: "features", featNames: "featNames", featValues: "featValues", filterThreshold: "filterThreshold" }, ngImport: i0, template: "<div>\r\n <!-- filter -->\r\n <div>\r\n @if (filterThreshold === 0 || features.length > filterThreshold) {\r\n <div class=\"form-row\">\r\n <mat-form-field id=\"filter\">\r\n <mat-label>filter</mat-label>\r\n <input matInput [formControl]=\"filter\" />\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n matSuffix\r\n (click)=\"filter.reset()\"\r\n [disabled]=\"!filter.value\"\r\n [attr.aria-label]=\"'Reset filter'\"\r\n >\r\n <mat-icon color=\"warn\" class=\"mat-warn\">cancel</mat-icon>\r\n </button>\r\n </mat-form-field>\r\n\r\n <span>{{ filteredFeatures.length }}</span>\r\n </div>\r\n }\r\n\r\n <!-- list -->\r\n @if (filteredFeatures.length) {\r\n <table>\r\n <tbody>\r\n @for (feature of filteredFeatures; track $index) {\r\n <tr>\r\n <th>\r\n @if (featNames?.length) {\r\n <span>{{\r\n feature.name | flatLookup : featNames : \"id\" : \"label\"\r\n }}</span>\r\n } @else {\r\n <span>{{ feature.name }}</span>\r\n }\r\n </th>\r\n <td>\r\n @if (featValues) {\r\n <span>{{\r\n feature.value | flatLookup : featValues[feature.name]\r\n }}</span>\r\n } @else {\r\n <span>{{ feature.value }}</span>\r\n }\r\n </td>\r\n </tr>\r\n }\r\n </tbody>\r\n </table>\r\n }\r\n </div>\r\n</div>\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}.in-row-button{margin-top:-16px}table{width:100%;border-collapse:collapse}th{text-align:left;background-color:#c8d9eb;color:#333;font-weight:400}th,td{border:1px solid silver;padding:4px}tr:nth-child(2n){background-color:#dfdfdf}#filter{width:7em}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i2.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { 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: "directive", type: i3.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i4.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i5.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: NgToolsModule }, { kind: "pipe", type: i6.FlatLookupPipe, name: "flatLookup" }] }); }
|
|
79
|
-
}
|
|
80
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: FeatureSetViewComponent, decorators: [{
|
|
81
|
-
type: Component,
|
|
82
|
-
args: [{ selector: 'gve-feature-set-view', standalone: true, imports: [
|
|
83
|
-
CommonModule,
|
|
84
|
-
ReactiveFormsModule,
|
|
85
|
-
MatButtonModule,
|
|
86
|
-
MatFormFieldModule,
|
|
87
|
-
MatIconModule,
|
|
88
|
-
MatInputModule,
|
|
89
|
-
NgToolsModule,
|
|
90
|
-
], template: "<div>\r\n <!-- filter -->\r\n <div>\r\n @if (filterThreshold === 0 || features.length > filterThreshold) {\r\n <div class=\"form-row\">\r\n <mat-form-field id=\"filter\">\r\n <mat-label>filter</mat-label>\r\n <input matInput [formControl]=\"filter\" />\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n matSuffix\r\n (click)=\"filter.reset()\"\r\n [disabled]=\"!filter.value\"\r\n [attr.aria-label]=\"'Reset filter'\"\r\n >\r\n <mat-icon color=\"warn\" class=\"mat-warn\">cancel</mat-icon>\r\n </button>\r\n </mat-form-field>\r\n\r\n <span>{{ filteredFeatures.length }}</span>\r\n </div>\r\n }\r\n\r\n <!-- list -->\r\n @if (filteredFeatures.length) {\r\n <table>\r\n <tbody>\r\n @for (feature of filteredFeatures; track $index) {\r\n <tr>\r\n <th>\r\n @if (featNames?.length) {\r\n <span>{{\r\n feature.name | flatLookup : featNames : \"id\" : \"label\"\r\n }}</span>\r\n } @else {\r\n <span>{{ feature.name }}</span>\r\n }\r\n </th>\r\n <td>\r\n @if (featValues) {\r\n <span>{{\r\n feature.value | flatLookup : featValues[feature.name]\r\n }}</span>\r\n } @else {\r\n <span>{{ feature.value }}</span>\r\n }\r\n </td>\r\n </tr>\r\n }\r\n </tbody>\r\n </table>\r\n }\r\n </div>\r\n</div>\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}.in-row-button{margin-top:-16px}table{width:100%;border-collapse:collapse}th{text-align:left;background-color:#c8d9eb;color:#333;font-weight:400}th,td{border:1px solid silver;padding:4px}tr:nth-child(2n){background-color:#dfdfdf}#filter{width:7em}\n"] }]
|
|
91
|
-
}], ctorParameters: () => [{ type: i1.FormBuilder }], propDecorators: { features: [{
|
|
92
|
-
type: Input
|
|
93
|
-
}], featNames: [{
|
|
94
|
-
type: Input
|
|
95
|
-
}], featValues: [{
|
|
96
|
-
type: Input
|
|
97
|
-
}], filterThreshold: [{
|
|
98
|
-
type: Input
|
|
99
|
-
}] } });
|
|
100
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"feature-set-view.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/gve-core/src/lib/components/feature-set-view/feature-set-view.component.ts","../../../../../../../projects/myrmidon/gve-core/src/lib/components/feature-set-view/feature-set-view.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,eAAe,CAAC;AACjD,OAAO,EAA4B,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAC/E,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAgB,YAAY,EAAE,oBAAoB,EAAE,MAAM,MAAM,CAAC;AAExE,OAAO,EAAE,eAAe,EAAE,MAAM,0BAA0B,CAAC;AAC3D,OAAO,EAAE,kBAAkB,EAAE,MAAM,8BAA8B,CAAC;AAClE,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAE,cAAc,EAAE,MAAM,yBAAyB,CAAC;AAEzD,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;;;;;;;;AASnD;;;;;;;;;;;;;;;;GAgBG;AAgBH,MAAM,OAAO,uBAAuB;IAIlC;;OAEG;IACH,IACW,QAAQ;QACjB,OAAO,IAAI,CAAC,SAAS,CAAC;IACxB,CAAC;IACD,IAAW,QAAQ,CAAC,KAAqC;QACvD,IAAI,IAAI,CAAC,SAAS,KAAK,KAAK,EAAE,CAAC;YAC7B,OAAO;QACT,CAAC;QACD,IAAI,CAAC,SAAS,GAAG,KAAK,IAAI,SAAS,CAAC;QACpC,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;QACpB,IAAI,CAAC,kBAAkB,EAAE,CAAC;IAC5B,CAAC;IA8BD,YAAY,WAAwB;QA9C5B,cAAS,GAAmC,EAAE,CAAC;QAiCvD;;;;;WAKG;QAEI,oBAAe,GAAG,CAAC,CAAC;QAIpB,qBAAgB,GAAmC,EAAE,CAAC;QAG3D,IAAI,CAAC,MAAM,GAAG,WAAW,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;IAC1C,CAAC;IAEO,kBAAkB,CAAC,IAAoB;QAC7C,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE;YACxD,OAAO,CAAC,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC;QAC1E,CAAC,CAAC,CAAC;IACL,CAAC;IAEM,QAAQ;QACb,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,MAAM,CAAC,YAAY;aACjC,IAAI,CAAC,oBAAoB,EAAE,EAAE,YAAY,CAAC,GAAG,CAAC,CAAC;aAC/C,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC;QACjC,CAAC,CAAC,CAAC;QACL,IAAI,CAAC,kBAAkB,EAAE,CAAC;IAC5B,CAAC;IAEM,WAAW;QAChB,IAAI,CAAC,IAAI,EAAE,WAAW,EAAE,CAAC;IAC3B,CAAC;+GArEU,uBAAuB;mGAAvB,uBAAuB,wMCnDpC,uiDAuDA,6YDfI,YAAY,8BACZ,mBAAmB,ykBACnB,eAAe,2IACf,kBAAkB,2aAClB,aAAa,mLACb,cAAc,0WACd,aAAa;;4FAKJ,uBAAuB;kBAfnC,SAAS;+BACE,sBAAsB,cACpB,IAAI,WACP;wBACP,YAAY;wBACZ,mBAAmB;wBACnB,eAAe;wBACf,kBAAkB;wBAClB,aAAa;wBACb,cAAc;wBACd,aAAa;qBACd;gFAYU,QAAQ;sBADlB,KAAK;gBAkBC,SAAS;sBADf,KAAK;gBASC,UAAU;sBADhB,KAAK;gBAUC,eAAe;sBADrB,KAAK","sourcesContent":["import { Component, Input } from '@angular/core';\r\nimport { FormBuilder, FormControl, ReactiveFormsModule } from '@angular/forms';\r\nimport { CommonModule } from '@angular/common';\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 { MatInputModule } from '@angular/material/input';\r\n\r\nimport { NgToolsModule } from '@myrmidon/ng-tools';\r\n\r\nimport {\r\n  Feature,\r\n  LabeledId,\r\n  OperationFeature,\r\n} from '@myrmidon/gve-snapshot-view';\r\nimport { FeatureMap } from '../feature-editor/feature-editor.component';\r\n\r\n/**\r\n * 🔑 `gve-feature-set-view`\r\n *\r\n * A component to display a list of features.\r\n * Used by the chain result view component.\r\n *\r\n * - ▶️ `features` (`Feature[]`): the features to display.\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 features.\r\n * - ▶️ `featValues` (`FeatureMap`): the feature values map. When specified\r\n * and the user selects a feature name present in the map keys, the corresponding\r\n * values will be used to populate the value selection.\r\n * - ▶️ `filterThreshold` (`number`): the threshold at which the features filter\r\n * should become visible. If set to 0, the filter is always visible; if set to -1,\r\n * it is always invisible; otherwise, it gets visible when the number of features\r\n * is greater than the threshold. Default is 5.\r\n */\r\n@Component({\r\n  selector: 'gve-feature-set-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    MatInputModule,\r\n    NgToolsModule,\r\n  ],\r\n  templateUrl: './feature-set-view.component.html',\r\n  styleUrl: './feature-set-view.component.css',\r\n})\r\nexport class FeatureSetViewComponent {\r\n  private _sub?: Subscription;\r\n  private _features: Feature[] | OperationFeature[] = [];\r\n\r\n  /**\r\n   * The features.\r\n   */\r\n  @Input()\r\n  public get features(): Feature[] | OperationFeature[] {\r\n    return this._features;\r\n  }\r\n  public set features(value: Feature[] | OperationFeature[]) {\r\n    if (this._features === value) {\r\n      return;\r\n    }\r\n    this._features = value || undefined;\r\n    this.filter.reset();\r\n    this.applyFeatureFilter();\r\n  }\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 featValues: FeatureMap | undefined;\r\n\r\n  /**\r\n   * The threshold at which the features filter should become visible.\r\n   * If set to 0, the filter is always visible; if set to -1, it is always\r\n   * invisible; otherwise, it gets visible when the number of features\r\n   * is greater than the threshold. Default is 5.\r\n   */\r\n  @Input()\r\n  public filterThreshold = 5;\r\n\r\n  // filter\r\n  public filter: FormControl<string | null>;\r\n  public filteredFeatures: Feature[] | OperationFeature[] = [];\r\n\r\n  constructor(formBuilder: FormBuilder) {\r\n    this.filter = formBuilder.control(null);\r\n  }\r\n\r\n  private applyFeatureFilter(name?: string | null): void {\r\n    this.filteredFeatures = this._features.filter((feature) => {\r\n      return !name || feature.name.toLowerCase().includes(name.toLowerCase());\r\n    });\r\n  }\r\n\r\n  public ngOnInit(): void {\r\n    this._sub = this.filter.valueChanges\r\n      .pipe(distinctUntilChanged(), debounceTime(300))\r\n      .subscribe((value) => {\r\n        this.applyFeatureFilter(value);\r\n      });\r\n    this.applyFeatureFilter();\r\n  }\r\n\r\n  public ngOnDestroy(): void {\r\n    this._sub?.unsubscribe();\r\n  }\r\n}\r\n","<div>\r\n  <!-- filter -->\r\n  <div>\r\n    @if (filterThreshold === 0 || features.length > filterThreshold) {\r\n    <div class=\"form-row\">\r\n      <mat-form-field id=\"filter\">\r\n        <mat-label>filter</mat-label>\r\n        <input matInput [formControl]=\"filter\" />\r\n        <button\r\n          type=\"button\"\r\n          mat-icon-button\r\n          matSuffix\r\n          (click)=\"filter.reset()\"\r\n          [disabled]=\"!filter.value\"\r\n          [attr.aria-label]=\"'Reset filter'\"\r\n        >\r\n          <mat-icon color=\"warn\"  class=\"mat-warn\">cancel</mat-icon>\r\n        </button>\r\n      </mat-form-field>\r\n\r\n      <span>{{ filteredFeatures.length }}</span>\r\n    </div>\r\n    }\r\n\r\n    <!-- list -->\r\n    @if (filteredFeatures.length) {\r\n    <table>\r\n      <tbody>\r\n        @for (feature of filteredFeatures; track $index) {\r\n        <tr>\r\n          <th>\r\n            @if (featNames?.length) {\r\n            <span>{{\r\n              feature.name | flatLookup : featNames : \"id\" : \"label\"\r\n            }}</span>\r\n            } @else {\r\n            <span>{{ feature.name }}</span>\r\n            }\r\n          </th>\r\n          <td>\r\n            @if (featValues) {\r\n            <span>{{\r\n              feature.value | flatLookup : featValues[feature.name]\r\n            }}</span>\r\n            } @else {\r\n            <span>{{ feature.value }}</span>\r\n            }\r\n          </td>\r\n        </tr>\r\n        }\r\n      </tbody>\r\n    </table>\r\n    }\r\n  </div>\r\n</div>\r\n"]}
|
|
@@ -1,126 +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 { MatInputModule } from '@angular/material/input';
|
|
9
|
-
import { MatSelectModule } from '@angular/material/select';
|
|
10
|
-
import { MatTooltipModule } from '@angular/material/tooltip';
|
|
11
|
-
import * as i0 from "@angular/core";
|
|
12
|
-
import * as i1 from "@angular/forms";
|
|
13
|
-
import * as i2 from "@angular/common";
|
|
14
|
-
import * as i3 from "@angular/material/button";
|
|
15
|
-
import * as i4 from "@angular/material/form-field";
|
|
16
|
-
import * as i5 from "@angular/material/icon";
|
|
17
|
-
import * as i6 from "@angular/material/input";
|
|
18
|
-
import * as i7 from "@angular/material/select";
|
|
19
|
-
import * as i8 from "@angular/material/core";
|
|
20
|
-
import * as i9 from "@angular/material/tooltip";
|
|
21
|
-
/**
|
|
22
|
-
* 🔑 `gve-ln-heights-editor`
|
|
23
|
-
*
|
|
24
|
-
* A component to edit line heights.
|
|
25
|
-
* Used by the `gve-snapshot-editor` component.
|
|
26
|
-
*
|
|
27
|
-
* - ▶️ `lineCount` (`number`): the number of lines.
|
|
28
|
-
* - ▶️ `heights` (`Record<number, number>`): the line heights.
|
|
29
|
-
* - 🔥 `heightsChange` (`EventEmitter<Record<number, number>>`): the event
|
|
30
|
-
* emitted when the heights change.
|
|
31
|
-
*/
|
|
32
|
-
export class LnHeightsEditorComponent {
|
|
33
|
-
/**
|
|
34
|
-
* The total number of lines in the text.
|
|
35
|
-
*/
|
|
36
|
-
get lineCount() {
|
|
37
|
-
return this._lineCount;
|
|
38
|
-
}
|
|
39
|
-
set lineCount(value) {
|
|
40
|
-
if (this._lineCount === value)
|
|
41
|
-
return;
|
|
42
|
-
this._lineCount = value;
|
|
43
|
-
this.lineNumbers = Array.from({ length: value }, (_, i) => i + 1);
|
|
44
|
-
}
|
|
45
|
-
/**
|
|
46
|
-
* The heights map of the lines. Each key is a line number and the value is
|
|
47
|
-
* the height of the line.
|
|
48
|
-
*/
|
|
49
|
-
get heights() {
|
|
50
|
-
return this._heights;
|
|
51
|
-
}
|
|
52
|
-
set heights(value) {
|
|
53
|
-
if (this._heights === value)
|
|
54
|
-
return;
|
|
55
|
-
this._heights = value || undefined;
|
|
56
|
-
}
|
|
57
|
-
constructor(formBuilder) {
|
|
58
|
-
this._lineCount = 0;
|
|
59
|
-
/**
|
|
60
|
-
* The event emitted when the heights change.
|
|
61
|
-
*/
|
|
62
|
-
this.heightsChange = new EventEmitter();
|
|
63
|
-
this.lineNumbers = [];
|
|
64
|
-
this.lineNumber = formBuilder.control(0, { nonNullable: true });
|
|
65
|
-
this.height = formBuilder.control(0, { nonNullable: true });
|
|
66
|
-
}
|
|
67
|
-
pruneHeights() {
|
|
68
|
-
// remove all the heigths with value=0
|
|
69
|
-
this._heights = Object.fromEntries(Object.entries(this._heights || {}).filter(([_, v]) => v !== 0));
|
|
70
|
-
// if heights are now empty, set to undefined
|
|
71
|
-
if (Object.keys(this._heights).length === 0) {
|
|
72
|
-
this._heights = undefined;
|
|
73
|
-
}
|
|
74
|
-
}
|
|
75
|
-
ngOnInit() {
|
|
76
|
-
this._subs = [];
|
|
77
|
-
// update heights[ln] when height changes and emit heightsChange
|
|
78
|
-
this._subs.push(this.height.valueChanges
|
|
79
|
-
.pipe(distinctUntilChanged(), debounceTime(200))
|
|
80
|
-
.subscribe((value) => {
|
|
81
|
-
if (!this._heights) {
|
|
82
|
-
this._heights = {};
|
|
83
|
-
}
|
|
84
|
-
this._heights[this.lineNumber.value] = value;
|
|
85
|
-
this.pruneHeights();
|
|
86
|
-
this.heightsChange.emit(this._heights);
|
|
87
|
-
}));
|
|
88
|
-
// update height when line number changes
|
|
89
|
-
this._subs.push(this.lineNumber.valueChanges
|
|
90
|
-
.pipe(distinctUntilChanged(), debounceTime(200))
|
|
91
|
-
.subscribe((value) => {
|
|
92
|
-
if (!this._heights)
|
|
93
|
-
return;
|
|
94
|
-
this.height.setValue(this._heights[value] || 0);
|
|
95
|
-
}));
|
|
96
|
-
}
|
|
97
|
-
ngOnDestroy() {
|
|
98
|
-
this._subs?.forEach((sub) => sub.unsubscribe());
|
|
99
|
-
}
|
|
100
|
-
reset() {
|
|
101
|
-
this._heights = undefined;
|
|
102
|
-
this.heightsChange.emit(undefined);
|
|
103
|
-
}
|
|
104
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: LnHeightsEditorComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
105
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.12", type: LnHeightsEditorComponent, isStandalone: true, selector: "gve-ln-heights-editor", inputs: { lineCount: "lineCount", heights: "heights" }, outputs: { heightsChange: "heightsChange" }, ngImport: i0, template: "@if (lineNumbers.length) {\r\n<div class=\"form-row\">\r\n <!-- line number -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>line</mat-label>\r\n <mat-select [formControl]=\"lineNumber\">\r\n <mat-option *ngFor=\"let n of lineNumbers\" [value]=\"n\">\r\n {{ n }}\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n\r\n <!-- height -->\r\n @if (lineNumber.value) {\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input matInput type=\"number\" [formControl]=\"height\" min=\"0\" />\r\n </mat-form-field>\r\n }\r\n\r\n <!-- reset button -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"warn\"\r\n (click)=\"reset()\"\r\n matTooltip=\"Remove all the heights\"\r\n >\r\n <mat-icon class=\"mat-warn\">clear</mat-icon>\r\n </button>\r\n</div>\r\n}\r\n", styles: [".input-nr{width:5em}.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: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.MinValidator, selector: "input[type=number][min][formControlName],input[type=number][min][formControl],input[type=number][min][ngModel]", inputs: ["min"] }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i3.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i4.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i4.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i5.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i6.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "ngmodule", type: MatSelectModule }, { kind: "component", type: i7.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: i8.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i9.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }] }); }
|
|
106
|
-
}
|
|
107
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: LnHeightsEditorComponent, decorators: [{
|
|
108
|
-
type: Component,
|
|
109
|
-
args: [{ selector: 'gve-ln-heights-editor', standalone: true, imports: [
|
|
110
|
-
CommonModule,
|
|
111
|
-
ReactiveFormsModule,
|
|
112
|
-
MatButtonModule,
|
|
113
|
-
MatFormFieldModule,
|
|
114
|
-
MatIconModule,
|
|
115
|
-
MatInputModule,
|
|
116
|
-
MatSelectModule,
|
|
117
|
-
MatTooltipModule,
|
|
118
|
-
], template: "@if (lineNumbers.length) {\r\n<div class=\"form-row\">\r\n <!-- line number -->\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>line</mat-label>\r\n <mat-select [formControl]=\"lineNumber\">\r\n <mat-option *ngFor=\"let n of lineNumbers\" [value]=\"n\">\r\n {{ n }}\r\n </mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n\r\n <!-- height -->\r\n @if (lineNumber.value) {\r\n <mat-form-field class=\"input-nr\">\r\n <mat-label>height</mat-label>\r\n <input matInput type=\"number\" [formControl]=\"height\" min=\"0\" />\r\n </mat-form-field>\r\n }\r\n\r\n <!-- reset button -->\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"warn\"\r\n (click)=\"reset()\"\r\n matTooltip=\"Remove all the heights\"\r\n >\r\n <mat-icon class=\"mat-warn\">clear</mat-icon>\r\n </button>\r\n</div>\r\n}\r\n", styles: [".input-nr{width:5em}.form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}\n"] }]
|
|
119
|
-
}], ctorParameters: () => [{ type: i1.FormBuilder }], propDecorators: { lineCount: [{
|
|
120
|
-
type: Input
|
|
121
|
-
}], heights: [{
|
|
122
|
-
type: Input
|
|
123
|
-
}], heightsChange: [{
|
|
124
|
-
type: Output
|
|
125
|
-
}] } });
|
|
126
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ln-heights-editor.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/gve-core/src/lib/components/ln-heights-editor/ln-heights-editor.component.ts","../../../../../../../projects/myrmidon/gve-core/src/lib/components/ln-heights-editor/ln-heights-editor.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 { MatInputModule } from '@angular/material/input';\r\nimport { MatSelectModule } from '@angular/material/select';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\n\r\n/**\r\n * 🔑 `gve-ln-heights-editor`\r\n *\r\n * A component to edit line heights.\r\n * Used by the `gve-snapshot-editor` component.\r\n *\r\n * - ▶️ `lineCount` (`number`): the number of lines.\r\n * - ▶️ `heights` (`Record<number, number>`): the line heights.\r\n * - 🔥 `heightsChange` (`EventEmitter<Record<number, number>>`): the event\r\n * emitted when the heights change.\r\n */\r\n@Component({\r\n  selector: 'gve-ln-heights-editor',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    ReactiveFormsModule,\r\n    MatButtonModule,\r\n    MatFormFieldModule,\r\n    MatIconModule,\r\n    MatInputModule,\r\n    MatSelectModule,\r\n    MatTooltipModule,\r\n  ],\r\n  templateUrl: './ln-heights-editor.component.html',\r\n  styleUrl: './ln-heights-editor.component.css',\r\n})\r\nexport class LnHeightsEditorComponent implements OnInit, OnDestroy {\r\n  private _subs?: Subscription[];\r\n  private _heights?: Record<number, number>;\r\n  private _lineCount: number = 0;\r\n\r\n  public lineNumber: FormControl<number>;\r\n  public height: FormControl<number>;\r\n\r\n  /**\r\n   * The total number of lines in the text.\r\n   */\r\n  @Input()\r\n  public get lineCount(): number {\r\n    return this._lineCount;\r\n  }\r\n  public set lineCount(value: number) {\r\n    if (this._lineCount === value) return;\r\n    this._lineCount = value;\r\n    this.lineNumbers = Array.from({ length: value }, (_, i) => i + 1);\r\n  }\r\n\r\n  /**\r\n   * The heights map of the lines. Each key is a line number and the value is\r\n   * the height of the line.\r\n   */\r\n  @Input()\r\n  public get heights(): Record<number, number> | undefined {\r\n    return this._heights;\r\n  }\r\n  public set heights(value: Record<number, number> | undefined | null) {\r\n    if (this._heights === value) return;\r\n    this._heights = value || undefined;\r\n  }\r\n\r\n  /**\r\n   * The event emitted when the heights change.\r\n   */\r\n  @Output()\r\n  public readonly heightsChange: EventEmitter<\r\n    Record<number, number> | undefined\r\n  > = new EventEmitter();\r\n\r\n  public lineNumbers: number[] = [];\r\n\r\n  constructor(formBuilder: FormBuilder) {\r\n    this.lineNumber = formBuilder.control(0, { nonNullable: true });\r\n    this.height = formBuilder.control(0, { nonNullable: true });\r\n  }\r\n\r\n  private pruneHeights(): void {\r\n    // remove all the heigths with value=0\r\n    this._heights = Object.fromEntries(\r\n      Object.entries(this._heights || {}).filter(([_, v]) => v !== 0)\r\n    );\r\n    // if heights are now empty, set to undefined\r\n    if (Object.keys(this._heights).length === 0) {\r\n      this._heights = undefined;\r\n    }\r\n  }\r\n\r\n  public ngOnInit(): void {\r\n    this._subs = [];\r\n    // update heights[ln] when height changes and emit heightsChange\r\n    this._subs.push(\r\n      this.height.valueChanges\r\n        .pipe(distinctUntilChanged(), debounceTime(200))\r\n        .subscribe((value) => {\r\n          if (!this._heights) {\r\n            this._heights = {};\r\n          }\r\n          this._heights[this.lineNumber.value] = value;\r\n          this.pruneHeights();\r\n          this.heightsChange.emit(this._heights);\r\n        })\r\n    );\r\n\r\n    // update height when line number changes\r\n    this._subs.push(\r\n      this.lineNumber.valueChanges\r\n        .pipe(distinctUntilChanged(), debounceTime(200))\r\n        .subscribe((value) => {\r\n          if (!this._heights) return;\r\n          this.height.setValue(this._heights[value] || 0);\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  public reset(): void {\r\n    this._heights = undefined;\r\n    this.heightsChange.emit(undefined);\r\n  }\r\n}\r\n","@if (lineNumbers.length) {\r\n<div class=\"form-row\">\r\n  <!-- line number -->\r\n  <mat-form-field class=\"input-nr\">\r\n    <mat-label>line</mat-label>\r\n    <mat-select [formControl]=\"lineNumber\">\r\n      <mat-option *ngFor=\"let n of lineNumbers\" [value]=\"n\">\r\n        {{ n }}\r\n      </mat-option>\r\n    </mat-select>\r\n  </mat-form-field>\r\n\r\n  <!-- height -->\r\n  @if (lineNumber.value) {\r\n  <mat-form-field class=\"input-nr\">\r\n    <mat-label>height</mat-label>\r\n    <input matInput type=\"number\" [formControl]=\"height\" min=\"0\" />\r\n  </mat-form-field>\r\n  }\r\n\r\n  <!-- reset button -->\r\n  <button\r\n    type=\"button\"\r\n    mat-icon-button\r\n    color=\"warn\"\r\n    (click)=\"reset()\"\r\n    matTooltip=\"Remove all the heights\"\r\n  >\r\n    <mat-icon class=\"mat-warn\">clear</mat-icon>\r\n  </button>\r\n</div>\r\n}\r\n"]}
|