@myrmidon/cadmus-refs-asserted-ids 2.0.0 → 2.1.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.
@@ -0,0 +1,405 @@
1
+ import { Component, EventEmitter, Inject, Input, Output, } from '@angular/core';
2
+ import { Validators, } from '@angular/forms';
3
+ import { debounceTime, distinctUntilChanged, forkJoin, take, } from 'rxjs';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "../services/item-ref-lookup.service";
6
+ import * as i2 from "../services/pin-ref-lookup.service";
7
+ import * as i3 from "@myrmidon/cadmus-api";
8
+ import * as i4 from "@angular/material/snack-bar";
9
+ import * as i5 from "@angular/forms";
10
+ import * as i6 from "@angular/common";
11
+ import * as i7 from "@angular/cdk/clipboard";
12
+ import * as i8 from "@angular/material/button";
13
+ import * as i9 from "@angular/material/checkbox";
14
+ import * as i10 from "@angular/material/expansion";
15
+ import * as i11 from "@angular/material/form-field";
16
+ import * as i12 from "@angular/material/icon";
17
+ import * as i13 from "@angular/material/input";
18
+ import * as i14 from "@angular/material/select";
19
+ import * as i15 from "@angular/material/core";
20
+ import * as i16 from "@myrmidon/cadmus-refs-lookup";
21
+ import * as i17 from "@myrmidon/ng-tools";
22
+ // from Cadmus general parts
23
+ const METADATA_PART_ID = 'it.vedph.metadata';
24
+ /*
25
+ * Scoped pin-based lookup component. This component provides a list
26
+ * of pin-based searches, with a lookup control. Whenever the user
27
+ * picks a pin value, he gets the details about its item and part, and
28
+ * item's metadata part, if any. He can then use these data to build
29
+ * some EID by variously assembling these components.
30
+ */
31
+ class PinTargetLookupComponent {
32
+ /**
33
+ * True when the by-type pin lookup mode is active.
34
+ * User can change mode unless modeSwitching is false.
35
+ */
36
+ get pinByTypeMode() {
37
+ return this.byTypeMode?.value;
38
+ }
39
+ set pinByTypeMode(value) {
40
+ if (!this.byTypeMode) {
41
+ this._startWithByTypeMode = value;
42
+ }
43
+ else {
44
+ this.byTypeMode.setValue(value || false);
45
+ this.byTypeMode.updateValueAndValidity();
46
+ }
47
+ }
48
+ /**
49
+ * The target to be edited.
50
+ */
51
+ get target() {
52
+ return this._target;
53
+ }
54
+ set target(value) {
55
+ if (this._target === value) {
56
+ return;
57
+ }
58
+ this._target = value || undefined;
59
+ this.updateForm(this._target);
60
+ }
61
+ constructor(_presetLookupDefs, itemLookupService, pinLookupService, _itemService, _thesService, _snackbar, formBuilder) {
62
+ this._presetLookupDefs = _presetLookupDefs;
63
+ this.itemLookupService = itemLookupService;
64
+ this.pinLookupService = pinLookupService;
65
+ this._itemService = _itemService;
66
+ this._thesService = _thesService;
67
+ this._snackbar = _snackbar;
68
+ this.partTypeKeys = [];
69
+ this.itemParts = [];
70
+ this._subs = [];
71
+ this.modelEntries = [];
72
+ // this is the default filter for the pin lookup, which will
73
+ // be merged with values provided by user here
74
+ this.filter = {
75
+ text: '',
76
+ limit: 10,
77
+ };
78
+ // form
79
+ this.item = formBuilder.control(null);
80
+ this.itemPart = formBuilder.control(null);
81
+ this.partTypeKey = formBuilder.control(null);
82
+ this.gid = formBuilder.control(null, [
83
+ Validators.required,
84
+ Validators.maxLength(300),
85
+ ]);
86
+ this.label = formBuilder.control(null, [
87
+ Validators.required,
88
+ Validators.maxLength(300),
89
+ ]);
90
+ this.byTypeMode = formBuilder.control(false, { nonNullable: true });
91
+ this.external = formBuilder.control(false, { nonNullable: true });
92
+ this.form = formBuilder.group({
93
+ item: this.item,
94
+ itemPart: this.itemPart,
95
+ partTypeKey: this.partTypeKey,
96
+ gid: this.gid,
97
+ label: this.label,
98
+ byTypeMode: this.byTypeMode,
99
+ external: this.external,
100
+ });
101
+ // events
102
+ this.targetChange = new EventEmitter();
103
+ this.editorClose = new EventEmitter();
104
+ }
105
+ forceByItem() {
106
+ this.pinByTypeMode = false;
107
+ this.canSwitchMode = false;
108
+ }
109
+ setupKeys() {
110
+ // use DI presets if no lookup definitions
111
+ if (!this.lookupDefinitions) {
112
+ this.lookupDefinitions = this._presetLookupDefs;
113
+ }
114
+ // keys are all the defined lookup searches
115
+ this.partTypeKeys = Object.keys(this.lookupDefinitions);
116
+ // if no keys, get them from thesaurus model-types;
117
+ // if this is not available, just force by item mode.
118
+ if (!this.partTypeKeys.length) {
119
+ if (this.modelEntries?.length) {
120
+ // set lookupDefinitions from thesaurus entries
121
+ this.lookupDefinitions = {};
122
+ this.modelEntries.forEach((e) => {
123
+ this.lookupDefinitions[e.value] = {
124
+ name: e.value,
125
+ typeId: e.id,
126
+ };
127
+ });
128
+ // set type keys from thesaurus entries
129
+ this.partTypeKeys = this.modelEntries.map((e) => e.value);
130
+ }
131
+ }
132
+ // if still no keys, force by item mode
133
+ if (!this.partTypeKeys.length) {
134
+ this.forceByItem();
135
+ }
136
+ else {
137
+ // if there is only one key, set it as the default
138
+ if (this.partTypeKeys.length === 1) {
139
+ this.partTypeKey.setValue(this.partTypeKeys[0]);
140
+ }
141
+ }
142
+ }
143
+ canBuildGid() {
144
+ return (!this.external.value && !this.gid.value) || this.gid.pristine;
145
+ }
146
+ canBuildLabel() {
147
+ return (!this.external.value && !this.label.value) || this.label.pristine;
148
+ }
149
+ ngOnInit() {
150
+ // set start mode if required
151
+ if (this._startWithByTypeMode) {
152
+ this.byTypeMode.setValue(true);
153
+ }
154
+ // whenever item changes, update item's parts and filter
155
+ this._subs.push(this.item.valueChanges
156
+ .pipe(distinctUntilChanged(), debounceTime(300))
157
+ .subscribe((item) => {
158
+ this.itemPart.setValue(null);
159
+ this.itemParts = item?.parts || [];
160
+ this.filter = {
161
+ ...this.filter,
162
+ itemId: item?.id,
163
+ };
164
+ }));
165
+ // whenever itemPart changes, update target and eventually gid
166
+ this._subs.push(this.itemPart.valueChanges
167
+ .pipe(distinctUntilChanged(), debounceTime(300))
168
+ .subscribe((part) => {
169
+ if (!this.gid.value || this.gid.pristine) {
170
+ this.gid.setValue(this.buildGid());
171
+ }
172
+ this.filter = {
173
+ ...this.filter,
174
+ partId: part?.id,
175
+ };
176
+ this.updateTarget();
177
+ }));
178
+ // whenever partTypeKey changes, update filter's options
179
+ this._subs.push(this.partTypeKey.valueChanges
180
+ .pipe(distinctUntilChanged(), debounceTime(300))
181
+ .subscribe((key) => {
182
+ this.pinFilterOptions = key
183
+ ? this.lookupDefinitions[key]
184
+ : undefined;
185
+ }));
186
+ // load model-types thesaurus entries
187
+ this._thesService.getThesaurus('model-types', true).subscribe({
188
+ next: (t) => {
189
+ this.modelEntries = t.entries || [];
190
+ if (this.modelEntries?.length) {
191
+ this.setupKeys();
192
+ }
193
+ else {
194
+ this.forceByItem();
195
+ }
196
+ },
197
+ error: () => {
198
+ this.forceByItem();
199
+ },
200
+ });
201
+ }
202
+ ngOnDestroy() {
203
+ for (let i = 0; i < this._subs.length; i++) {
204
+ this._subs[i].unsubscribe();
205
+ }
206
+ }
207
+ buildGid() {
208
+ // the GID is the part ID if any, or the item ID, followed by
209
+ // the pin's value (=EID)
210
+ const pin = this.lookupData?.pin;
211
+ if (!pin) {
212
+ return null;
213
+ }
214
+ return pin.partId
215
+ ? `P${pin.partId}/${pin.value}`
216
+ : `I${pin.itemId}/${pin.value}`;
217
+ }
218
+ buildLabel() {
219
+ if (!this.lookupData?.pin) {
220
+ return null;
221
+ }
222
+ const sb = [];
223
+ // pin value
224
+ sb.push(this.lookupData.pin.value);
225
+ // item title
226
+ sb.push(' | ');
227
+ sb.push(this.lookupData.item.title || this.lookupData.item.id);
228
+ // part type and role
229
+ const e = this.modelEntries?.find((e) => e.id === this.lookupData.pin.partTypeId);
230
+ sb.push(' (');
231
+ sb.push(e?.value || this.lookupData.pin.partTypeId);
232
+ if (this.lookupData.pin.roleId) {
233
+ sb.push(`, ${this.lookupData.pin.roleId}`);
234
+ }
235
+ sb.push(')');
236
+ return sb.join('');
237
+ }
238
+ getTarget() {
239
+ if (this.external.value) {
240
+ return {
241
+ gid: this.gid.value || '',
242
+ label: this.label.value || '',
243
+ };
244
+ }
245
+ else {
246
+ const pin = this.lookupData?.pin;
247
+ return {
248
+ gid: this.gid.value || '',
249
+ label: this.label.value || '',
250
+ itemId: pin?.itemId || '',
251
+ partId: pin?.partId || '',
252
+ partTypeId: pin?.partTypeId || '',
253
+ roleId: pin?.roleId || '',
254
+ name: pin?.name || '',
255
+ value: pin?.value || '',
256
+ };
257
+ }
258
+ }
259
+ updateTarget() {
260
+ this._noFormUpdate = true;
261
+ if (this.canBuildGid()) {
262
+ this.gid.setValue(this.buildGid());
263
+ this.gid.markAsDirty();
264
+ }
265
+ if (this.canBuildLabel()) {
266
+ this.label.setValue(this.buildLabel());
267
+ this.label.markAsDirty();
268
+ }
269
+ this.target = this.getTarget();
270
+ this._noFormUpdate = false;
271
+ }
272
+ updateForm(target) {
273
+ if (this._noFormUpdate) {
274
+ return;
275
+ }
276
+ // build pin info from target
277
+ if (!target) {
278
+ this.lookupData = undefined;
279
+ this.form.reset();
280
+ return;
281
+ }
282
+ this.external.setValue(!target.name);
283
+ this.lookupData = {
284
+ pin: {
285
+ itemId: target.itemId || '',
286
+ partId: target.partId || '',
287
+ partTypeId: target.partTypeId || '',
288
+ roleId: target.roleId || '',
289
+ name: target.name || '',
290
+ value: target.value || '',
291
+ },
292
+ };
293
+ // get item
294
+ if (target.itemId) {
295
+ this._itemService.getItem(target.itemId, true).subscribe({
296
+ next: (item) => {
297
+ this.item.setValue(item);
298
+ this.form.markAsPristine();
299
+ },
300
+ error: (error) => {
301
+ if (error) {
302
+ console.error(JSON.stringify(error));
303
+ }
304
+ },
305
+ });
306
+ }
307
+ }
308
+ /**
309
+ * Called when the item lookup changes (item is looked up
310
+ * by its title).
311
+ *
312
+ * @param item The item got from lookup.
313
+ */
314
+ onItemLookupChange(item) {
315
+ // load item's parts
316
+ this._itemService.getItem(item.id, true).subscribe({
317
+ next: (i) => {
318
+ // setting the item will trigger its parts update
319
+ this.item.setValue(i);
320
+ this.updateTarget();
321
+ },
322
+ error: (error) => {
323
+ if (error) {
324
+ console.error(JSON.stringify(error));
325
+ }
326
+ this.itemPart.setValue(null);
327
+ this.itemParts = [];
328
+ this.updateTarget();
329
+ },
330
+ });
331
+ }
332
+ loadItemInfo(pin) {
333
+ forkJoin({
334
+ item: this._itemService.getItem(pin.itemId, false),
335
+ part: this._itemService.getPartFromTypeAndRole(pin.itemId, METADATA_PART_ID),
336
+ })
337
+ .pipe(take(1))
338
+ .subscribe({
339
+ next: (result) => {
340
+ this.lookupData = {
341
+ pin: pin,
342
+ item: result.item,
343
+ metaPart: result.part,
344
+ };
345
+ this.updateTarget();
346
+ },
347
+ error: (error) => {
348
+ this.lookupData = undefined;
349
+ console.error(error ? JSON.stringify(error) : 'Error loading item/metadata');
350
+ },
351
+ });
352
+ }
353
+ /**
354
+ * Called when the pin lookup change. A pin is looked up by its
355
+ * name and value (=the filter's text), and optionally by:
356
+ * - its index lookup definition (selected by partTypeKey).
357
+ * - its item (defined by item, in filter).
358
+ * - its part (defined by itemPart, in filter).
359
+ *
360
+ * @param info The pin info from pin lookup.
361
+ */
362
+ onPinLookupChange(info) {
363
+ this.loadItemInfo(info);
364
+ }
365
+ onCopied() {
366
+ this._snackbar.open('Copied to clipboard', 'OK', {
367
+ duration: 1500,
368
+ });
369
+ }
370
+ close() {
371
+ this.editorClose.emit();
372
+ }
373
+ save() {
374
+ if (this.form.invalid) {
375
+ return;
376
+ }
377
+ this._target = this.getTarget();
378
+ this.targetChange.emit(this._target);
379
+ }
380
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.1", ngImport: i0, type: PinTargetLookupComponent, deps: [{ token: 'indexLookupDefinitions' }, { token: i1.ItemRefLookupService }, { token: i2.PinRefLookupService }, { token: i3.ItemService }, { token: i3.ThesaurusService }, { token: i4.MatSnackBar }, { token: i5.FormBuilder }], target: i0.ɵɵFactoryTarget.Component }); }
381
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.1", type: PinTargetLookupComponent, selector: "cadmus-pin-target-lookup", inputs: { pinByTypeMode: "pinByTypeMode", canSwitchMode: "canSwitchMode", canEditTarget: "canEditTarget", lookupDefinitions: "lookupDefinitions", target: "target" }, outputs: { editorClose: "editorClose", targetChange: "targetChange" }, ngImport: i0, template: "<form [formGroup]=\"form\" (submit)=\"save()\">\r\n <!-- external -->\r\n <mat-checkbox [formControl]=\"external\">external</mat-checkbox>\r\n\r\n <div class=\"form-row\">\r\n <!-- label -->\r\n <div>\r\n <mat-form-field *ngIf=\"external.value || canEditTarget\">\r\n <mat-label>label</mat-label>\r\n <input matInput [formControl]=\"label\" />\r\n <mat-error\r\n *ngIf=\"$any(label).errors?.required && (label.dirty || label.touched)\"\r\n >label required</mat-error\r\n >\r\n <mat-error\r\n *ngIf=\"\r\n $any(label).errors?.maxLength && (label.dirty || label.touched)\r\n \"\r\n >label too long</mat-error\r\n >\r\n </mat-form-field>\r\n <div\r\n *ngIf=\"!external.value && !canEditTarget && label.value\"\r\n class=\"info\"\r\n >\r\n <span class=\"label\">label</span>{{ label.value }}\r\n </div>\r\n </div>\r\n\r\n <!-- gid -->\r\n <div>\r\n <mat-form-field *ngIf=\"external.value || canEditTarget\">\r\n <mat-label>GID</mat-label>\r\n <input matInput [formControl]=\"gid\" />\r\n <mat-error\r\n *ngIf=\"$any(gid).errors?.required && (gid.dirty || gid.touched)\"\r\n >GID required</mat-error\r\n >\r\n <mat-error\r\n *ngIf=\"$any(gid).errors?.maxLength && (gid.dirty || gid.touched)\"\r\n >GID too long</mat-error\r\n >\r\n </mat-form-field>\r\n <div *ngIf=\"!external.value && !canEditTarget && gid.value\" class=\"info\">\r\n <span class=\"label\">GID</span> <span class=\"gid\">{{ gid.value }}</span>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- BY ITEM -->\r\n <div *ngIf=\"!external.value\">\r\n <fieldset *ngIf=\"!byTypeMode.value\" class=\"form-row\">\r\n <legend>pin filters</legend>\r\n <!-- item filter -->\r\n <cadmus-ref-lookup\r\n [service]=\"itemLookupService\"\r\n label=\"item\"\r\n (itemChange)=\"onItemLookupChange($event)\"\r\n ></cadmus-ref-lookup>\r\n\r\n <!-- part filter -->\r\n <mat-form-field *ngIf=\"itemParts.length\">\r\n <mat-label>part</mat-label>\r\n <mat-select [formControl]=\"itemPart\">\r\n <mat-option [value]=\"null\">(any)</mat-option>\r\n <mat-option *ngFor=\"let p of itemParts\" [value]=\"p\">{{\r\n p.typeId | flatLookup : modelEntries : \"id\" : \"value\"\r\n }}</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </fieldset>\r\n\r\n <!-- BY TYPE -->\r\n <div *ngIf=\"byTypeMode.value\">\r\n <!-- par type filter -->\r\n <mat-form-field *ngIf=\"partTypeKeys?.length\">\r\n <mat-label>part type</mat-label>\r\n <mat-select [formControl]=\"partTypeKey\">\r\n <mat-option *ngFor=\"let k of partTypeKeys\" [value]=\"k\">{{\r\n k\r\n }}</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- PIN -->\r\n <div class=\"form-row\">\r\n <!-- pin lookup -->\r\n <cadmus-ref-lookup\r\n [service]=\"pinLookupService\"\r\n [baseFilter]=\"filter\"\r\n [options]=\"pinFilterOptions\"\r\n label=\"pin\"\r\n (itemChange)=\"onPinLookupChange($event)\"\r\n ></cadmus-ref-lookup>\r\n <!-- mode switcher -->\r\n <div>\r\n <mat-checkbox [formControl]=\"byTypeMode\" *ngIf=\"canSwitchMode\">\r\n by type</mat-checkbox\r\n >\r\n </div>\r\n </div>\r\n\r\n <!-- data -->\r\n <mat-expansion-panel id=\"data\" *ngIf=\"lookupData?.pin?.name\">\r\n <mat-expansion-panel-header>pin data</mat-expansion-panel-header>\r\n <!-- table -->\r\n <table>\r\n <thead>\r\n <th></th>\r\n <th>source</th>\r\n <th>value</th>\r\n </thead>\r\n <tbody>\r\n <!-- pin -->\r\n <tr *ngIf=\"lookupData?.pin?.value\">\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"lookupData!.pin!.value\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td>pin</td>\r\n <td>{{ lookupData!.pin.value }}</td>\r\n </tr>\r\n <!-- item ID -->\r\n <tr *ngIf=\"lookupData?.pin?.itemId\">\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"lookupData!.pin!.itemId\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td>item ID</td>\r\n <td>{{ lookupData!.pin.itemId }}</td>\r\n </tr>\r\n <!-- item title -->\r\n <tr *ngIf=\"lookupData?.item?.title\">\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"lookupData!.item!.title\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td>item title</td>\r\n <td>{{ lookupData!.item!.title }}</td>\r\n </tr>\r\n <!-- part ID -->\r\n <tr *ngIf=\"lookupData?.pin?.partId\">\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"lookupData!.pin!.partId\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td>part ID</td>\r\n <td>{{ lookupData!.pin.partId }}</td>\r\n </tr>\r\n <!-- part type ID -->\r\n <tr *ngIf=\"lookupData?.pin?.partTypeId\">\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"lookupData!.pin!.partTypeId\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td>part type ID</td>\r\n <td>{{ lookupData!.pin!.partTypeId }}</td>\r\n </tr>\r\n <!-- part role ID -->\r\n <tr *ngIf=\"lookupData?.pin?.roleId\">\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"lookupData!.pin!.roleId!\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td>part role ID</td>\r\n <td>{{ lookupData!.pin!.roleId }}</td>\r\n </tr>\r\n\r\n <!-- part's metadata -->\r\n <tr\r\n *ngFor=\"\r\n let m of lookupData?.metaPart?.metadata || [];\r\n let i = index\r\n \"\r\n >\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"m.value\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td class=\"metadata\">{{ m.name }}</td>\r\n <td class=\"metadata\">{{ m.value }}</td>\r\n </tr>\r\n </tbody>\r\n </table>\r\n </mat-expansion-panel>\r\n </div>\r\n\r\n <!-- buttons -->\r\n <div>\r\n <button mat-flat-button type=\"button\" (click)=\"close()\">\r\n <mat-icon color=\"warn\">close</mat-icon>\r\n </button>\r\n <button mat-flat-button type=\"submit\" [disabled]=\"form.invalid\">\r\n <mat-icon color=\"primary\">check_circle</mat-icon> target\r\n </button>\r\n </div>\r\n</form>\r\n", styles: [".metadata{color:#4a3001}.info{border:1px solid silver;border-radius:6px;background-color:silver;color:#fff;padding:4px;margin:8px 0}.info .label{background-color:#fff;color:silver;margin:0 6px}#data{margin:8px 0}table{border:1px solid silver;border-radius:6px;padding:4px;margin-top:8px}tr:nth-child(odd){background-color:#f0f0f0}th{font-weight:400;text-align:left;color:silver}.form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}.label{border:1px solid silver;border-radius:6px;padding:4px}fieldset{border:1px solid silver;border-radius:6px;padding:4px 8px;margin-bottom:8px}legend{color:silver}\n"], dependencies: [{ kind: "directive", type: i6.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i5.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i5.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: i5.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i5.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i5.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i5.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i7.CdkCopyToClipboard, selector: "[cdkCopyToClipboard]", inputs: ["cdkCopyToClipboard", "cdkCopyToClipboardAttempts"], outputs: ["cdkCopyToClipboardCopied"] }, { kind: "component", type: i8.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i8.MatIconButton, selector: "button[mat-icon-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i9.MatCheckbox, selector: "mat-checkbox", inputs: ["disableRipple", "color", "tabIndex"], exportAs: ["matCheckbox"] }, { kind: "component", type: i10.MatExpansionPanel, selector: "mat-expansion-panel", inputs: ["disabled", "expanded", "hideToggle", "togglePosition"], outputs: ["opened", "closed", "expandedChange", "afterExpand", "afterCollapse"], exportAs: ["matExpansionPanel"] }, { kind: "component", type: i10.MatExpansionPanelHeader, selector: "mat-expansion-panel-header", inputs: ["tabIndex", "expandedHeight", "collapsedHeight"] }, { kind: "component", type: i11.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i11.MatLabel, selector: "mat-label" }, { kind: "directive", type: i11.MatError, selector: "mat-error, [matError]", inputs: ["id"] }, { kind: "component", type: i12.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i13.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: "component", type: i14.MatSelect, selector: "mat-select", inputs: ["disabled", "disableRipple", "tabIndex", "hideSingleSelectionIndicator"], exportAs: ["matSelect"] }, { kind: "component", type: i15.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "component", type: i16.RefLookupComponent, selector: "cadmus-ref-lookup", inputs: ["label", "limit", "baseFilter", "service", "item", "required", "hasMore", "linkTemplate", "optDialog", "options"], outputs: ["itemChange", "moreRequest"] }, { kind: "pipe", type: i17.FlatLookupPipe, name: "flatLookup" }] }); }
382
+ }
383
+ export { PinTargetLookupComponent };
384
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.1", ngImport: i0, type: PinTargetLookupComponent, decorators: [{
385
+ type: Component,
386
+ args: [{ selector: 'cadmus-pin-target-lookup', template: "<form [formGroup]=\"form\" (submit)=\"save()\">\r\n <!-- external -->\r\n <mat-checkbox [formControl]=\"external\">external</mat-checkbox>\r\n\r\n <div class=\"form-row\">\r\n <!-- label -->\r\n <div>\r\n <mat-form-field *ngIf=\"external.value || canEditTarget\">\r\n <mat-label>label</mat-label>\r\n <input matInput [formControl]=\"label\" />\r\n <mat-error\r\n *ngIf=\"$any(label).errors?.required && (label.dirty || label.touched)\"\r\n >label required</mat-error\r\n >\r\n <mat-error\r\n *ngIf=\"\r\n $any(label).errors?.maxLength && (label.dirty || label.touched)\r\n \"\r\n >label too long</mat-error\r\n >\r\n </mat-form-field>\r\n <div\r\n *ngIf=\"!external.value && !canEditTarget && label.value\"\r\n class=\"info\"\r\n >\r\n <span class=\"label\">label</span>{{ label.value }}\r\n </div>\r\n </div>\r\n\r\n <!-- gid -->\r\n <div>\r\n <mat-form-field *ngIf=\"external.value || canEditTarget\">\r\n <mat-label>GID</mat-label>\r\n <input matInput [formControl]=\"gid\" />\r\n <mat-error\r\n *ngIf=\"$any(gid).errors?.required && (gid.dirty || gid.touched)\"\r\n >GID required</mat-error\r\n >\r\n <mat-error\r\n *ngIf=\"$any(gid).errors?.maxLength && (gid.dirty || gid.touched)\"\r\n >GID too long</mat-error\r\n >\r\n </mat-form-field>\r\n <div *ngIf=\"!external.value && !canEditTarget && gid.value\" class=\"info\">\r\n <span class=\"label\">GID</span> <span class=\"gid\">{{ gid.value }}</span>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- BY ITEM -->\r\n <div *ngIf=\"!external.value\">\r\n <fieldset *ngIf=\"!byTypeMode.value\" class=\"form-row\">\r\n <legend>pin filters</legend>\r\n <!-- item filter -->\r\n <cadmus-ref-lookup\r\n [service]=\"itemLookupService\"\r\n label=\"item\"\r\n (itemChange)=\"onItemLookupChange($event)\"\r\n ></cadmus-ref-lookup>\r\n\r\n <!-- part filter -->\r\n <mat-form-field *ngIf=\"itemParts.length\">\r\n <mat-label>part</mat-label>\r\n <mat-select [formControl]=\"itemPart\">\r\n <mat-option [value]=\"null\">(any)</mat-option>\r\n <mat-option *ngFor=\"let p of itemParts\" [value]=\"p\">{{\r\n p.typeId | flatLookup : modelEntries : \"id\" : \"value\"\r\n }}</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </fieldset>\r\n\r\n <!-- BY TYPE -->\r\n <div *ngIf=\"byTypeMode.value\">\r\n <!-- par type filter -->\r\n <mat-form-field *ngIf=\"partTypeKeys?.length\">\r\n <mat-label>part type</mat-label>\r\n <mat-select [formControl]=\"partTypeKey\">\r\n <mat-option *ngFor=\"let k of partTypeKeys\" [value]=\"k\">{{\r\n k\r\n }}</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n\r\n <!-- PIN -->\r\n <div class=\"form-row\">\r\n <!-- pin lookup -->\r\n <cadmus-ref-lookup\r\n [service]=\"pinLookupService\"\r\n [baseFilter]=\"filter\"\r\n [options]=\"pinFilterOptions\"\r\n label=\"pin\"\r\n (itemChange)=\"onPinLookupChange($event)\"\r\n ></cadmus-ref-lookup>\r\n <!-- mode switcher -->\r\n <div>\r\n <mat-checkbox [formControl]=\"byTypeMode\" *ngIf=\"canSwitchMode\">\r\n by type</mat-checkbox\r\n >\r\n </div>\r\n </div>\r\n\r\n <!-- data -->\r\n <mat-expansion-panel id=\"data\" *ngIf=\"lookupData?.pin?.name\">\r\n <mat-expansion-panel-header>pin data</mat-expansion-panel-header>\r\n <!-- table -->\r\n <table>\r\n <thead>\r\n <th></th>\r\n <th>source</th>\r\n <th>value</th>\r\n </thead>\r\n <tbody>\r\n <!-- pin -->\r\n <tr *ngIf=\"lookupData?.pin?.value\">\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"lookupData!.pin!.value\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td>pin</td>\r\n <td>{{ lookupData!.pin.value }}</td>\r\n </tr>\r\n <!-- item ID -->\r\n <tr *ngIf=\"lookupData?.pin?.itemId\">\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"lookupData!.pin!.itemId\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td>item ID</td>\r\n <td>{{ lookupData!.pin.itemId }}</td>\r\n </tr>\r\n <!-- item title -->\r\n <tr *ngIf=\"lookupData?.item?.title\">\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"lookupData!.item!.title\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td>item title</td>\r\n <td>{{ lookupData!.item!.title }}</td>\r\n </tr>\r\n <!-- part ID -->\r\n <tr *ngIf=\"lookupData?.pin?.partId\">\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"lookupData!.pin!.partId\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td>part ID</td>\r\n <td>{{ lookupData!.pin.partId }}</td>\r\n </tr>\r\n <!-- part type ID -->\r\n <tr *ngIf=\"lookupData?.pin?.partTypeId\">\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"lookupData!.pin!.partTypeId\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td>part type ID</td>\r\n <td>{{ lookupData!.pin!.partTypeId }}</td>\r\n </tr>\r\n <!-- part role ID -->\r\n <tr *ngIf=\"lookupData?.pin?.roleId\">\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"lookupData!.pin!.roleId!\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td>part role ID</td>\r\n <td>{{ lookupData!.pin!.roleId }}</td>\r\n </tr>\r\n\r\n <!-- part's metadata -->\r\n <tr\r\n *ngFor=\"\r\n let m of lookupData?.metaPart?.metadata || [];\r\n let i = index\r\n \"\r\n >\r\n <td>\r\n <button\r\n type=\"button\"\r\n mat-icon-button\r\n color=\"primary\"\r\n [cdkCopyToClipboard]=\"m.value\"\r\n (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n >\r\n <mat-icon>content_copy</mat-icon>\r\n </button>\r\n </td>\r\n <td class=\"metadata\">{{ m.name }}</td>\r\n <td class=\"metadata\">{{ m.value }}</td>\r\n </tr>\r\n </tbody>\r\n </table>\r\n </mat-expansion-panel>\r\n </div>\r\n\r\n <!-- buttons -->\r\n <div>\r\n <button mat-flat-button type=\"button\" (click)=\"close()\">\r\n <mat-icon color=\"warn\">close</mat-icon>\r\n </button>\r\n <button mat-flat-button type=\"submit\" [disabled]=\"form.invalid\">\r\n <mat-icon color=\"primary\">check_circle</mat-icon> target\r\n </button>\r\n </div>\r\n</form>\r\n", styles: [".metadata{color:#4a3001}.info{border:1px solid silver;border-radius:6px;background-color:silver;color:#fff;padding:4px;margin:8px 0}.info .label{background-color:#fff;color:silver;margin:0 6px}#data{margin:8px 0}table{border:1px solid silver;border-radius:6px;padding:4px;margin-top:8px}tr:nth-child(odd){background-color:#f0f0f0}th{font-weight:400;text-align:left;color:silver}.form-row{display:flex;gap:8px;align-items:center;flex-wrap:wrap}.form-row *{flex:0 0 auto}.label{border:1px solid silver;border-radius:6px;padding:4px}fieldset{border:1px solid silver;border-radius:6px;padding:4px 8px;margin-bottom:8px}legend{color:silver}\n"] }]
387
+ }], ctorParameters: function () { return [{ type: undefined, decorators: [{
388
+ type: Inject,
389
+ args: ['indexLookupDefinitions']
390
+ }] }, { type: i1.ItemRefLookupService }, { type: i2.PinRefLookupService }, { type: i3.ItemService }, { type: i3.ThesaurusService }, { type: i4.MatSnackBar }, { type: i5.FormBuilder }]; }, propDecorators: { pinByTypeMode: [{
391
+ type: Input
392
+ }], canSwitchMode: [{
393
+ type: Input
394
+ }], canEditTarget: [{
395
+ type: Input
396
+ }], lookupDefinitions: [{
397
+ type: Input
398
+ }], target: [{
399
+ type: Input
400
+ }], editorClose: [{
401
+ type: Output
402
+ }], targetChange: [{
403
+ type: Output
404
+ }] } });
405
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pin-target-lookup.component.js","sourceRoot":"","sources":["../../../../../../projects/myrmidon/cadmus-refs-asserted-ids/src/lib/pin-target-lookup/pin-target-lookup.component.ts","../../../../../../projects/myrmidon/cadmus-refs-asserted-ids/src/lib/pin-target-lookup/pin-target-lookup.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EACT,YAAY,EACZ,MAAM,EACN,KAAK,EAGL,MAAM,GACP,MAAM,eAAe,CAAC;AACvB,OAAO,EAIL,UAAU,GACX,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAEL,YAAY,EACZ,oBAAoB,EACpB,QAAQ,EACR,IAAI,GACL,MAAM,MAAM,CAAC;;;;;;;;;;;;;;;;;;;AAmBd,4BAA4B;AAC5B,MAAM,gBAAgB,GAAG,mBAAmB,CAAC;AAmC7C;;;;;;GAMG;AACH,MAKa,wBAAwB;IAMnC;;;OAGG;IACH,IACW,aAAa;QACtB,OAAO,IAAI,CAAC,UAAU,EAAE,KAAK,CAAC;IAChC,CAAC;IACD,IAAW,aAAa,CAAC,KAA0B;QACjD,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE;YACpB,IAAI,CAAC,oBAAoB,GAAG,KAAK,CAAC;SACnC;aAAM;YACL,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,KAAK,IAAI,KAAK,CAAC,CAAC;YACzC,IAAI,CAAC,UAAU,CAAC,sBAAsB,EAAE,CAAC;SAC1C;IACH,CAAC;IAwBD;;OAEG;IACH,IACW,MAAM;QACf,OAAO,IAAI,CAAC,OAAO,CAAC;IACtB,CAAC;IACD,IAAW,MAAM,CAAC,KAAmC;QACnD,IAAI,IAAI,CAAC,OAAO,KAAK,KAAK,EAAE;YAC1B,OAAO;SACR;QACD,IAAI,CAAC,OAAO,GAAG,KAAK,IAAI,SAAS,CAAC;QAClC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAChC,CAAC;IAmCD,YAEU,iBAAyC,EAC1C,iBAAuC,EACvC,gBAAqC,EACpC,YAAyB,EACzB,YAA8B,EAC9B,SAAsB,EAC9B,WAAwB;QANhB,sBAAiB,GAAjB,iBAAiB,CAAwB;QAC1C,sBAAiB,GAAjB,iBAAiB,CAAsB;QACvC,qBAAgB,GAAhB,gBAAgB,CAAqB;QACpC,iBAAY,GAAZ,YAAY,CAAa;QACzB,iBAAY,GAAZ,YAAY,CAAkB;QAC9B,cAAS,GAAT,SAAS,CAAa;QAG9B,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;QACvB,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;QACpB,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC;QAChB,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;QACvB,4DAA4D;QAC5D,8CAA8C;QAC9C,IAAI,CAAC,MAAM,GAAG;YACZ,IAAI,EAAE,EAAE;YACR,KAAK,EAAE,EAAE;SACV,CAAC;QACF,OAAO;QACP,IAAI,CAAC,IAAI,GAAG,WAAW,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QACtC,IAAI,CAAC,QAAQ,GAAG,WAAW,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QAC1C,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QAC7C,IAAI,CAAC,GAAG,GAAG,WAAW,CAAC,OAAO,CAAC,IAAI,EAAE;YACnC,UAAU,CAAC,QAAQ;YACnB,UAAU,CAAC,SAAS,CAAC,GAAG,CAAC;SAC1B,CAAC,CAAC;QACH,IAAI,CAAC,KAAK,GAAG,WAAW,CAAC,OAAO,CAAC,IAAI,EAAE;YACrC,UAAU,CAAC,QAAQ;YACnB,UAAU,CAAC,SAAS,CAAC,GAAG,CAAC;SAC1B,CAAC,CAAC;QACH,IAAI,CAAC,UAAU,GAAG,WAAW,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,WAAW,EAAE,IAAI,EAAE,CAAC,CAAC;QACpE,IAAI,CAAC,QAAQ,GAAG,WAAW,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,WAAW,EAAE,IAAI,EAAE,CAAC,CAAC;QAClE,IAAI,CAAC,IAAI,GAAG,WAAW,CAAC,KAAK,CAAC;YAC5B,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,WAAW,EAAE,IAAI,CAAC,WAAW;YAC7B,GAAG,EAAE,IAAI,CAAC,GAAG;YACb,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,UAAU,EAAE,IAAI,CAAC,UAAU;YAC3B,QAAQ,EAAE,IAAI,CAAC,QAAQ;SACxB,CAAC,CAAC;QACH,SAAS;QACT,IAAI,CAAC,YAAY,GAAG,IAAI,YAAY,EAAa,CAAC;QAClD,IAAI,CAAC,WAAW,GAAG,IAAI,YAAY,EAAO,CAAC;IAC7C,CAAC;IAEO,WAAW;QACjB,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAC3B,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;IAC7B,CAAC;IAEO,SAAS;QACf,0CAA0C;QAC1C,IAAI,CAAC,IAAI,CAAC,iBAAiB,EAAE;YAC3B,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,CAAC;SACjD;QACD,2CAA2C;QAC3C,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;QAExD,mDAAmD;QACnD,qDAAqD;QACrD,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE;YAC7B,IAAI,IAAI,CAAC,YAAY,EAAE,MAAM,EAAE;gBAC7B,+CAA+C;gBAC/C,IAAI,CAAC,iBAAiB,GAAG,EAAE,CAAC;gBAC5B,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;oBAC9B,IAAI,CAAC,iBAAkB,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG;wBACjC,IAAI,EAAE,CAAC,CAAC,KAAK;wBACb,MAAM,EAAE,CAAC,CAAC,EAAE;qBACb,CAAC;gBACJ,CAAC,CAAC,CAAC;gBACH,uCAAuC;gBACvC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;aAC3D;SACF;QAED,uCAAuC;QACvC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE;YAC7B,IAAI,CAAC,WAAW,EAAE,CAAC;SACpB;aAAM;YACL,kDAAkD;YAClD,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE;gBAClC,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC;aACjD;SACF;IACH,CAAC;IAEO,WAAW;QACjB,OAAO,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC;IACxE,CAAC;IAEO,aAAa;QACnB,OAAO,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC;IAC5E,CAAC;IAEM,QAAQ;QACb,6BAA6B;QAC7B,IAAI,IAAI,CAAC,oBAAoB,EAAE;YAC7B,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;SAChC;QAED,wDAAwD;QACxD,IAAI,CAAC,KAAK,CAAC,IAAI,CACb,IAAI,CAAC,IAAI,CAAC,YAAY;aACnB,IAAI,CAAC,oBAAoB,EAAE,EAAE,YAAY,CAAC,GAAG,CAAC,CAAC;aAC/C,SAAS,CAAC,CAAC,IAAI,EAAE,EAAE;YAClB,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;YAC7B,IAAI,CAAC,SAAS,GAAG,IAAI,EAAE,KAAK,IAAI,EAAE,CAAC;YACnC,IAAI,CAAC,MAAM,GAAG;gBACZ,GAAG,IAAI,CAAC,MAAM;gBACd,MAAM,EAAE,IAAI,EAAE,EAAE;aACjB,CAAC;QACJ,CAAC,CAAC,CACL,CAAC;QAEF,8DAA8D;QAC9D,IAAI,CAAC,KAAK,CAAC,IAAI,CACb,IAAI,CAAC,QAAQ,CAAC,YAAY;aACvB,IAAI,CAAC,oBAAoB,EAAE,EAAE,YAAY,CAAC,GAAG,CAAC,CAAC;aAC/C,SAAS,CAAC,CAAC,IAAI,EAAE,EAAE;YAClB,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,IAAI,IAAI,CAAC,GAAG,CAAC,QAAQ,EAAE;gBACxC,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC;aACpC;YACD,IAAI,CAAC,MAAM,GAAG;gBACZ,GAAG,IAAI,CAAC,MAAM;gBACd,MAAM,EAAE,IAAI,EAAE,EAAE;aACjB,CAAC;YACF,IAAI,CAAC,YAAY,EAAE,CAAC;QACtB,CAAC,CAAC,CACL,CAAC;QAEF,wDAAwD;QACxD,IAAI,CAAC,KAAK,CAAC,IAAI,CACb,IAAI,CAAC,WAAW,CAAC,YAAY;aAC1B,IAAI,CAAC,oBAAoB,EAAE,EAAE,YAAY,CAAC,GAAG,CAAC,CAAC;aAC/C,SAAS,CAAC,CAAC,GAAG,EAAE,EAAE;YACjB,IAAI,CAAC,gBAAgB,GAAG,GAAG;gBACzB,CAAC,CAAC,IAAI,CAAC,iBAAkB,CAAC,GAAG,CAAC;gBAC9B,CAAC,CAAC,SAAS,CAAC;QAChB,CAAC,CAAC,CACL,CAAC;QAEF,qCAAqC;QACrC,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC,aAAa,EAAE,IAAI,CAAC,CAAC,SAAS,CAAC;YAC5D,IAAI,EAAE,CAAC,CAAC,EAAE,EAAE;gBACV,IAAI,CAAC,YAAY,GAAG,CAAC,CAAC,OAAO,IAAI,EAAE,CAAC;gBACpC,IAAI,IAAI,CAAC,YAAY,EAAE,MAAM,EAAE;oBAC7B,IAAI,CAAC,SAAS,EAAE,CAAC;iBAClB;qBAAM;oBACL,IAAI,CAAC,WAAW,EAAE,CAAC;iBACpB;YACH,CAAC;YACD,KAAK,EAAE,GAAG,EAAE;gBACV,IAAI,CAAC,WAAW,EAAE,CAAC;YACrB,CAAC;SACF,CAAC,CAAC;IACL,CAAC;IAEM,WAAW;QAChB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;YAC1C,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC;SAC7B;IACH,CAAC;IAEO,QAAQ;QACd,6DAA6D;QAC7D,yBAAyB;QACzB,MAAM,GAAG,GAAG,IAAI,CAAC,UAAU,EAAE,GAAG,CAAC;QACjC,IAAI,CAAC,GAAG,EAAE;YACR,OAAO,IAAI,CAAC;SACb;QACD,OAAO,GAAG,CAAC,MAAM;YACf,CAAC,CAAC,IAAI,GAAG,CAAC,MAAM,IAAI,GAAG,CAAC,KAAK,EAAE;YAC/B,CAAC,CAAC,IAAI,GAAG,CAAC,MAAM,IAAI,GAAG,CAAC,KAAK,EAAE,CAAC;IACpC,CAAC;IAEO,UAAU;QAChB,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,GAAG,EAAE;YACzB,OAAO,IAAI,CAAC;SACb;QACD,MAAM,EAAE,GAAa,EAAE,CAAC;QACxB,YAAY;QACZ,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAM,CAAC,CAAC;QACpC,aAAa;QACb,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACf,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,IAAK,CAAC,KAAK,IAAI,IAAI,CAAC,UAAU,CAAC,IAAK,CAAC,EAAE,CAAC,CAAC;QACjE,qBAAqB;QACrB,MAAM,CAAC,GAAG,IAAI,CAAC,YAAY,EAAE,IAAI,CAC/B,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,IAAI,CAAC,UAAW,CAAC,GAAG,CAAC,UAAU,CAChD,CAAC;QACF,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACd,EAAE,CAAC,IAAI,CAAC,CAAC,EAAE,KAAK,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QACpD,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,MAAM,EAAE;YAC9B,EAAE,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC;SAC5C;QACD,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACb,OAAO,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACrB,CAAC;IAEO,SAAS;QACf,IAAI,IAAI,CAAC,QAAQ,CAAC,KAAK,EAAE;YACvB,OAAO;gBACL,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,KAAK,IAAI,EAAE;gBACzB,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,KAAK,IAAI,EAAE;aAC9B,CAAC;SACH;aAAM;YACL,MAAM,GAAG,GAAG,IAAI,CAAC,UAAU,EAAE,GAAG,CAAC;YACjC,OAAO;gBACL,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,KAAK,IAAI,EAAE;gBACzB,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,KAAK,IAAI,EAAE;gBAC7B,MAAM,EAAE,GAAG,EAAE,MAAM,IAAI,EAAE;gBACzB,MAAM,EAAE,GAAG,EAAE,MAAM,IAAI,EAAE;gBACzB,UAAU,EAAE,GAAG,EAAE,UAAU,IAAI,EAAE;gBACjC,MAAM,EAAE,GAAG,EAAE,MAAM,IAAI,EAAE;gBACzB,IAAI,EAAE,GAAG,EAAE,IAAI,IAAI,EAAE;gBACrB,KAAK,EAAE,GAAG,EAAE,KAAK,IAAI,EAAE;aACxB,CAAC;SACH;IACH,CAAC;IAEO,YAAY;QAClB,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;QAE1B,IAAI,IAAI,CAAC,WAAW,EAAE,EAAE;YACtB,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC;YACnC,IAAI,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC;SACxB;QACD,IAAI,IAAI,CAAC,aAAa,EAAE,EAAE;YACxB,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC;YACvC,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC;SAC1B;QACD,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,SAAS,EAAE,CAAC;QAE/B,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;IAC7B,CAAC;IAEO,UAAU,CAAC,MAAkB;QACnC,IAAI,IAAI,CAAC,aAAa,EAAE;YACtB,OAAO;SACR;QACD,6BAA6B;QAC7B,IAAI,CAAC,MAAM,EAAE;YACX,IAAI,CAAC,UAAU,GAAG,SAAS,CAAC;YAC5B,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC;YAClB,OAAO;SACR;QACD,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACrC,IAAI,CAAC,UAAU,GAAG;YAChB,GAAG,EAAE;gBACH,MAAM,EAAE,MAAM,CAAC,MAAM,IAAI,EAAE;gBAC3B,MAAM,EAAE,MAAM,CAAC,MAAM,IAAI,EAAE;gBAC3B,UAAU,EAAE,MAAM,CAAC,UAAU,IAAI,EAAE;gBACnC,MAAM,EAAE,MAAM,CAAC,MAAM,IAAI,EAAE;gBAC3B,IAAI,EAAE,MAAM,CAAC,IAAI,IAAI,EAAE;gBACvB,KAAK,EAAE,MAAM,CAAC,KAAK,IAAI,EAAE;aAC1B;SACF,CAAC;QACF,WAAW;QACX,IAAI,MAAM,CAAC,MAAM,EAAE;YACjB,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC,SAAS,CAAC;gBACvD,IAAI,EAAE,CAAC,IAAI,EAAE,EAAE;oBACb,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;oBACzB,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC;gBAC7B,CAAC;gBACD,KAAK,EAAE,CAAC,KAAK,EAAE,EAAE;oBACf,IAAI,KAAK,EAAE;wBACT,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC;qBACtC;gBACH,CAAC;aACF,CAAC,CAAC;SACJ;IACH,CAAC;IAED;;;;;OAKG;IACI,kBAAkB,CAAC,IAAU;QAClC,oBAAoB;QACpB,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,EAAE,IAAI,CAAC,CAAC,SAAS,CAAC;YACjD,IAAI,EAAE,CAAC,CAAC,EAAE,EAAE;gBACV,iDAAiD;gBACjD,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;gBACtB,IAAI,CAAC,YAAY,EAAE,CAAC;YACtB,CAAC;YACD,KAAK,EAAE,CAAC,KAAK,EAAE,EAAE;gBACf,IAAI,KAAK,EAAE;oBACT,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC;iBACtC;gBACD,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;gBAC7B,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;gBACpB,IAAI,CAAC,YAAY,EAAE,CAAC;YACtB,CAAC;SACF,CAAC,CAAC;IACL,CAAC;IAEO,YAAY,CAAC,GAAgB;QACnC,QAAQ,CAAC;YACP,IAAI,EAAE,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,EAAE,KAAK,CAAC;YAClD,IAAI,EAAE,IAAI,CAAC,YAAY,CAAC,sBAAsB,CAC5C,GAAG,CAAC,MAAM,EACV,gBAAgB,CACjB;SACF,CAAC;aACC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;aACb,SAAS,CAAC;YACT,IAAI,EAAE,CAAC,MAAM,EAAE,EAAE;gBACf,IAAI,CAAC,UAAU,GAAG;oBAChB,GAAG,EAAE,GAAG;oBACR,IAAI,EAAE,MAAM,CAAC,IAAI;oBACjB,QAAQ,EAAE,MAAM,CAAC,IAAoB;iBACtC,CAAC;gBACF,IAAI,CAAC,YAAY,EAAE,CAAC;YACtB,CAAC;YACD,KAAK,EAAE,CAAC,KAAK,EAAE,EAAE;gBACf,IAAI,CAAC,UAAU,GAAG,SAAS,CAAC;gBAC5B,OAAO,CAAC,KAAK,CACX,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,6BAA6B,CAC9D,CAAC;YACJ,CAAC;SACF,CAAC,CAAC;IACP,CAAC;IAED;;;;;;;;OAQG;IACI,iBAAiB,CAAC,IAAiB;QACxC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;IAC1B,CAAC;IAEM,QAAQ;QACb,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,qBAAqB,EAAE,IAAI,EAAE;YAC/C,QAAQ,EAAE,IAAI;SACf,CAAC,CAAC;IACL,CAAC;IAEM,KAAK;QACV,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC;IAC1B,CAAC;IAEM,IAAI;QACT,IAAI,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE;YACrB,OAAO;SACR;QACD,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,SAAS,EAAE,CAAC;QAChC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACvC,CAAC;8GAjcU,wBAAwB,kBA8FzB,wBAAwB;kGA9FvB,wBAAwB,6SCxFrC,+7QAuPA;;SD/Ja,wBAAwB;2FAAxB,wBAAwB;kBALpC,SAAS;+BACE,0BAA0B;;0BAkGjC,MAAM;2BAAC,wBAAwB;8NAnFvB,aAAa;sBADvB,KAAK;gBAiBC,aAAa;sBADnB,KAAK;gBAOC,aAAa;sBADnB,KAAK;gBAWC,iBAAiB;sBADvB,KAAK;gBAOK,MAAM;sBADhB,KAAK;gBAgBC,WAAW;sBADjB,MAAM;gBAOA,YAAY;sBADlB,MAAM","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  Inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output,\n} from '@angular/core';\nimport {\n  FormBuilder,\n  FormControl,\n  FormGroup,\n  Validators,\n} from '@angular/forms';\nimport {\n  Subscription,\n  debounceTime,\n  distinctUntilChanged,\n  forkJoin,\n  take,\n} from 'rxjs';\nimport { MatSnackBar } from '@angular/material/snack-bar';\n\nimport { ItemService, ThesaurusService } from '@myrmidon/cadmus-api';\nimport {\n  DataPinInfo,\n  IndexLookupDefinition,\n  IndexLookupDefinitions,\n  Item,\n  Part,\n  ThesaurusEntry,\n} from '@myrmidon/cadmus-core';\n\nimport {\n  PinRefLookupFilter,\n  PinRefLookupService,\n} from '../services/pin-ref-lookup.service';\nimport { ItemRefLookupService } from '../services/item-ref-lookup.service';\n\n// from Cadmus general parts\nconst METADATA_PART_ID = 'it.vedph.metadata';\ninterface MetadataPart extends Part {\n  metadata: {\n    type?: string;\n    name: string;\n    value: string;\n  }[];\n}\n\n/**\n * Pin lookup data used internally by the component.\n */\nexport interface PinLookupData {\n  pin: DataPinInfo;\n  item?: Item;\n  metaPart?: MetadataPart;\n}\n\n/**\n * A pin-based target. This includes pin's name and value, and\n * the item's ID and optional part IDs. The label is a user friendly\n * string representation of the target, while the gid is a globally\n * unique identifier for the target.\n */\nexport interface PinTarget {\n  gid: string;\n  label: string;\n  itemId?: string;\n  partId?: string;\n  partTypeId?: string;\n  roleId?: string;\n  name?: string;\n  value?: string;\n}\n\n/*\n * Scoped pin-based lookup component. This component provides a list\n * of pin-based searches, with a lookup control. Whenever the user\n * picks a pin value, he gets the details about its item and part, and\n * item's metadata part, if any. He can then use these data to build\n * some EID by variously assembling these components.\n */\n@Component({\n  selector: 'cadmus-pin-target-lookup',\n  templateUrl: './pin-target-lookup.component.html',\n  styleUrls: ['./pin-target-lookup.component.css'],\n})\nexport class PinTargetLookupComponent implements OnInit, OnDestroy {\n  private _subs: Subscription[];\n  private _target: PinTarget | undefined;\n  private _noFormUpdate?: boolean;\n  private _startWithByTypeMode?: boolean;\n\n  /**\n   * True when the by-type pin lookup mode is active.\n   * User can change mode unless modeSwitching is false.\n   */\n  @Input()\n  public get pinByTypeMode(): boolean | undefined {\n    return this.byTypeMode?.value;\n  }\n  public set pinByTypeMode(value: boolean | undefined) {\n    if (!this.byTypeMode) {\n      this._startWithByTypeMode = value;\n    } else {\n      this.byTypeMode.setValue(value || false);\n      this.byTypeMode.updateValueAndValidity();\n    }\n  }\n\n  /**\n   * True when the user can switch between by-type and by-item mode.\n   */\n  @Input()\n  public canSwitchMode?: boolean;\n  /**\n   * True when the user can edit the target's gid/label for internal\n   * targets.\n   */\n  @Input()\n  public canEditTarget?: boolean;\n  /**\n   * The lookup definitions to be used for the by-type lookup. If\n   * not specified, the lookup definitions will be got via injection\n   * when available; if the injected definitions are empty, the\n   * lookup definitions will be built from the model-types thesaurus;\n   * if this is not available either, the by-type lookup will be\n   * disabled.\n   */\n  @Input()\n  public lookupDefinitions?: IndexLookupDefinitions;\n\n  /**\n   * The target to be edited.\n   */\n  @Input()\n  public get target(): PinTarget | undefined | null {\n    return this._target;\n  }\n  public set target(value: PinTarget | undefined | null) {\n    if (this._target === value) {\n      return;\n    }\n    this._target = value || undefined;\n    this.updateForm(this._target);\n  }\n\n  /**\n   * Emitted when user closes the editor.\n   */\n  @Output()\n  public editorClose: EventEmitter<any>;\n  /**\n   * Emitted whenever the target changes.\n   * The event's value is the new target.\n   */\n  @Output()\n  public targetChange: EventEmitter<PinTarget>;\n\n  // by type\n  public modelEntries: ThesaurusEntry[];\n  public partTypeKeys: string[];\n  // by item\n  public itemParts: Part[];\n  // form - by item\n  public item: FormControl<Item | null>;\n  public itemPart: FormControl<Part | null>;\n  // form - by type\n  public partTypeKey: FormControl<string | null>;\n  // form - both\n  public gid: FormControl<string | null>;\n  public label: FormControl<string | null>;\n  public byTypeMode: FormControl<boolean>;\n  public external: FormControl<boolean>;\n  public form: FormGroup;\n\n  public filter: PinRefLookupFilter;\n  public pinFilterOptions?: IndexLookupDefinition;\n  public lookupData?: PinLookupData;\n\n  constructor(\n    @Inject('indexLookupDefinitions')\n    private _presetLookupDefs: IndexLookupDefinitions,\n    public itemLookupService: ItemRefLookupService,\n    public pinLookupService: PinRefLookupService,\n    private _itemService: ItemService,\n    private _thesService: ThesaurusService,\n    private _snackbar: MatSnackBar,\n    formBuilder: FormBuilder\n  ) {\n    this.partTypeKeys = [];\n    this.itemParts = [];\n    this._subs = [];\n    this.modelEntries = [];\n    // this is the default filter for the pin lookup, which will\n    // be merged with values provided by user here\n    this.filter = {\n      text: '',\n      limit: 10,\n    };\n    // form\n    this.item = formBuilder.control(null);\n    this.itemPart = formBuilder.control(null);\n    this.partTypeKey = formBuilder.control(null);\n    this.gid = formBuilder.control(null, [\n      Validators.required,\n      Validators.maxLength(300),\n    ]);\n    this.label = formBuilder.control(null, [\n      Validators.required,\n      Validators.maxLength(300),\n    ]);\n    this.byTypeMode = formBuilder.control(false, { nonNullable: true });\n    this.external = formBuilder.control(false, { nonNullable: true });\n    this.form = formBuilder.group({\n      item: this.item,\n      itemPart: this.itemPart,\n      partTypeKey: this.partTypeKey,\n      gid: this.gid,\n      label: this.label,\n      byTypeMode: this.byTypeMode,\n      external: this.external,\n    });\n    // events\n    this.targetChange = new EventEmitter<PinTarget>();\n    this.editorClose = new EventEmitter<any>();\n  }\n\n  private forceByItem(): void {\n    this.pinByTypeMode = false;\n    this.canSwitchMode = false;\n  }\n\n  private setupKeys(): void {\n    // use DI presets if no lookup definitions\n    if (!this.lookupDefinitions) {\n      this.lookupDefinitions = this._presetLookupDefs;\n    }\n    // keys are all the defined lookup searches\n    this.partTypeKeys = Object.keys(this.lookupDefinitions);\n\n    // if no keys, get them from thesaurus model-types;\n    // if this is not available, just force by item mode.\n    if (!this.partTypeKeys.length) {\n      if (this.modelEntries?.length) {\n        // set lookupDefinitions from thesaurus entries\n        this.lookupDefinitions = {};\n        this.modelEntries.forEach((e) => {\n          this.lookupDefinitions![e.value] = {\n            name: e.value,\n            typeId: e.id,\n          };\n        });\n        // set type keys from thesaurus entries\n        this.partTypeKeys = this.modelEntries.map((e) => e.value);\n      }\n    }\n\n    // if still no keys, force by item mode\n    if (!this.partTypeKeys.length) {\n      this.forceByItem();\n    } else {\n      // if there is only one key, set it as the default\n      if (this.partTypeKeys.length === 1) {\n        this.partTypeKey.setValue(this.partTypeKeys[0]);\n      }\n    }\n  }\n\n  private canBuildGid(): boolean {\n    return (!this.external.value && !this.gid.value) || this.gid.pristine;\n  }\n\n  private canBuildLabel(): boolean {\n    return (!this.external.value && !this.label.value) || this.label.pristine;\n  }\n\n  public ngOnInit(): void {\n    // set start mode if required\n    if (this._startWithByTypeMode) {\n      this.byTypeMode.setValue(true);\n    }\n\n    // whenever item changes, update item's parts and filter\n    this._subs.push(\n      this.item.valueChanges\n        .pipe(distinctUntilChanged(), debounceTime(300))\n        .subscribe((item) => {\n          this.itemPart.setValue(null);\n          this.itemParts = item?.parts || [];\n          this.filter = {\n            ...this.filter,\n            itemId: item?.id,\n          };\n        })\n    );\n\n    // whenever itemPart changes, update target and eventually gid\n    this._subs.push(\n      this.itemPart.valueChanges\n        .pipe(distinctUntilChanged(), debounceTime(300))\n        .subscribe((part) => {\n          if (!this.gid.value || this.gid.pristine) {\n            this.gid.setValue(this.buildGid());\n          }\n          this.filter = {\n            ...this.filter,\n            partId: part?.id,\n          };\n          this.updateTarget();\n        })\n    );\n\n    // whenever partTypeKey changes, update filter's options\n    this._subs.push(\n      this.partTypeKey.valueChanges\n        .pipe(distinctUntilChanged(), debounceTime(300))\n        .subscribe((key) => {\n          this.pinFilterOptions = key\n            ? this.lookupDefinitions![key]\n            : undefined;\n        })\n    );\n\n    // load model-types thesaurus entries\n    this._thesService.getThesaurus('model-types', true).subscribe({\n      next: (t) => {\n        this.modelEntries = t.entries || [];\n        if (this.modelEntries?.length) {\n          this.setupKeys();\n        } else {\n          this.forceByItem();\n        }\n      },\n      error: () => {\n        this.forceByItem();\n      },\n    });\n  }\n\n  public ngOnDestroy(): void {\n    for (let i = 0; i < this._subs.length; i++) {\n      this._subs[i].unsubscribe();\n    }\n  }\n\n  private buildGid(): string | null {\n    // the GID is the part ID if any, or the item ID, followed by\n    // the pin's value (=EID)\n    const pin = this.lookupData?.pin;\n    if (!pin) {\n      return null;\n    }\n    return pin.partId\n      ? `P${pin.partId}/${pin.value}`\n      : `I${pin.itemId}/${pin.value}`;\n  }\n\n  private buildLabel(): string | null {\n    if (!this.lookupData?.pin) {\n      return null;\n    }\n    const sb: string[] = [];\n    // pin value\n    sb.push(this.lookupData.pin.value!);\n    // item title\n    sb.push(' | ');\n    sb.push(this.lookupData.item!.title || this.lookupData.item!.id);\n    // part type and role\n    const e = this.modelEntries?.find(\n      (e) => e.id === this.lookupData!.pin.partTypeId\n    );\n    sb.push(' (');\n    sb.push(e?.value || this.lookupData.pin.partTypeId);\n    if (this.lookupData.pin.roleId) {\n      sb.push(`, ${this.lookupData.pin.roleId}`);\n    }\n    sb.push(')');\n    return sb.join('');\n  }\n\n  private getTarget(): PinTarget {\n    if (this.external.value) {\n      return {\n        gid: this.gid.value || '',\n        label: this.label.value || '',\n      };\n    } else {\n      const pin = this.lookupData?.pin;\n      return {\n        gid: this.gid.value || '',\n        label: this.label.value || '',\n        itemId: pin?.itemId || '',\n        partId: pin?.partId || '',\n        partTypeId: pin?.partTypeId || '',\n        roleId: pin?.roleId || '',\n        name: pin?.name || '',\n        value: pin?.value || '',\n      };\n    }\n  }\n\n  private updateTarget(): void {\n    this._noFormUpdate = true;\n\n    if (this.canBuildGid()) {\n      this.gid.setValue(this.buildGid());\n      this.gid.markAsDirty();\n    }\n    if (this.canBuildLabel()) {\n      this.label.setValue(this.buildLabel());\n      this.label.markAsDirty();\n    }\n    this.target = this.getTarget();\n\n    this._noFormUpdate = false;\n  }\n\n  private updateForm(target?: PinTarget): void {\n    if (this._noFormUpdate) {\n      return;\n    }\n    // build pin info from target\n    if (!target) {\n      this.lookupData = undefined;\n      this.form.reset();\n      return;\n    }\n    this.external.setValue(!target.name);\n    this.lookupData = {\n      pin: {\n        itemId: target.itemId || '',\n        partId: target.partId || '',\n        partTypeId: target.partTypeId || '',\n        roleId: target.roleId || '',\n        name: target.name || '',\n        value: target.value || '',\n      },\n    };\n    // get item\n    if (target.itemId) {\n      this._itemService.getItem(target.itemId, true).subscribe({\n        next: (item) => {\n          this.item.setValue(item);\n          this.form.markAsPristine();\n        },\n        error: (error) => {\n          if (error) {\n            console.error(JSON.stringify(error));\n          }\n        },\n      });\n    }\n  }\n\n  /**\n   * Called when the item lookup changes (item is looked up\n   * by its title).\n   *\n   * @param item The item got from lookup.\n   */\n  public onItemLookupChange(item: Item): void {\n    // load item's parts\n    this._itemService.getItem(item.id, true).subscribe({\n      next: (i) => {\n        // setting the item will trigger its parts update\n        this.item.setValue(i);\n        this.updateTarget();\n      },\n      error: (error) => {\n        if (error) {\n          console.error(JSON.stringify(error));\n        }\n        this.itemPart.setValue(null);\n        this.itemParts = [];\n        this.updateTarget();\n      },\n    });\n  }\n\n  private loadItemInfo(pin: DataPinInfo): void {\n    forkJoin({\n      item: this._itemService.getItem(pin.itemId, false),\n      part: this._itemService.getPartFromTypeAndRole(\n        pin.itemId,\n        METADATA_PART_ID\n      ),\n    })\n      .pipe(take(1))\n      .subscribe({\n        next: (result) => {\n          this.lookupData = {\n            pin: pin,\n            item: result.item,\n            metaPart: result.part as MetadataPart,\n          };\n          this.updateTarget();\n        },\n        error: (error) => {\n          this.lookupData = undefined;\n          console.error(\n            error ? JSON.stringify(error) : 'Error loading item/metadata'\n          );\n        },\n      });\n  }\n\n  /**\n   * Called when the pin lookup change. A pin is looked up by its\n   * name and value (=the filter's text), and optionally by:\n   * - its index lookup definition (selected by partTypeKey).\n   * - its item (defined by item, in filter).\n   * - its part (defined by itemPart, in filter).\n   *\n   * @param info The pin info from pin lookup.\n   */\n  public onPinLookupChange(info: DataPinInfo): void {\n    this.loadItemInfo(info);\n  }\n\n  public onCopied(): void {\n    this._snackbar.open('Copied to clipboard', 'OK', {\n      duration: 1500,\n    });\n  }\n\n  public close(): void {\n    this.editorClose.emit();\n  }\n\n  public save(): void {\n    if (this.form.invalid) {\n      return;\n    }\n    this._target = this.getTarget();\n    this.targetChange.emit(this._target);\n  }\n}\n","<form [formGroup]=\"form\" (submit)=\"save()\">\r\n  <!-- external -->\r\n  <mat-checkbox [formControl]=\"external\">external</mat-checkbox>\r\n\r\n  <div class=\"form-row\">\r\n    <!-- label -->\r\n    <div>\r\n      <mat-form-field *ngIf=\"external.value || canEditTarget\">\r\n        <mat-label>label</mat-label>\r\n        <input matInput [formControl]=\"label\" />\r\n        <mat-error\r\n          *ngIf=\"$any(label).errors?.required && (label.dirty || label.touched)\"\r\n          >label required</mat-error\r\n        >\r\n        <mat-error\r\n          *ngIf=\"\r\n            $any(label).errors?.maxLength && (label.dirty || label.touched)\r\n          \"\r\n          >label too long</mat-error\r\n        >\r\n      </mat-form-field>\r\n      <div\r\n        *ngIf=\"!external.value && !canEditTarget && label.value\"\r\n        class=\"info\"\r\n      >\r\n        <span class=\"label\">label</span>{{ label.value }}\r\n      </div>\r\n    </div>\r\n\r\n    <!-- gid -->\r\n    <div>\r\n      <mat-form-field *ngIf=\"external.value || canEditTarget\">\r\n        <mat-label>GID</mat-label>\r\n        <input matInput [formControl]=\"gid\" />\r\n        <mat-error\r\n          *ngIf=\"$any(gid).errors?.required && (gid.dirty || gid.touched)\"\r\n          >GID required</mat-error\r\n        >\r\n        <mat-error\r\n          *ngIf=\"$any(gid).errors?.maxLength && (gid.dirty || gid.touched)\"\r\n          >GID too long</mat-error\r\n        >\r\n      </mat-form-field>\r\n      <div *ngIf=\"!external.value && !canEditTarget && gid.value\" class=\"info\">\r\n        <span class=\"label\">GID</span> <span class=\"gid\">{{ gid.value }}</span>\r\n      </div>\r\n    </div>\r\n  </div>\r\n\r\n  <!-- BY ITEM -->\r\n  <div *ngIf=\"!external.value\">\r\n    <fieldset *ngIf=\"!byTypeMode.value\" class=\"form-row\">\r\n      <legend>pin filters</legend>\r\n      <!-- item filter -->\r\n      <cadmus-ref-lookup\r\n        [service]=\"itemLookupService\"\r\n        label=\"item\"\r\n        (itemChange)=\"onItemLookupChange($event)\"\r\n      ></cadmus-ref-lookup>\r\n\r\n      <!-- part filter -->\r\n      <mat-form-field *ngIf=\"itemParts.length\">\r\n        <mat-label>part</mat-label>\r\n        <mat-select [formControl]=\"itemPart\">\r\n          <mat-option [value]=\"null\">(any)</mat-option>\r\n          <mat-option *ngFor=\"let p of itemParts\" [value]=\"p\">{{\r\n            p.typeId | flatLookup : modelEntries : \"id\" : \"value\"\r\n          }}</mat-option>\r\n        </mat-select>\r\n      </mat-form-field>\r\n    </fieldset>\r\n\r\n    <!-- BY TYPE -->\r\n    <div *ngIf=\"byTypeMode.value\">\r\n      <!-- par type filter -->\r\n      <mat-form-field *ngIf=\"partTypeKeys?.length\">\r\n        <mat-label>part type</mat-label>\r\n        <mat-select [formControl]=\"partTypeKey\">\r\n          <mat-option *ngFor=\"let k of partTypeKeys\" [value]=\"k\">{{\r\n            k\r\n          }}</mat-option>\r\n        </mat-select>\r\n      </mat-form-field>\r\n    </div>\r\n\r\n    <!-- PIN -->\r\n    <div class=\"form-row\">\r\n      <!-- pin lookup -->\r\n      <cadmus-ref-lookup\r\n        [service]=\"pinLookupService\"\r\n        [baseFilter]=\"filter\"\r\n        [options]=\"pinFilterOptions\"\r\n        label=\"pin\"\r\n        (itemChange)=\"onPinLookupChange($event)\"\r\n      ></cadmus-ref-lookup>\r\n      <!-- mode switcher -->\r\n      <div>\r\n        <mat-checkbox [formControl]=\"byTypeMode\" *ngIf=\"canSwitchMode\">\r\n          by type</mat-checkbox\r\n        >\r\n      </div>\r\n    </div>\r\n\r\n    <!-- data -->\r\n    <mat-expansion-panel id=\"data\" *ngIf=\"lookupData?.pin?.name\">\r\n      <mat-expansion-panel-header>pin data</mat-expansion-panel-header>\r\n      <!-- table -->\r\n      <table>\r\n        <thead>\r\n          <th></th>\r\n          <th>source</th>\r\n          <th>value</th>\r\n        </thead>\r\n        <tbody>\r\n          <!-- pin -->\r\n          <tr *ngIf=\"lookupData?.pin?.value\">\r\n            <td>\r\n              <button\r\n                type=\"button\"\r\n                mat-icon-button\r\n                color=\"primary\"\r\n                [cdkCopyToClipboard]=\"lookupData!.pin!.value\"\r\n                (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n              >\r\n                <mat-icon>content_copy</mat-icon>\r\n              </button>\r\n            </td>\r\n            <td>pin</td>\r\n            <td>{{ lookupData!.pin.value }}</td>\r\n          </tr>\r\n          <!-- item ID -->\r\n          <tr *ngIf=\"lookupData?.pin?.itemId\">\r\n            <td>\r\n              <button\r\n                type=\"button\"\r\n                mat-icon-button\r\n                color=\"primary\"\r\n                [cdkCopyToClipboard]=\"lookupData!.pin!.itemId\"\r\n                (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n              >\r\n                <mat-icon>content_copy</mat-icon>\r\n              </button>\r\n            </td>\r\n            <td>item ID</td>\r\n            <td>{{ lookupData!.pin.itemId }}</td>\r\n          </tr>\r\n          <!-- item title -->\r\n          <tr *ngIf=\"lookupData?.item?.title\">\r\n            <td>\r\n              <button\r\n                type=\"button\"\r\n                mat-icon-button\r\n                color=\"primary\"\r\n                [cdkCopyToClipboard]=\"lookupData!.item!.title\"\r\n                (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n              >\r\n                <mat-icon>content_copy</mat-icon>\r\n              </button>\r\n            </td>\r\n            <td>item title</td>\r\n            <td>{{ lookupData!.item!.title }}</td>\r\n          </tr>\r\n          <!-- part ID -->\r\n          <tr *ngIf=\"lookupData?.pin?.partId\">\r\n            <td>\r\n              <button\r\n                type=\"button\"\r\n                mat-icon-button\r\n                color=\"primary\"\r\n                [cdkCopyToClipboard]=\"lookupData!.pin!.partId\"\r\n                (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n              >\r\n                <mat-icon>content_copy</mat-icon>\r\n              </button>\r\n            </td>\r\n            <td>part ID</td>\r\n            <td>{{ lookupData!.pin.partId }}</td>\r\n          </tr>\r\n          <!-- part type ID -->\r\n          <tr *ngIf=\"lookupData?.pin?.partTypeId\">\r\n            <td>\r\n              <button\r\n                type=\"button\"\r\n                mat-icon-button\r\n                color=\"primary\"\r\n                [cdkCopyToClipboard]=\"lookupData!.pin!.partTypeId\"\r\n                (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n              >\r\n                <mat-icon>content_copy</mat-icon>\r\n              </button>\r\n            </td>\r\n            <td>part type ID</td>\r\n            <td>{{ lookupData!.pin!.partTypeId }}</td>\r\n          </tr>\r\n          <!-- part role ID -->\r\n          <tr *ngIf=\"lookupData?.pin?.roleId\">\r\n            <td>\r\n              <button\r\n                type=\"button\"\r\n                mat-icon-button\r\n                color=\"primary\"\r\n                [cdkCopyToClipboard]=\"lookupData!.pin!.roleId!\"\r\n                (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n              >\r\n                <mat-icon>content_copy</mat-icon>\r\n              </button>\r\n            </td>\r\n            <td>part role ID</td>\r\n            <td>{{ lookupData!.pin!.roleId }}</td>\r\n          </tr>\r\n\r\n          <!-- part's metadata -->\r\n          <tr\r\n            *ngFor=\"\r\n              let m of lookupData?.metaPart?.metadata || [];\r\n              let i = index\r\n            \"\r\n          >\r\n            <td>\r\n              <button\r\n                type=\"button\"\r\n                mat-icon-button\r\n                color=\"primary\"\r\n                [cdkCopyToClipboard]=\"m.value\"\r\n                (cdkCopyToClipboardCopied)=\"onCopied()\"\r\n              >\r\n                <mat-icon>content_copy</mat-icon>\r\n              </button>\r\n            </td>\r\n            <td class=\"metadata\">{{ m.name }}</td>\r\n            <td class=\"metadata\">{{ m.value }}</td>\r\n          </tr>\r\n        </tbody>\r\n      </table>\r\n    </mat-expansion-panel>\r\n  </div>\r\n\r\n  <!-- buttons -->\r\n  <div>\r\n    <button mat-flat-button type=\"button\" (click)=\"close()\">\r\n      <mat-icon color=\"warn\">close</mat-icon>\r\n    </button>\r\n    <button mat-flat-button type=\"submit\" [disabled]=\"form.invalid\">\r\n      <mat-icon color=\"primary\">check_circle</mat-icon> target\r\n    </button>\r\n  </div>\r\n</form>\r\n"]}