@myrmidon/gve-core 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +83 -0
- package/esm2022/lib/components/animation-timeline/animation-timeline.component.mjs +176 -0
- package/esm2022/lib/components/animation-tween/animation-tween.component.mjs +161 -0
- package/esm2022/lib/components/animation-vars/animation-vars.component.mjs +141 -0
- package/esm2022/lib/components/base-text-char/base-text-char.component.mjs +37 -0
- package/esm2022/lib/components/base-text-editor/base-text-editor.component.mjs +111 -0
- package/esm2022/lib/components/base-text-view/base-text-view.component.mjs +125 -0
- package/esm2022/lib/components/chain-operation-editor/chain-operation-editor.component.mjs +512 -0
- package/esm2022/lib/components/chain-result-view/chain-result-view.component.mjs +190 -0
- package/esm2022/lib/components/feature-editor/feature-editor.component.mjs +185 -0
- package/esm2022/lib/components/feature-set-editor/feature-set-editor.component.mjs +152 -0
- package/esm2022/lib/components/feature-set-view/feature-set-view.component.mjs +80 -0
- package/esm2022/lib/components/ln-heights-editor/ln-heights-editor.component.mjs +95 -0
- package/esm2022/lib/components/operation-source-editor/operation-source-editor.component.mjs +106 -0
- package/esm2022/lib/components/simple-tree/simple-tree.component.mjs +56 -0
- package/esm2022/lib/components/snapshot-editor/snapshot-editor.component.mjs +510 -0
- package/esm2022/lib/components/steps-map/steps-map.component.mjs +55 -0
- package/esm2022/lib/models.mjs +2 -0
- package/esm2022/lib/services/gve-api.service.mjs +52 -0
- package/esm2022/lib/services/settings.service.mjs +86 -0
- package/esm2022/lib/validators/svg-validators.mjs +28 -0
- package/esm2022/myrmidon-gve-core.mjs +5 -0
- package/esm2022/public-api.mjs +23 -0
- package/fesm2022/myrmidon-gve-core.mjs +2592 -0
- package/fesm2022/myrmidon-gve-core.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/components/animation-timeline/animation-timeline.component.d.ts +42 -0
- package/lib/components/animation-tween/animation-tween.component.d.ts +47 -0
- package/lib/components/animation-vars/animation-vars.component.d.ts +30 -0
- package/lib/components/base-text-char/base-text-char.component.d.ts +29 -0
- package/lib/components/base-text-editor/base-text-editor.component.d.ts +40 -0
- package/lib/components/base-text-view/base-text-view.component.d.ts +29 -0
- package/lib/components/chain-operation-editor/chain-operation-editor.component.d.ts +102 -0
- package/lib/components/chain-result-view/chain-result-view.component.d.ts +44 -0
- package/lib/components/feature-editor/feature-editor.component.d.ts +72 -0
- package/lib/components/feature-set-editor/feature-set-editor.component.d.ts +55 -0
- package/lib/components/feature-set-view/feature-set-view.component.d.ts +36 -0
- package/lib/components/ln-heights-editor/ln-heights-editor.component.d.ts +22 -0
- package/lib/components/operation-source-editor/operation-source-editor.component.d.ts +24 -0
- package/lib/components/simple-tree/simple-tree.component.d.ts +16 -0
- package/lib/components/snapshot-editor/snapshot-editor.component.d.ts +105 -0
- package/lib/components/steps-map/steps-map.component.d.ts +29 -0
- package/lib/models.d.ts +7 -0
- package/lib/services/gve-api.service.d.ts +57 -0
- package/lib/services/settings.service.d.ts +53 -0
- package/lib/validators/svg-validators.d.ts +4 -0
- package/package.json +45 -0
- package/public-api.d.ts +19 -0
|
@@ -0,0 +1,141 @@
|
|
|
1
|
+
import { Component, EventEmitter, Input, Output, ViewChild, } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { ReactiveFormsModule, Validators, } from '@angular/forms';
|
|
4
|
+
import { MatButtonModule } from '@angular/material/button';
|
|
5
|
+
import { MatCheckboxModule } from '@angular/material/checkbox';
|
|
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/tooltip";
|
|
19
|
+
export class AnimationVarsComponent {
|
|
20
|
+
get vars() {
|
|
21
|
+
return this._vars;
|
|
22
|
+
}
|
|
23
|
+
set vars(value) {
|
|
24
|
+
if (this._vars === value) {
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
this._vars = value;
|
|
28
|
+
this.updateForm(value);
|
|
29
|
+
}
|
|
30
|
+
constructor(formBuilder) {
|
|
31
|
+
this._vars = {};
|
|
32
|
+
this.varsChange = new EventEmitter();
|
|
33
|
+
this.editedVars = [];
|
|
34
|
+
this.name = formBuilder.control('', {
|
|
35
|
+
nonNullable: true,
|
|
36
|
+
validators: [Validators.required, Validators.maxLength(100)],
|
|
37
|
+
});
|
|
38
|
+
this.value = formBuilder.control('', {
|
|
39
|
+
nonNullable: true,
|
|
40
|
+
validators: [Validators.required, Validators.maxLength(1000)],
|
|
41
|
+
});
|
|
42
|
+
this.form = formBuilder.group({
|
|
43
|
+
name: this.name,
|
|
44
|
+
value: this.value,
|
|
45
|
+
});
|
|
46
|
+
}
|
|
47
|
+
updateForm(vars) {
|
|
48
|
+
this.editedVars = Object.entries(vars)
|
|
49
|
+
.map(([name, value]) => ({ name, value }))
|
|
50
|
+
.sort((a, b) => a.name.localeCompare(b.name));
|
|
51
|
+
}
|
|
52
|
+
getVars() {
|
|
53
|
+
return this.editedVars.reduce((acc, { name, value }) => ({ ...acc, [name]: value }), {});
|
|
54
|
+
}
|
|
55
|
+
addVar() {
|
|
56
|
+
this.form.reset();
|
|
57
|
+
// focus the name input
|
|
58
|
+
setTimeout(() => this.nameInput?.nativeElement.focus(), 0);
|
|
59
|
+
}
|
|
60
|
+
editVar(index) {
|
|
61
|
+
const { name, value } = this.editedVars[index];
|
|
62
|
+
this.form.setValue({ name, value });
|
|
63
|
+
this.form.markAsPristine();
|
|
64
|
+
// focus the name input
|
|
65
|
+
setTimeout(() => this.nameInput?.nativeElement.focus(), 0);
|
|
66
|
+
}
|
|
67
|
+
deleteVar(index) {
|
|
68
|
+
this.editedVars = this.editedVars.filter((_, i) => i !== index);
|
|
69
|
+
this.varsChange.emit(this.getVars());
|
|
70
|
+
}
|
|
71
|
+
parseValue(value) {
|
|
72
|
+
if (!value) {
|
|
73
|
+
return '';
|
|
74
|
+
}
|
|
75
|
+
if (value === 'true') {
|
|
76
|
+
return true;
|
|
77
|
+
}
|
|
78
|
+
else if (value === 'false') {
|
|
79
|
+
return false;
|
|
80
|
+
}
|
|
81
|
+
else if (/^\d+$/.test(value)) {
|
|
82
|
+
return parseInt(value, 10);
|
|
83
|
+
}
|
|
84
|
+
else if (/^\d+\.\d+$/.test(value)) {
|
|
85
|
+
return parseFloat(value);
|
|
86
|
+
}
|
|
87
|
+
return value;
|
|
88
|
+
}
|
|
89
|
+
saveVar() {
|
|
90
|
+
if (this.form.invalid) {
|
|
91
|
+
return;
|
|
92
|
+
}
|
|
93
|
+
const { name, value } = this.form.value;
|
|
94
|
+
// parse value assuming that if it looks like a boolean/number
|
|
95
|
+
// its value should be a boolean/number rather than a string
|
|
96
|
+
const v = this.parseValue(value);
|
|
97
|
+
const editedVars = [...this.editedVars];
|
|
98
|
+
// if name already exists, update the value; else add it
|
|
99
|
+
const existingIndex = this.editedVars.findIndex((vr) => vr.name === name);
|
|
100
|
+
if (existingIndex !== -1) {
|
|
101
|
+
editedVars[existingIndex] = { name, value: v };
|
|
102
|
+
this.editedVars = editedVars;
|
|
103
|
+
return;
|
|
104
|
+
}
|
|
105
|
+
else {
|
|
106
|
+
editedVars.push({ name, value: v });
|
|
107
|
+
this.editedVars = editedVars.sort((a, b) => a.name.localeCompare(b.name));
|
|
108
|
+
}
|
|
109
|
+
this.varsChange.emit(this.getVars());
|
|
110
|
+
setTimeout(() => {
|
|
111
|
+
this.nameInput?.nativeElement.focus();
|
|
112
|
+
this.form.reset();
|
|
113
|
+
this.form.markAsPristine();
|
|
114
|
+
this.form.updateValueAndValidity();
|
|
115
|
+
}, 0);
|
|
116
|
+
}
|
|
117
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: AnimationVarsComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
118
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.4", type: AnimationVarsComponent, isStandalone: true, selector: "gve-animation-vars", inputs: { vars: "vars" }, outputs: { varsChange: "varsChange" }, viewQueries: [{ propertyName: "nameInput", first: true, predicate: ["vname"], descendants: true, static: true }], ngImport: i0, template: "<div>\r\n <div>\r\n <button type=\"button\" mat-raised-button color=\"primary\" (click)=\"addVar()\">\r\n <mat-icon>add</mat-icon>\r\n property\r\n </button>\r\n </div>\r\n\r\n @if (editedVars.length) {\r\n <table>\r\n <thead>\r\n <tr>\r\n <th></th>\r\n <th>name</th>\r\n <th>value</th>\r\n </tr>\r\n </thead>\r\n <tbody>\r\n @for (v of editedVars; track v.name; let index=$index) {\r\n <tr>\r\n <td class=\"fit-width\">\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n (click)=\"editVar(index)\"\r\n matTooltip=\"Edit this property\"\r\n >\r\n <mat-icon class=\"mat-primary\">edit</mat-icon>\r\n </button>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"warn\"\r\n (click)=\"deleteVar(index)\"\r\n matTooltip=\"Delete this property\"\r\n >\r\n <mat-icon class=\"mat-warn\">remove_circle</mat-icon>\r\n </button>\r\n </td>\r\n <td>{{ v.name }}</td>\r\n <td>{{ v.value }}</td>\r\n </tr>\r\n }\r\n </tbody>\r\n </table>\r\n } @else {\r\n <p class=\"muted\">(no vars)</p>\r\n }\r\n\r\n <form [formGroup]=\"form\" (submit)=\"saveVar()\">\r\n <fieldset>\r\n <div class=\"form-row\">\r\n <!-- name -->\r\n <mat-form-field>\r\n <mat-label>name</mat-label>\r\n <input matInput [formControl]=\"name\" #vname />\r\n <mat-error\r\n *ngIf=\"$any(name).errors?.required && (name.dirty || name.touched)\"\r\n >name required</mat-error\r\n >\r\n <mat-error\r\n *ngIf=\"$any(name).errors?.maxLength && (name.dirty || name.touched)\"\r\n >name too long</mat-error\r\n >\r\n </mat-form-field>\r\n\r\n <!-- value -->\r\n <mat-form-field>\r\n <mat-label>value</mat-label>\r\n <input matInput [formControl]=\"value\" />\r\n <mat-error\r\n *ngIf=\"\r\n $any(value).errors?.required && (value.dirty || value.touched)\r\n \"\r\n >value required</mat-error\r\n >\r\n <mat-error\r\n *ngIf=\"\r\n $any(value).errors?.maxLength && (value.dirty || value.touched)\r\n \"\r\n >value too long</mat-error\r\n >\r\n </mat-form-field>\r\n\r\n <button\r\n type=\"submit\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [disabled]=\"form.invalid\"\r\n >\r\n <mat-icon class=\"mat-primary\">add_circle</mat-icon>\r\n </button>\r\n </div>\r\n </fieldset>\r\n </form>\r\n</div>\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}fieldset{border:1px solid silver;border-radius:4px;padding:8px;margin:8px 0}table{width:100%;border-collapse:collapse}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: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i3.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i3.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatCheckboxModule }, { 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: "directive", type: i4.MatError, selector: "mat-error, [matError]", inputs: ["id"] }, { 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"] }] }); }
|
|
119
|
+
}
|
|
120
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: AnimationVarsComponent, decorators: [{
|
|
121
|
+
type: Component,
|
|
122
|
+
args: [{ selector: 'gve-animation-vars', standalone: true, imports: [
|
|
123
|
+
CommonModule,
|
|
124
|
+
ReactiveFormsModule,
|
|
125
|
+
MatButtonModule,
|
|
126
|
+
MatCheckboxModule,
|
|
127
|
+
MatFormFieldModule,
|
|
128
|
+
MatIconModule,
|
|
129
|
+
MatInputModule,
|
|
130
|
+
MatSelectModule,
|
|
131
|
+
MatTooltipModule,
|
|
132
|
+
], template: "<div>\r\n <div>\r\n <button type=\"button\" mat-raised-button color=\"primary\" (click)=\"addVar()\">\r\n <mat-icon>add</mat-icon>\r\n property\r\n </button>\r\n </div>\r\n\r\n @if (editedVars.length) {\r\n <table>\r\n <thead>\r\n <tr>\r\n <th></th>\r\n <th>name</th>\r\n <th>value</th>\r\n </tr>\r\n </thead>\r\n <tbody>\r\n @for (v of editedVars; track v.name; let index=$index) {\r\n <tr>\r\n <td class=\"fit-width\">\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n (click)=\"editVar(index)\"\r\n matTooltip=\"Edit this property\"\r\n >\r\n <mat-icon class=\"mat-primary\">edit</mat-icon>\r\n </button>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"warn\"\r\n (click)=\"deleteVar(index)\"\r\n matTooltip=\"Delete this property\"\r\n >\r\n <mat-icon class=\"mat-warn\">remove_circle</mat-icon>\r\n </button>\r\n </td>\r\n <td>{{ v.name }}</td>\r\n <td>{{ v.value }}</td>\r\n </tr>\r\n }\r\n </tbody>\r\n </table>\r\n } @else {\r\n <p class=\"muted\">(no vars)</p>\r\n }\r\n\r\n <form [formGroup]=\"form\" (submit)=\"saveVar()\">\r\n <fieldset>\r\n <div class=\"form-row\">\r\n <!-- name -->\r\n <mat-form-field>\r\n <mat-label>name</mat-label>\r\n <input matInput [formControl]=\"name\" #vname />\r\n <mat-error\r\n *ngIf=\"$any(name).errors?.required && (name.dirty || name.touched)\"\r\n >name required</mat-error\r\n >\r\n <mat-error\r\n *ngIf=\"$any(name).errors?.maxLength && (name.dirty || name.touched)\"\r\n >name too long</mat-error\r\n >\r\n </mat-form-field>\r\n\r\n <!-- value -->\r\n <mat-form-field>\r\n <mat-label>value</mat-label>\r\n <input matInput [formControl]=\"value\" />\r\n <mat-error\r\n *ngIf=\"\r\n $any(value).errors?.required && (value.dirty || value.touched)\r\n \"\r\n >value required</mat-error\r\n >\r\n <mat-error\r\n *ngIf=\"\r\n $any(value).errors?.maxLength && (value.dirty || value.touched)\r\n \"\r\n >value too long</mat-error\r\n >\r\n </mat-form-field>\r\n\r\n <button\r\n type=\"submit\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [disabled]=\"form.invalid\"\r\n >\r\n <mat-icon class=\"mat-primary\">add_circle</mat-icon>\r\n </button>\r\n </div>\r\n </fieldset>\r\n </form>\r\n</div>\r\n", styles: [".form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}fieldset{border:1px solid silver;border-radius:4px;padding:8px;margin:8px 0}table{width:100%;border-collapse:collapse}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"] }]
|
|
133
|
+
}], ctorParameters: () => [{ type: i1.FormBuilder }], propDecorators: { vars: [{
|
|
134
|
+
type: Input
|
|
135
|
+
}], varsChange: [{
|
|
136
|
+
type: Output
|
|
137
|
+
}], nameInput: [{
|
|
138
|
+
type: ViewChild,
|
|
139
|
+
args: ['vname', { static: true }]
|
|
140
|
+
}] } });
|
|
141
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"animation-vars.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/gve-core/src/lib/components/animation-vars/animation-vars.component.ts","../../../../../../../projects/myrmidon/gve-core/src/lib/components/animation-vars/animation-vars.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  ElementRef,\r\n  EventEmitter,\r\n  Input,\r\n  Output,\r\n  ViewChild,\r\n} from '@angular/core';\r\n\r\nimport { CommonModule } from '@angular/common';\r\nimport {\r\n  FormBuilder,\r\n  FormControl,\r\n  FormGroup,\r\n  ReactiveFormsModule,\r\n  Validators,\r\n} from '@angular/forms';\r\nimport { MatButtonModule } from '@angular/material/button';\r\nimport { MatCheckboxModule } from '@angular/material/checkbox';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatIconModule } from '@angular/material/icon';\r\nimport { MatInputModule } from '@angular/material/input';\r\nimport { MatSelectModule } from '@angular/material/select';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\n\r\nimport { GveAnimationVars } from '@myrmidon/gve-snapshot-view';\r\n\r\ninterface VarViewModel {\r\n  name: string;\r\n  value: string | number | boolean;\r\n}\r\n\r\n@Component({\r\n  selector: 'gve-animation-vars',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    ReactiveFormsModule,\r\n    MatButtonModule,\r\n    MatCheckboxModule,\r\n    MatFormFieldModule,\r\n    MatIconModule,\r\n    MatInputModule,\r\n    MatSelectModule,\r\n    MatTooltipModule,\r\n  ],\r\n  templateUrl: './animation-vars.component.html',\r\n  styleUrl: './animation-vars.component.css',\r\n})\r\nexport class AnimationVarsComponent {\r\n  private _vars: GveAnimationVars = {};\r\n\r\n  @Input()\r\n  public get vars(): GveAnimationVars {\r\n    return this._vars;\r\n  }\r\n  public set vars(value: GveAnimationVars) {\r\n    if (this._vars === value) {\r\n      return;\r\n    }\r\n    this._vars = value;\r\n    this.updateForm(value);\r\n  }\r\n\r\n  @Output()\r\n  public varsChange: EventEmitter<GveAnimationVars> =\r\n    new EventEmitter<GveAnimationVars>();\r\n\r\n  @ViewChild('vname', { static: true })\r\n  public nameInput?: ElementRef;\r\n\r\n  public name: FormControl<string>;\r\n  public value: FormControl<string>;\r\n  public form: FormGroup;\r\n\r\n  public editedVars: VarViewModel[] = [];\r\n\r\n  constructor(formBuilder: FormBuilder) {\r\n    this.name = formBuilder.control<string>('', {\r\n      nonNullable: true,\r\n      validators: [Validators.required, Validators.maxLength(100)],\r\n    });\r\n    this.value = formBuilder.control<string>('', {\r\n      nonNullable: true,\r\n      validators: [Validators.required, Validators.maxLength(1000)],\r\n    });\r\n    this.form = formBuilder.group({\r\n      name: this.name,\r\n      value: this.value,\r\n    });\r\n  }\r\n\r\n  private updateForm(vars: GveAnimationVars): void {\r\n    this.editedVars = Object.entries(vars)\r\n      .map(([name, value]) => ({ name, value }))\r\n      .sort((a, b) => a.name.localeCompare(b.name));\r\n  }\r\n\r\n  private getVars(): GveAnimationVars {\r\n    return this.editedVars.reduce(\r\n      (acc, { name, value }) => ({ ...acc, [name]: value }),\r\n      {}\r\n    );\r\n  }\r\n\r\n  public addVar(): void {\r\n    this.form.reset();\r\n    // focus the name input\r\n    setTimeout(() => this.nameInput?.nativeElement.focus(), 0);\r\n  }\r\n\r\n  public editVar(index: number): void {\r\n    const { name, value } = this.editedVars[index];\r\n    this.form.setValue({ name, value });\r\n    this.form.markAsPristine();\r\n    // focus the name input\r\n    setTimeout(() => this.nameInput?.nativeElement.focus(), 0);\r\n  }\r\n\r\n  public deleteVar(index: number): void {\r\n    this.editedVars = this.editedVars.filter((_, i) => i !== index);\r\n    this.varsChange.emit(this.getVars());\r\n  }\r\n\r\n  private parseValue(value: string): string | number | boolean {\r\n    if (!value) {\r\n      return '';\r\n    }\r\n    if (value === 'true') {\r\n      return true;\r\n    } else if (value === 'false') {\r\n      return false;\r\n    } else if (/^\\d+$/.test(value)) {\r\n      return parseInt(value, 10);\r\n    } else if (/^\\d+\\.\\d+$/.test(value)) {\r\n      return parseFloat(value);\r\n    }\r\n    return value;\r\n  }\r\n\r\n  public saveVar(): void {\r\n    if (this.form.invalid) {\r\n      return;\r\n    }\r\n    const { name, value } = this.form.value;\r\n\r\n    // parse value assuming that if it looks like a boolean/number\r\n    // its value should be a boolean/number rather than a string\r\n    const v = this.parseValue(value);\r\n\r\n    const editedVars = [...this.editedVars];\r\n\r\n    // if name already exists, update the value; else add it\r\n    const existingIndex = this.editedVars.findIndex((vr) => vr.name === name);\r\n    if (existingIndex !== -1) {\r\n      editedVars[existingIndex] = { name, value: v };\r\n      this.editedVars = editedVars;\r\n      return;\r\n    } else {\r\n      editedVars.push({ name, value: v });\r\n      this.editedVars = editedVars.sort((a, b) => a.name.localeCompare(b.name));\r\n    }\r\n    this.varsChange.emit(this.getVars());\r\n\r\n    setTimeout(() => {\r\n      this.nameInput?.nativeElement.focus();\r\n      this.form.reset();\r\n      this.form.markAsPristine();\r\n      this.form.updateValueAndValidity();\r\n    }, 0);\r\n  }\r\n}\r\n","<div>\r\n  <div>\r\n    <button type=\"button\" mat-raised-button color=\"primary\" (click)=\"addVar()\">\r\n      <mat-icon>add</mat-icon>\r\n      property\r\n    </button>\r\n  </div>\r\n\r\n  @if (editedVars.length) {\r\n  <table>\r\n    <thead>\r\n      <tr>\r\n        <th></th>\r\n        <th>name</th>\r\n        <th>value</th>\r\n      </tr>\r\n    </thead>\r\n    <tbody>\r\n      @for (v of editedVars; track v.name; let index=$index) {\r\n      <tr>\r\n        <td class=\"fit-width\">\r\n          <button\r\n            type=\"button\"\r\n            mat-icon-button\r\n            color=\"primary\"\r\n            (click)=\"editVar(index)\"\r\n            matTooltip=\"Edit this property\"\r\n          >\r\n            <mat-icon class=\"mat-primary\">edit</mat-icon>\r\n          </button>\r\n          <button\r\n            type=\"button\"\r\n            mat-icon-button\r\n            color=\"warn\"\r\n            (click)=\"deleteVar(index)\"\r\n            matTooltip=\"Delete this property\"\r\n          >\r\n            <mat-icon class=\"mat-warn\">remove_circle</mat-icon>\r\n          </button>\r\n        </td>\r\n        <td>{{ v.name }}</td>\r\n        <td>{{ v.value }}</td>\r\n      </tr>\r\n      }\r\n    </tbody>\r\n  </table>\r\n  } @else {\r\n  <p class=\"muted\">(no vars)</p>\r\n  }\r\n\r\n  <form [formGroup]=\"form\" (submit)=\"saveVar()\">\r\n    <fieldset>\r\n      <div class=\"form-row\">\r\n        <!-- name -->\r\n        <mat-form-field>\r\n          <mat-label>name</mat-label>\r\n          <input matInput [formControl]=\"name\" #vname />\r\n          <mat-error\r\n            *ngIf=\"$any(name).errors?.required && (name.dirty || name.touched)\"\r\n            >name required</mat-error\r\n          >\r\n          <mat-error\r\n            *ngIf=\"$any(name).errors?.maxLength && (name.dirty || name.touched)\"\r\n            >name too long</mat-error\r\n          >\r\n        </mat-form-field>\r\n\r\n        <!-- value -->\r\n        <mat-form-field>\r\n          <mat-label>value</mat-label>\r\n          <input matInput [formControl]=\"value\" />\r\n          <mat-error\r\n            *ngIf=\"\r\n              $any(value).errors?.required && (value.dirty || value.touched)\r\n            \"\r\n            >value required</mat-error\r\n          >\r\n          <mat-error\r\n            *ngIf=\"\r\n              $any(value).errors?.maxLength && (value.dirty || value.touched)\r\n            \"\r\n            >value too long</mat-error\r\n          >\r\n        </mat-form-field>\r\n\r\n        <button\r\n          type=\"submit\"\r\n          mat-icon-button\r\n          color=\"primary\"\r\n          [disabled]=\"form.invalid\"\r\n        >\r\n          <mat-icon class=\"mat-primary\">add_circle</mat-icon>\r\n        </button>\r\n      </div>\r\n    </fieldset>\r\n  </form>\r\n</div>\r\n"]}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
2
|
+
import { MatRippleModule } from '@angular/material/core';
|
|
3
|
+
import { NgToolsModule } from '@myrmidon/ng-tools';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@angular/material/core";
|
|
6
|
+
import * as i2 from "@myrmidon/ng-tools";
|
|
7
|
+
/**
|
|
8
|
+
* A component that displays a single character from a base text.
|
|
9
|
+
*/
|
|
10
|
+
export class BaseTextCharComponent {
|
|
11
|
+
constructor() {
|
|
12
|
+
this.defaultColor = '#D8D8D8';
|
|
13
|
+
this.defaultBorderColor = '#D8D8D8';
|
|
14
|
+
this.defaultEmSize = 1.5;
|
|
15
|
+
this.charPick = new EventEmitter();
|
|
16
|
+
}
|
|
17
|
+
get char() {
|
|
18
|
+
return this._char;
|
|
19
|
+
}
|
|
20
|
+
set char(value) {
|
|
21
|
+
this._char = value || undefined;
|
|
22
|
+
}
|
|
23
|
+
onCharClick(event) {
|
|
24
|
+
this.charPick.emit({ char: this._char, event });
|
|
25
|
+
}
|
|
26
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: BaseTextCharComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
27
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.4", type: BaseTextCharComponent, isStandalone: true, selector: "gve-base-text-char", inputs: { char: "char" }, outputs: { charPick: "charPick" }, ngImport: i0, template: "@if (char) {\r\n<div matRipple id=\"container\" (click)=\"onCharClick($event)\">\r\n <div\r\n id=\"c-label\"\r\n [style.fontSize]=\"char.emSize + 'em'\"\r\n [style.borderColor]=\"char.borderColor\"\r\n >\r\n {{ char.label }}\r\n </div>\r\n <div\r\n id=\"c-id\"\r\n [style.fontSize]=\"char.emSize / 2 + 'em'\"\r\n [style.color]=\"char.color | colorToContrast\"\r\n [style.borderColor]=\"char.borderColor\"\r\n [style.backgroundColor]=\"char.color\"\r\n >\r\n {{ char.id }}\r\n </div>\r\n</div>\r\n}\r\n", styles: ["div#container{cursor:pointer;flex-direction:column;align-items:center}div#c-label{border:1px solid silver;border-radius:6px;padding:6px;height:1.5em;align-items:center;justify-content:center;text-align:center}div#c-id{margin-top:4px;margin-bottom:16px;border:1px solid silver;border-radius:6px;padding:6px;align-items:center;justify-content:center;text-align:center}\n"], dependencies: [{ kind: "ngmodule", type: MatRippleModule }, { kind: "directive", type: i1.MatRipple, selector: "[mat-ripple], [matRipple]", inputs: ["matRippleColor", "matRippleUnbounded", "matRippleCentered", "matRippleRadius", "matRippleAnimation", "matRippleDisabled", "matRippleTrigger"], exportAs: ["matRipple"] }, { kind: "ngmodule", type: NgToolsModule }, { kind: "pipe", type: i2.ColorToContrastPipe, name: "colorToContrast" }] }); }
|
|
28
|
+
}
|
|
29
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: BaseTextCharComponent, decorators: [{
|
|
30
|
+
type: Component,
|
|
31
|
+
args: [{ selector: 'gve-base-text-char', standalone: true, imports: [MatRippleModule, NgToolsModule], template: "@if (char) {\r\n<div matRipple id=\"container\" (click)=\"onCharClick($event)\">\r\n <div\r\n id=\"c-label\"\r\n [style.fontSize]=\"char.emSize + 'em'\"\r\n [style.borderColor]=\"char.borderColor\"\r\n >\r\n {{ char.label }}\r\n </div>\r\n <div\r\n id=\"c-id\"\r\n [style.fontSize]=\"char.emSize / 2 + 'em'\"\r\n [style.color]=\"char.color | colorToContrast\"\r\n [style.borderColor]=\"char.borderColor\"\r\n [style.backgroundColor]=\"char.color\"\r\n >\r\n {{ char.id }}\r\n </div>\r\n</div>\r\n}\r\n", styles: ["div#container{cursor:pointer;flex-direction:column;align-items:center}div#c-label{border:1px solid silver;border-radius:6px;padding:6px;height:1.5em;align-items:center;justify-content:center;text-align:center}div#c-id{margin-top:4px;margin-bottom:16px;border:1px solid silver;border-radius:6px;padding:6px;align-items:center;justify-content:center;text-align:center}\n"] }]
|
|
32
|
+
}], propDecorators: { char: [{
|
|
33
|
+
type: Input
|
|
34
|
+
}], charPick: [{
|
|
35
|
+
type: Output
|
|
36
|
+
}] } });
|
|
37
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS10ZXh0LWNoYXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbXlybWlkb24vZ3ZlLWNvcmUvc3JjL2xpYi9jb21wb25lbnRzL2Jhc2UtdGV4dC1jaGFyL2Jhc2UtdGV4dC1jaGFyLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL215cm1pZG9uL2d2ZS1jb3JlL3NyYy9saWIvY29tcG9uZW50cy9iYXNlLXRleHQtY2hhci9iYXNlLXRleHQtY2hhci5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLFlBQVksRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3ZFLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUV6RCxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7Ozs7QUFnQm5EOztHQUVHO0FBUUgsTUFBTSxPQUFPLHFCQUFxQjtJQVBsQztRQVVTLGlCQUFZLEdBQUcsU0FBUyxDQUFDO1FBQ3pCLHVCQUFrQixHQUFHLFNBQVMsQ0FBQztRQUMvQixrQkFBYSxHQUFHLEdBQUcsQ0FBQztRQVdwQixhQUFRLEdBQ2IsSUFBSSxZQUFZLEVBQXFCLENBQUM7S0FLekM7SUFmQyxJQUNXLElBQUk7UUFDYixPQUFPLElBQUksQ0FBQyxLQUFLLENBQUM7SUFDcEIsQ0FBQztJQUNELElBQVcsSUFBSSxDQUFDLEtBQXNDO1FBQ3BELElBQUksQ0FBQyxLQUFLLEdBQUcsS0FBSyxJQUFJLFNBQVMsQ0FBQztJQUNsQyxDQUFDO0lBTU0sV0FBVyxDQUFDLEtBQWlCO1FBQ2xDLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLEVBQUUsSUFBSSxFQUFFLElBQUksQ0FBQyxLQUFNLEVBQUUsS0FBSyxFQUFFLENBQUMsQ0FBQztJQUNuRCxDQUFDOzhHQXJCVSxxQkFBcUI7a0dBQXJCLHFCQUFxQiwySUM3QmxDLDJoQkFvQkEseWFES1ksZUFBZSxrU0FBRSxhQUFhOzsyRkFJN0IscUJBQXFCO2tCQVBqQyxTQUFTOytCQUNFLG9CQUFvQixjQUNsQixJQUFJLFdBQ1AsQ0FBQyxlQUFlLEVBQUUsYUFBYSxDQUFDOzhCQVk5QixJQUFJO3NCQURkLEtBQUs7Z0JBU0MsUUFBUTtzQkFEZCxNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBFdmVudEVtaXR0ZXIsIElucHV0LCBPdXRwdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgTWF0UmlwcGxlTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvY29yZSc7XHJcblxyXG5pbXBvcnQgeyBOZ1Rvb2xzTW9kdWxlIH0gZnJvbSAnQG15cm1pZG9uL25nLXRvb2xzJztcclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgQmFzZVRleHRDaGFyIHtcclxuICBpZDogbnVtYmVyO1xyXG4gIHZhbHVlOiBzdHJpbmc7XHJcbiAgbGFiZWw6IHN0cmluZztcclxuICBjb2xvcjogc3RyaW5nO1xyXG4gIGJvcmRlckNvbG9yOiBzdHJpbmc7XHJcbiAgZW1TaXplOiBudW1iZXI7XHJcbn1cclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgQmFzZVRleHRDaGFyRXZlbnQge1xyXG4gIGNoYXI6IEJhc2VUZXh0Q2hhcjtcclxuICBldmVudDogTW91c2VFdmVudDtcclxufVxyXG5cclxuLyoqXHJcbiAqIEEgY29tcG9uZW50IHRoYXQgZGlzcGxheXMgYSBzaW5nbGUgY2hhcmFjdGVyIGZyb20gYSBiYXNlIHRleHQuXHJcbiAqL1xyXG5AQ29tcG9uZW50KHtcclxuICBzZWxlY3RvcjogJ2d2ZS1iYXNlLXRleHQtY2hhcicsXHJcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcclxuICBpbXBvcnRzOiBbTWF0UmlwcGxlTW9kdWxlLCBOZ1Rvb2xzTW9kdWxlXSxcclxuICB0ZW1wbGF0ZVVybDogJy4vYmFzZS10ZXh0LWNoYXIuY29tcG9uZW50Lmh0bWwnLFxyXG4gIHN0eWxlVXJsOiAnLi9iYXNlLXRleHQtY2hhci5jb21wb25lbnQuY3NzJyxcclxufSlcclxuZXhwb3J0IGNsYXNzIEJhc2VUZXh0Q2hhckNvbXBvbmVudCB7XHJcbiAgcHJpdmF0ZSBfY2hhcjogQmFzZVRleHRDaGFyIHwgdW5kZWZpbmVkO1xyXG5cclxuICBwdWJsaWMgZGVmYXVsdENvbG9yID0gJyNEOEQ4RDgnO1xyXG4gIHB1YmxpYyBkZWZhdWx0Qm9yZGVyQ29sb3IgPSAnI0Q4RDhEOCc7XHJcbiAgcHVibGljIGRlZmF1bHRFbVNpemUgPSAxLjU7XHJcblxyXG4gIEBJbnB1dCgpXHJcbiAgcHVibGljIGdldCBjaGFyKCk6IEJhc2VUZXh0Q2hhciB8IHVuZGVmaW5lZCB7XHJcbiAgICByZXR1cm4gdGhpcy5fY2hhcjtcclxuICB9XHJcbiAgcHVibGljIHNldCBjaGFyKHZhbHVlOiBCYXNlVGV4dENoYXIgfCB1bmRlZmluZWQgfCBudWxsKSB7XHJcbiAgICB0aGlzLl9jaGFyID0gdmFsdWUgfHwgdW5kZWZpbmVkO1xyXG4gIH1cclxuXHJcbiAgQE91dHB1dCgpXHJcbiAgcHVibGljIGNoYXJQaWNrOiBFdmVudEVtaXR0ZXI8QmFzZVRleHRDaGFyRXZlbnQ+ID1cclxuICAgIG5ldyBFdmVudEVtaXR0ZXI8QmFzZVRleHRDaGFyRXZlbnQ+KCk7XHJcblxyXG4gIHB1YmxpYyBvbkNoYXJDbGljayhldmVudDogTW91c2VFdmVudCk6IHZvaWQge1xyXG4gICAgdGhpcy5jaGFyUGljay5lbWl0KHsgY2hhcjogdGhpcy5fY2hhciEsIGV2ZW50IH0pO1xyXG4gIH1cclxufVxyXG4iLCJAaWYgKGNoYXIpIHtcclxuPGRpdiBtYXRSaXBwbGUgaWQ9XCJjb250YWluZXJcIiAoY2xpY2spPVwib25DaGFyQ2xpY2soJGV2ZW50KVwiPlxyXG4gIDxkaXZcclxuICAgIGlkPVwiYy1sYWJlbFwiXHJcbiAgICBbc3R5bGUuZm9udFNpemVdPVwiY2hhci5lbVNpemUgKyAnZW0nXCJcclxuICAgIFtzdHlsZS5ib3JkZXJDb2xvcl09XCJjaGFyLmJvcmRlckNvbG9yXCJcclxuICA+XHJcbiAgICB7eyBjaGFyLmxhYmVsIH19XHJcbiAgPC9kaXY+XHJcbiAgPGRpdlxyXG4gICAgaWQ9XCJjLWlkXCJcclxuICAgIFtzdHlsZS5mb250U2l6ZV09XCJjaGFyLmVtU2l6ZSAvIDIgKyAnZW0nXCJcclxuICAgIFtzdHlsZS5jb2xvcl09XCJjaGFyLmNvbG9yIHwgY29sb3JUb0NvbnRyYXN0XCJcclxuICAgIFtzdHlsZS5ib3JkZXJDb2xvcl09XCJjaGFyLmJvcmRlckNvbG9yXCJcclxuICAgIFtzdHlsZS5iYWNrZ3JvdW5kQ29sb3JdPVwiY2hhci5jb2xvclwiXHJcbiAgPlxyXG4gICAge3sgY2hhci5pZCB9fVxyXG4gIDwvZGl2PlxyXG48L2Rpdj5cclxufVxyXG4iXX0=
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
3
|
+
import { ReactiveFormsModule, Validators, } from '@angular/forms';
|
|
4
|
+
import { MatButtonModule } from '@angular/material/button';
|
|
5
|
+
import { MatFormFieldModule } from '@angular/material/form-field';
|
|
6
|
+
import { MatIconModule } from '@angular/material/icon';
|
|
7
|
+
import { MatInputModule } from '@angular/material/input';
|
|
8
|
+
import { MatTooltipModule } from '@angular/material/tooltip';
|
|
9
|
+
import { SnapshotViewService, } from '@myrmidon/gve-snapshot-view';
|
|
10
|
+
import { BaseTextViewComponent, } from '../base-text-view/base-text-view.component';
|
|
11
|
+
import { FeatureSetEditorComponent } from '../feature-set-editor/feature-set-editor.component';
|
|
12
|
+
import * as i0 from "@angular/core";
|
|
13
|
+
import * as i1 from "@angular/forms";
|
|
14
|
+
import * as i2 from "@myrmidon/ng-mat-tools";
|
|
15
|
+
import * as i3 from "@angular/common";
|
|
16
|
+
import * as i4 from "@angular/material/button";
|
|
17
|
+
import * as i5 from "@angular/material/form-field";
|
|
18
|
+
import * as i6 from "@angular/material/input";
|
|
19
|
+
import * as i7 from "@angular/material/tooltip";
|
|
20
|
+
/**
|
|
21
|
+
* Editor for snapshot's base text. This can receive a string or an array
|
|
22
|
+
* of `CharNode` objects, and lets the user edit the text, character by
|
|
23
|
+
* character, optionally also setting its features.
|
|
24
|
+
*/
|
|
25
|
+
export class BaseTextEditorComponent {
|
|
26
|
+
/**
|
|
27
|
+
* The text to edit. In input this can be a string, an array of `CharNode`
|
|
28
|
+
* objects. In output this will be an array of `CharNode` objects.
|
|
29
|
+
*/
|
|
30
|
+
get text() {
|
|
31
|
+
return this._text;
|
|
32
|
+
}
|
|
33
|
+
set text(value) {
|
|
34
|
+
this.resetSelectedChar();
|
|
35
|
+
if (this._text === value) {
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
if (!value) {
|
|
39
|
+
this._text = [];
|
|
40
|
+
}
|
|
41
|
+
else {
|
|
42
|
+
this._text = Array.isArray(value)
|
|
43
|
+
? value
|
|
44
|
+
: SnapshotViewService.stringToBaseChars(value);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
constructor(formBuilder, _dialogService) {
|
|
48
|
+
this._dialogService = _dialogService;
|
|
49
|
+
this._text = [];
|
|
50
|
+
/**
|
|
51
|
+
* Emits the edited text as an array of `CharNode`'s whenever it changes.
|
|
52
|
+
*/
|
|
53
|
+
this.textChange = new EventEmitter();
|
|
54
|
+
this.userText = formBuilder.control('', {
|
|
55
|
+
nonNullable: true,
|
|
56
|
+
validators: [Validators.required, Validators.maxLength(10000)],
|
|
57
|
+
});
|
|
58
|
+
this.form = formBuilder.group({
|
|
59
|
+
userText: this.userText,
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
resetSelectedChar() {
|
|
63
|
+
this.selectedChar = undefined;
|
|
64
|
+
this.selectedCharLabel = undefined;
|
|
65
|
+
}
|
|
66
|
+
onSelectedChar(event) {
|
|
67
|
+
this.selectedChar = this._text.find((c) => c.id === event.char.id);
|
|
68
|
+
this.selectedCharLabel = event.char.label;
|
|
69
|
+
}
|
|
70
|
+
onFeaturesChange(features) {
|
|
71
|
+
this.selectedChar.features = features;
|
|
72
|
+
this.textChange.emit(this._text);
|
|
73
|
+
}
|
|
74
|
+
onRangePick(range) {
|
|
75
|
+
this.textRange = range;
|
|
76
|
+
}
|
|
77
|
+
patchTextFromUser() {
|
|
78
|
+
// TODO
|
|
79
|
+
}
|
|
80
|
+
setTextFromUser() {
|
|
81
|
+
this._dialogService
|
|
82
|
+
.confirm('Confirm', 'Reset text?')
|
|
83
|
+
.subscribe((yes) => {
|
|
84
|
+
if (yes) {
|
|
85
|
+
this.text = this.userText.value;
|
|
86
|
+
this.textChange.emit(this._text);
|
|
87
|
+
}
|
|
88
|
+
});
|
|
89
|
+
}
|
|
90
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: BaseTextEditorComponent, deps: [{ token: i1.FormBuilder }, { token: i2.DialogService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
91
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.4", type: BaseTextEditorComponent, isStandalone: true, selector: "gve-base-text-editor", inputs: { text: "text" }, outputs: { textChange: "textChange" }, ngImport: i0, template: "<form [formGroup]=\"form\" (submit)=\"setTextFromUser()\">\r\n <!-- text -->\r\n <div>\r\n <mat-form-field class=\"full-width\">\r\n <mat-label>text</mat-label>\r\n <textarea matInput [formControl]=\"userText\" rows=\"5\"></textarea>\r\n <mat-error\r\n *ngIf=\"\r\n $any(userText).errors?.required &&\r\n (userText.dirty || userText.touched)\r\n \"\r\n >text required</mat-error\r\n >\r\n <mat-error\r\n *ngIf=\"\r\n $any(userText).errors?.maxLength &&\r\n (userText.dirty || userText.touched)\r\n \"\r\n >text too long</mat-error\r\n >\r\n </mat-form-field>\r\n </div>\r\n <!-- buttons -->\r\n <div class=\"form-row\">\r\n <!-- set -->\r\n <button\r\n type=\"button\"\r\n mat-flat-button\r\n class=\"mat-warn\"\r\n matTooltip=\"Reset characters to newly entered text\"\r\n [disabled]=\"!userText.value\"\r\n (click)=\"setTextFromUser()\"\r\n >\r\n set\r\n </button>\r\n <!-- patch -->\r\n <button\r\n type=\"button\"\r\n mat-flat-button\r\n class=\"mat-primary\"\r\n matTooltip=\"Patch characters with newly entered text\"\r\n [disabled]=\"!userText.value\"\r\n (click)=\"patchTextFromUser()\"\r\n >\r\n patch\r\n </button>\r\n </div>\r\n\r\n <!-- base text -->\r\n <div id=\"text-view\">\r\n <gve-base-text-view\r\n [text]=\"text\"\r\n (charPick)=\"onSelectedChar($event)\"\r\n (rangePick)=\"textRange = $event\"\r\n />\r\n\r\n <!-- text range -->\r\n @if (textRange) {\r\n <div id=\"text-range\">{{ textRange.at }} \u00D7 {{ textRange.run }}</div>\r\n }\r\n </div>\r\n\r\n <!-- char features -->\r\n @if (selectedChar) {\r\n <fieldset>\r\n <legend>features</legend>\r\n <gve-feature-set-editor\r\n [features]=\"selectedChar.features || []\"\r\n (featuresChange)=\"onFeaturesChange($event)\"\r\n />\r\n </fieldset>\r\n }\r\n</form>\r\n", styles: [".full-width{width:100%}#text-view{margin:8px 0}fieldset{border:1px solid #ccc;padding:10px;margin:10px 0;border-radius:5px}legend{color:silver}.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: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i4.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i5.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i5.MatLabel, selector: "mat-label" }, { kind: "directive", type: i5.MatError, selector: "mat-error, [matError]", inputs: ["id"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "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: MatTooltipModule }, { kind: "directive", type: i7.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "component", type: BaseTextViewComponent, selector: "gve-base-text-view", inputs: ["defaultColor", "defaultBorderColor", "selectionColor", "hasLineNumber", "text"], outputs: ["charPick", "rangePick"] }, { kind: "component", type: FeatureSetEditorComponent, selector: "gve-feature-set-editor", inputs: ["isVar", "featNames", "featValues", "noFilter", "features"], outputs: ["featuresChange"] }] }); }
|
|
92
|
+
}
|
|
93
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: BaseTextEditorComponent, decorators: [{
|
|
94
|
+
type: Component,
|
|
95
|
+
args: [{ selector: 'gve-base-text-editor', standalone: true, imports: [
|
|
96
|
+
CommonModule,
|
|
97
|
+
ReactiveFormsModule,
|
|
98
|
+
MatButtonModule,
|
|
99
|
+
MatFormFieldModule,
|
|
100
|
+
MatIconModule,
|
|
101
|
+
MatInputModule,
|
|
102
|
+
MatTooltipModule,
|
|
103
|
+
BaseTextViewComponent,
|
|
104
|
+
FeatureSetEditorComponent,
|
|
105
|
+
], template: "<form [formGroup]=\"form\" (submit)=\"setTextFromUser()\">\r\n <!-- text -->\r\n <div>\r\n <mat-form-field class=\"full-width\">\r\n <mat-label>text</mat-label>\r\n <textarea matInput [formControl]=\"userText\" rows=\"5\"></textarea>\r\n <mat-error\r\n *ngIf=\"\r\n $any(userText).errors?.required &&\r\n (userText.dirty || userText.touched)\r\n \"\r\n >text required</mat-error\r\n >\r\n <mat-error\r\n *ngIf=\"\r\n $any(userText).errors?.maxLength &&\r\n (userText.dirty || userText.touched)\r\n \"\r\n >text too long</mat-error\r\n >\r\n </mat-form-field>\r\n </div>\r\n <!-- buttons -->\r\n <div class=\"form-row\">\r\n <!-- set -->\r\n <button\r\n type=\"button\"\r\n mat-flat-button\r\n class=\"mat-warn\"\r\n matTooltip=\"Reset characters to newly entered text\"\r\n [disabled]=\"!userText.value\"\r\n (click)=\"setTextFromUser()\"\r\n >\r\n set\r\n </button>\r\n <!-- patch -->\r\n <button\r\n type=\"button\"\r\n mat-flat-button\r\n class=\"mat-primary\"\r\n matTooltip=\"Patch characters with newly entered text\"\r\n [disabled]=\"!userText.value\"\r\n (click)=\"patchTextFromUser()\"\r\n >\r\n patch\r\n </button>\r\n </div>\r\n\r\n <!-- base text -->\r\n <div id=\"text-view\">\r\n <gve-base-text-view\r\n [text]=\"text\"\r\n (charPick)=\"onSelectedChar($event)\"\r\n (rangePick)=\"textRange = $event\"\r\n />\r\n\r\n <!-- text range -->\r\n @if (textRange) {\r\n <div id=\"text-range\">{{ textRange.at }} \u00D7 {{ textRange.run }}</div>\r\n }\r\n </div>\r\n\r\n <!-- char features -->\r\n @if (selectedChar) {\r\n <fieldset>\r\n <legend>features</legend>\r\n <gve-feature-set-editor\r\n [features]=\"selectedChar.features || []\"\r\n (featuresChange)=\"onFeaturesChange($event)\"\r\n />\r\n </fieldset>\r\n }\r\n</form>\r\n", styles: [".full-width{width:100%}#text-view{margin:8px 0}fieldset{border:1px solid #ccc;padding:10px;margin:10px 0;border-radius:5px}legend{color:silver}.form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}\n"] }]
|
|
106
|
+
}], ctorParameters: () => [{ type: i1.FormBuilder }, { type: i2.DialogService }], propDecorators: { text: [{
|
|
107
|
+
type: Input
|
|
108
|
+
}], textChange: [{
|
|
109
|
+
type: Output
|
|
110
|
+
}] } });
|
|
111
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-text-editor.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/gve-core/src/lib/components/base-text-editor/base-text-editor.component.ts","../../../../../../../projects/myrmidon/gve-core/src/lib/components/base-text-editor/base-text-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { Component, EventEmitter, Input, Output } from '@angular/core';\r\nimport {\r\n  FormBuilder,\r\n  FormControl,\r\n  FormGroup,\r\n  ReactiveFormsModule,\r\n  Validators,\r\n} from '@angular/forms';\r\nimport { 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 { MatTooltipModule } from '@angular/material/tooltip';\r\n\r\nimport {\r\n  CharNode,\r\n  Feature,\r\n  SnapshotViewService,\r\n} from '@myrmidon/gve-snapshot-view';\r\nimport { DialogService } from '@myrmidon/ng-mat-tools';\r\n\r\nimport { BaseTextCharEvent } from '../base-text-char/base-text-char.component';\r\nimport {\r\n  BaseTextViewComponent,\r\n  VarBaseTextRange,\r\n} from '../base-text-view/base-text-view.component';\r\nimport { FeatureSetEditorComponent } from '../feature-set-editor/feature-set-editor.component';\r\n\r\n/**\r\n * Editor for snapshot's base text. This can receive a string or an array\r\n * of `CharNode` objects, and lets the user edit the text, character by\r\n * character, optionally also setting its features.\r\n */\r\n@Component({\r\n  selector: 'gve-base-text-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    MatTooltipModule,\r\n    BaseTextViewComponent,\r\n    FeatureSetEditorComponent,\r\n  ],\r\n  templateUrl: './base-text-editor.component.html',\r\n  styleUrl: './base-text-editor.component.css',\r\n})\r\nexport class BaseTextEditorComponent {\r\n  private _text: CharNode[] = [];\r\n\r\n  /**\r\n   * The text to edit. In input this can be a string, an array of `CharNode`\r\n   * objects. In output this will be an array of `CharNode` objects.\r\n   */\r\n  @Input()\r\n  public get text(): CharNode[] {\r\n    return this._text;\r\n  }\r\n  public set text(value: string | CharNode[] | undefined | null) {\r\n    this.resetSelectedChar();\r\n    if (this._text === value) {\r\n      return;\r\n    }\r\n    if (!value) {\r\n      this._text = [];\r\n    } else {\r\n      this._text = Array.isArray(value)\r\n        ? value\r\n        : SnapshotViewService.stringToBaseChars(value);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Emits the edited text as an array of `CharNode`'s whenever it changes.\r\n   */\r\n  @Output()\r\n  public readonly textChange = new EventEmitter<CharNode[]>();\r\n\r\n  public userText: FormControl<string>;\r\n  public form: FormGroup;\r\n\r\n  public selectedChar?: CharNode;\r\n  public selectedCharLabel?: string;\r\n  public textRange?: VarBaseTextRange;\r\n\r\n  constructor(formBuilder: FormBuilder, private _dialogService: DialogService) {\r\n    this.userText = formBuilder.control('', {\r\n      nonNullable: true,\r\n      validators: [Validators.required, Validators.maxLength(10000)],\r\n    });\r\n    this.form = formBuilder.group({\r\n      userText: this.userText,\r\n    });\r\n  }\r\n\r\n  private resetSelectedChar(): void {\r\n    this.selectedChar = undefined;\r\n    this.selectedCharLabel = undefined;\r\n  }\r\n\r\n  public onSelectedChar(event: BaseTextCharEvent): void {\r\n    this.selectedChar = this._text.find((c) => c.id === event.char.id);\r\n    this.selectedCharLabel = event.char.label;\r\n  }\r\n\r\n  public onFeaturesChange(features: Feature[]): void {\r\n    this.selectedChar!.features = features;\r\n    this.textChange.emit(this._text);\r\n  }\r\n\r\n  public onRangePick(range: VarBaseTextRange) {\r\n    this.textRange = range;\r\n  }\r\n\r\n  public patchTextFromUser(): void {\r\n    // TODO\r\n  }\r\n\r\n  public setTextFromUser(): void {\r\n    this._dialogService\r\n      .confirm('Confirm', 'Reset text?')\r\n      .subscribe((yes: boolean) => {\r\n        if (yes) {\r\n          this.text = this.userText.value;\r\n          this.textChange.emit(this._text);\r\n        }\r\n      });\r\n  }\r\n}\r\n","<form [formGroup]=\"form\" (submit)=\"setTextFromUser()\">\r\n  <!-- text -->\r\n  <div>\r\n    <mat-form-field class=\"full-width\">\r\n      <mat-label>text</mat-label>\r\n      <textarea matInput [formControl]=\"userText\" rows=\"5\"></textarea>\r\n      <mat-error\r\n        *ngIf=\"\r\n          $any(userText).errors?.required &&\r\n          (userText.dirty || userText.touched)\r\n        \"\r\n        >text required</mat-error\r\n      >\r\n      <mat-error\r\n        *ngIf=\"\r\n          $any(userText).errors?.maxLength &&\r\n          (userText.dirty || userText.touched)\r\n        \"\r\n        >text too long</mat-error\r\n      >\r\n    </mat-form-field>\r\n  </div>\r\n  <!-- buttons -->\r\n  <div class=\"form-row\">\r\n    <!-- set -->\r\n    <button\r\n      type=\"button\"\r\n      mat-flat-button\r\n      class=\"mat-warn\"\r\n      matTooltip=\"Reset characters to newly entered text\"\r\n      [disabled]=\"!userText.value\"\r\n      (click)=\"setTextFromUser()\"\r\n    >\r\n      set\r\n    </button>\r\n    <!-- patch -->\r\n    <button\r\n      type=\"button\"\r\n      mat-flat-button\r\n      class=\"mat-primary\"\r\n      matTooltip=\"Patch characters with newly entered text\"\r\n      [disabled]=\"!userText.value\"\r\n      (click)=\"patchTextFromUser()\"\r\n    >\r\n      patch\r\n    </button>\r\n  </div>\r\n\r\n  <!-- base text -->\r\n  <div id=\"text-view\">\r\n    <gve-base-text-view\r\n      [text]=\"text\"\r\n      (charPick)=\"onSelectedChar($event)\"\r\n      (rangePick)=\"textRange = $event\"\r\n    />\r\n\r\n    <!-- text range -->\r\n    @if (textRange) {\r\n    <div id=\"text-range\">{{ textRange.at }} × {{ textRange.run }}</div>\r\n    }\r\n  </div>\r\n\r\n  <!-- char features -->\r\n  @if (selectedChar) {\r\n  <fieldset>\r\n    <legend>features</legend>\r\n    <gve-feature-set-editor\r\n      [features]=\"selectedChar.features || []\"\r\n      (featuresChange)=\"onFeaturesChange($event)\"\r\n    />\r\n  </fieldset>\r\n  }\r\n</form>\r\n"]}
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
3
|
+
import { BaseTextCharComponent, } from '../base-text-char/base-text-char.component';
|
|
4
|
+
import { SnapshotViewService } from '@myrmidon/gve-snapshot-view';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
/**
|
|
7
|
+
* A component to display a selectable base text.
|
|
8
|
+
*/
|
|
9
|
+
export class BaseTextViewComponent {
|
|
10
|
+
constructor() {
|
|
11
|
+
this.defaultColor = '#DBDBDB';
|
|
12
|
+
this.defaultBorderColor = '#DBDBDB';
|
|
13
|
+
this.selectionColor = '#3E92CC';
|
|
14
|
+
this.hasLineNumber = false;
|
|
15
|
+
this.charPick = new EventEmitter();
|
|
16
|
+
this.rangePick = new EventEmitter();
|
|
17
|
+
this.lines = [];
|
|
18
|
+
}
|
|
19
|
+
get text() {
|
|
20
|
+
return this._text;
|
|
21
|
+
}
|
|
22
|
+
set text(value) {
|
|
23
|
+
this._text = value || undefined;
|
|
24
|
+
this.buildLines();
|
|
25
|
+
}
|
|
26
|
+
buildLines() {
|
|
27
|
+
if (!this._text) {
|
|
28
|
+
this.lines = [];
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
const newLines = [];
|
|
32
|
+
newLines.push([]);
|
|
33
|
+
if (Array.isArray(this._text)) {
|
|
34
|
+
const nodes = this._text;
|
|
35
|
+
for (let i = 0; i < nodes.length; i++) {
|
|
36
|
+
const char = {
|
|
37
|
+
id: nodes[i].id,
|
|
38
|
+
value: nodes[i].data,
|
|
39
|
+
label: nodes[i].label,
|
|
40
|
+
color: this.defaultColor,
|
|
41
|
+
borderColor: this.defaultBorderColor,
|
|
42
|
+
emSize: 1.5,
|
|
43
|
+
};
|
|
44
|
+
newLines[newLines.length - 1].push(char);
|
|
45
|
+
if (char.value === '\n') {
|
|
46
|
+
newLines.push([]);
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
else {
|
|
51
|
+
for (let i = 0; i < this._text.length; i++) {
|
|
52
|
+
const char = {
|
|
53
|
+
id: i + 1,
|
|
54
|
+
value: this._text[i],
|
|
55
|
+
label: SnapshotViewService.translateSpecialChar(this._text[i]),
|
|
56
|
+
color: this.defaultColor,
|
|
57
|
+
borderColor: this.defaultBorderColor,
|
|
58
|
+
emSize: 1.5,
|
|
59
|
+
};
|
|
60
|
+
newLines[newLines.length - 1].push(char);
|
|
61
|
+
if (char.value === '\n') {
|
|
62
|
+
newLines.push([]);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
this.lines = newLines;
|
|
67
|
+
}
|
|
68
|
+
resetColors() {
|
|
69
|
+
for (const line of this.lines) {
|
|
70
|
+
for (const c of line) {
|
|
71
|
+
c.color = this.defaultColor;
|
|
72
|
+
c.borderColor = this.defaultBorderColor;
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
onCharPick(event) {
|
|
77
|
+
this.resetColors();
|
|
78
|
+
if (this._lastSelectedChar &&
|
|
79
|
+
this._lastSelectedChar !== event.char &&
|
|
80
|
+
event.event.ctrlKey) {
|
|
81
|
+
const minId = Math.min(this._lastSelectedChar.id, event.char.id);
|
|
82
|
+
const maxId = Math.max(this._lastSelectedChar.id, event.char.id);
|
|
83
|
+
for (const line of this.lines) {
|
|
84
|
+
for (const c of line) {
|
|
85
|
+
if (c.id >= minId && c.id <= maxId) {
|
|
86
|
+
c.color = this.selectionColor;
|
|
87
|
+
c.borderColor = this.selectionColor;
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
this.charPick.emit(event);
|
|
92
|
+
this.rangePick.emit({ at: minId, run: maxId - minId + 1 });
|
|
93
|
+
return;
|
|
94
|
+
}
|
|
95
|
+
else {
|
|
96
|
+
// select current char
|
|
97
|
+
event.char.color = this.selectionColor;
|
|
98
|
+
event.char.borderColor = this.selectionColor;
|
|
99
|
+
this.charPick.emit(event);
|
|
100
|
+
this.rangePick.emit({ at: event.char.id, run: 1 });
|
|
101
|
+
}
|
|
102
|
+
this._lastSelectedChar = event.char;
|
|
103
|
+
}
|
|
104
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: BaseTextViewComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
105
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.4", type: BaseTextViewComponent, isStandalone: true, selector: "gve-base-text-view", inputs: { defaultColor: "defaultColor", defaultBorderColor: "defaultBorderColor", selectionColor: "selectionColor", hasLineNumber: "hasLineNumber", text: "text" }, outputs: { charPick: "charPick", rangePick: "rangePick" }, ngImport: i0, template: "<div id=\"text\">\r\n @for (line of lines; track line) {\r\n <div class=\"line\">\r\n @if (hasLineNumber) {\r\n <div class=\"nr\">\r\n {{ lines.indexOf(line) + 1 }}\r\n </div>\r\n } @for (c of line; track c.id) {\r\n <gve-base-text-char [char]=\"c\" (charPick)=\"onCharPick($event)\" />\r\n }\r\n </div>\r\n }\r\n</div>\r\n", styles: [".line{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.line *{flex:0 0 auto}.nr{font-size:.8em;font-weight:700;color:silver;margin:0 4px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: BaseTextCharComponent, selector: "gve-base-text-char", inputs: ["char"], outputs: ["charPick"] }] }); }
|
|
106
|
+
}
|
|
107
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: BaseTextViewComponent, decorators: [{
|
|
108
|
+
type: Component,
|
|
109
|
+
args: [{ selector: 'gve-base-text-view', standalone: true, imports: [CommonModule, BaseTextCharComponent], template: "<div id=\"text\">\r\n @for (line of lines; track line) {\r\n <div class=\"line\">\r\n @if (hasLineNumber) {\r\n <div class=\"nr\">\r\n {{ lines.indexOf(line) + 1 }}\r\n </div>\r\n } @for (c of line; track c.id) {\r\n <gve-base-text-char [char]=\"c\" (charPick)=\"onCharPick($event)\" />\r\n }\r\n </div>\r\n }\r\n</div>\r\n", styles: [".line{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.line *{flex:0 0 auto}.nr{font-size:.8em;font-weight:700;color:silver;margin:0 4px}\n"] }]
|
|
110
|
+
}], propDecorators: { defaultColor: [{
|
|
111
|
+
type: Input
|
|
112
|
+
}], defaultBorderColor: [{
|
|
113
|
+
type: Input
|
|
114
|
+
}], selectionColor: [{
|
|
115
|
+
type: Input
|
|
116
|
+
}], hasLineNumber: [{
|
|
117
|
+
type: Input
|
|
118
|
+
}], text: [{
|
|
119
|
+
type: Input
|
|
120
|
+
}], charPick: [{
|
|
121
|
+
type: Output
|
|
122
|
+
}], rangePick: [{
|
|
123
|
+
type: Output
|
|
124
|
+
}] } });
|
|
125
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-text-view.component.js","sourceRoot":"","sources":["../../../../../../../projects/myrmidon/gve-core/src/lib/components/base-text-view/base-text-view.component.ts","../../../../../../../projects/myrmidon/gve-core/src/lib/components/base-text-view/base-text-view.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { Component, EventEmitter, Input, Output } from '@angular/core';\r\n\r\nimport {\r\n  BaseTextChar,\r\n  BaseTextCharComponent,\r\n  BaseTextCharEvent,\r\n} from '../base-text-char/base-text-char.component';\r\nimport { CharNode, SnapshotViewService } from '@myrmidon/gve-snapshot-view';\r\n\r\nexport interface VarBaseTextRange {\r\n  at: number;\r\n  run: number;\r\n}\r\n\r\n/**\r\n * A component to display a selectable base text.\r\n */\r\n@Component({\r\n  selector: 'gve-base-text-view',\r\n  standalone: true,\r\n  imports: [CommonModule, BaseTextCharComponent],\r\n  templateUrl: './base-text-view.component.html',\r\n  styleUrl: './base-text-view.component.css',\r\n})\r\nexport class BaseTextViewComponent {\r\n  private _text?: string | CharNode[];\r\n  private _lastSelectedChar?: BaseTextChar;\r\n\r\n  @Input()\r\n  public defaultColor = '#DBDBDB';\r\n\r\n  @Input()\r\n  public defaultBorderColor = '#DBDBDB';\r\n\r\n  @Input()\r\n  public selectionColor = '#3E92CC';\r\n\r\n  @Input()\r\n  public hasLineNumber = false;\r\n\r\n  @Input()\r\n  public get text(): string | CharNode[] | undefined {\r\n    return this._text;\r\n  }\r\n  public set text(value: string | CharNode[] | undefined | null) {\r\n    this._text = value || undefined;\r\n    this.buildLines();\r\n  }\r\n\r\n  @Output()\r\n  public charPick: EventEmitter<BaseTextCharEvent> =\r\n    new EventEmitter<BaseTextCharEvent>();\r\n\r\n  @Output()\r\n  public rangePick: EventEmitter<VarBaseTextRange> =\r\n    new EventEmitter<VarBaseTextRange>();\r\n\r\n  public lines: BaseTextChar[][] = [];\r\n\r\n  private buildLines(): void {\r\n    if (!this._text) {\r\n      this.lines = [];\r\n      return;\r\n    }\r\n\r\n    const newLines: BaseTextChar[][] = [];\r\n    newLines.push([]);\r\n\r\n    if (Array.isArray(this._text)) {\r\n      const nodes = this._text as CharNode[];\r\n      for (let i = 0; i < nodes.length; i++) {\r\n        const char: BaseTextChar = {\r\n          id: nodes[i].id,\r\n          value: nodes[i].data,\r\n          label: nodes[i].label,\r\n          color: this.defaultColor,\r\n          borderColor: this.defaultBorderColor,\r\n          emSize: 1.5,\r\n        };\r\n        newLines[newLines.length - 1].push(char);\r\n\r\n        if (char.value === '\\n') {\r\n          newLines.push([]);\r\n        }\r\n      }\r\n    } else {\r\n      for (let i = 0; i < this._text.length; i++) {\r\n        const char: BaseTextChar = {\r\n          id: i + 1,\r\n          value: this._text[i],\r\n          label: SnapshotViewService.translateSpecialChar(this._text[i]),\r\n          color: this.defaultColor,\r\n          borderColor: this.defaultBorderColor,\r\n          emSize: 1.5,\r\n        };\r\n        newLines[newLines.length - 1].push(char);\r\n\r\n        if (char.value === '\\n') {\r\n          newLines.push([]);\r\n        }\r\n      }\r\n    }\r\n\r\n    this.lines = newLines;\r\n  }\r\n\r\n  private resetColors(): void {\r\n    for (const line of this.lines) {\r\n      for (const c of line) {\r\n        c.color = this.defaultColor;\r\n        c.borderColor = this.defaultBorderColor;\r\n      }\r\n    }\r\n  }\r\n\r\n  public onCharPick(event: BaseTextCharEvent): void {\r\n    this.resetColors();\r\n\r\n    if (\r\n      this._lastSelectedChar &&\r\n      this._lastSelectedChar !== event.char &&\r\n      event.event.ctrlKey\r\n    ) {\r\n      const minId = Math.min(this._lastSelectedChar.id, event.char.id);\r\n      const maxId = Math.max(this._lastSelectedChar.id, event.char.id);\r\n\r\n      for (const line of this.lines) {\r\n        for (const c of line) {\r\n          if (c.id >= minId && c.id <= maxId) {\r\n            c.color = this.selectionColor;\r\n            c.borderColor = this.selectionColor;\r\n          }\r\n        }\r\n      }\r\n\r\n      this.charPick.emit(event);\r\n      this.rangePick.emit({ at: minId, run: maxId - minId + 1 });\r\n      return;\r\n    } else {\r\n      // select current char\r\n      event.char.color = this.selectionColor;\r\n      event.char.borderColor = this.selectionColor;\r\n      this.charPick.emit(event);\r\n      this.rangePick.emit({ at: event.char.id, run: 1 });\r\n    }\r\n\r\n    this._lastSelectedChar = event.char;\r\n  }\r\n}\r\n","<div id=\"text\">\r\n  @for (line of lines; track line) {\r\n  <div class=\"line\">\r\n    @if (hasLineNumber) {\r\n    <div class=\"nr\">\r\n      {{ lines.indexOf(line) + 1 }}\r\n    </div>\r\n    } @for (c of line; track c.id) {\r\n    <gve-base-text-char [char]=\"c\" (charPick)=\"onCharPick($event)\" />\r\n    }\r\n  </div>\r\n  }\r\n</div>\r\n"]}
|