@myrmidon/gve-core 0.0.5 → 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (35) hide show
  1. package/README.md +2 -0
  2. package/fesm2022/myrmidon-gve-core.mjs +370 -259
  3. package/fesm2022/myrmidon-gve-core.mjs.map +1 -1
  4. package/lib/components/batch-operation-editor/batch-operation-editor.component.d.ts +46 -0
  5. package/lib/components/chain-result-view/chain-result-view.component.d.ts +4 -1
  6. package/lib/components/feature-editor/feature-editor.component.d.ts +1 -1
  7. package/lib/components/feature-set-editor/feature-set-editor.component.d.ts +1 -1
  8. package/lib/components/snapshot-editor/snapshot-editor.component.d.ts +11 -9
  9. package/lib/components/steps-map/steps-map.component.d.ts +2 -0
  10. package/lib/models.d.ts +8 -0
  11. package/lib/services/gve-api.service.d.ts +33 -0
  12. package/package.json +10 -12
  13. package/public-api.d.ts +1 -0
  14. package/esm2022/lib/components/animation-timeline/animation-timeline.component.mjs +0 -207
  15. package/esm2022/lib/components/animation-timeline-set/animation-timeline-set.component.mjs +0 -154
  16. package/esm2022/lib/components/animation-tween/animation-tween.component.mjs +0 -184
  17. package/esm2022/lib/components/base-text-char/base-text-char.component.mjs +0 -46
  18. package/esm2022/lib/components/base-text-editor/base-text-editor.component.mjs +0 -115
  19. package/esm2022/lib/components/base-text-view/base-text-view.component.mjs +0 -159
  20. package/esm2022/lib/components/chain-operation-editor/chain-operation-editor.component.mjs +0 -570
  21. package/esm2022/lib/components/chain-result-view/chain-result-view.component.mjs +0 -222
  22. package/esm2022/lib/components/feature-editor/feature-editor.component.mjs +0 -200
  23. package/esm2022/lib/components/feature-set-editor/feature-set-editor.component.mjs +0 -175
  24. package/esm2022/lib/components/feature-set-view/feature-set-view.component.mjs +0 -100
  25. package/esm2022/lib/components/ln-heights-editor/ln-heights-editor.component.mjs +0 -126
  26. package/esm2022/lib/components/operation-source-editor/operation-source-editor.component.mjs +0 -131
  27. package/esm2022/lib/components/simple-tree/simple-tree.component.mjs +0 -72
  28. package/esm2022/lib/components/snapshot-editor/snapshot-editor.component.mjs +0 -863
  29. package/esm2022/lib/components/steps-map/steps-map.component.mjs +0 -77
  30. package/esm2022/lib/models.mjs +0 -2
  31. package/esm2022/lib/services/gve-api.service.mjs +0 -65
  32. package/esm2022/lib/services/settings.service.mjs +0 -87
  33. package/esm2022/lib/validators/svg-validators.mjs +0 -34
  34. package/esm2022/myrmidon-gve-core.mjs +0 -5
  35. package/esm2022/public-api.mjs +0 -22
@@ -1,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.8", 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.8", 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.8", 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.8", 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.8", 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 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 <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.8", 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 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 <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;8GArEU,uBAAuB;kGAAvB,uBAAuB,wMCnDpC,8jDAuDA,6YDfI,YAAY,8BACZ,mBAAmB,ykBACnB,eAAe,2IACf,kBAAkB,2aAClB,aAAa,mLACb,cAAc,0WACd,aAAa;;2FAKJ,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 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      <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.8", 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.8", 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.8", 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"]}