@cqa-lib/cqa-ui 1.1.525 → 1.1.526
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/esm2020/lib/assets/images/image-assets.constants.mjs +3 -1
- package/esm2020/lib/compare-runs/compare-runs.component.mjs +1 -1
- package/esm2020/lib/execution-screen/db-query-execution-item/db-query-execution-item.component.mjs +1 -1
- package/esm2020/lib/execution-screen/db-verification-step/db-verification-step.component.mjs +1 -1
- package/esm2020/lib/iterations-loop/iterations-loop.component.mjs +1 -1
- package/esm2020/lib/segment-control/segment-control.component.mjs +6 -3
- package/esm2020/lib/simulator/simulator.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-document-generation-template-step/step-builder-document-generation-template-step.component.mjs +1 -1
- package/esm2020/lib/table/dynamic-table/dynamic-table.component.mjs +148 -4
- package/esm2020/lib/templates/modular-table-template/dialogs/delete-folder-dialog.component.mjs +181 -0
- package/esm2020/lib/templates/modular-table-template/dialogs/move-to-folder-dialog.component.mjs +264 -0
- package/esm2020/lib/templates/modular-table-template/dialogs/new-folder-dialog.component.mjs +352 -0
- package/esm2020/lib/templates/modular-table-template/directives/folder-drag.directive.mjs +45 -0
- package/esm2020/lib/templates/modular-table-template/directives/folder-drop.directive.mjs +95 -0
- package/esm2020/lib/templates/modular-table-template/directives/row-drag.directive.mjs +44 -0
- package/esm2020/lib/templates/modular-table-template/folder-sidebar/folder-sidebar.component.mjs +479 -0
- package/esm2020/lib/templates/modular-table-template/modular-table-template.component.mjs +1475 -0
- package/esm2020/lib/templates/modular-table-template/modular-table-template.models.mjs +79 -0
- package/esm2020/lib/templates/table-template.component.mjs +88 -12
- package/esm2020/lib/test-case-details/api-edit-step/api-edit-step.component.mjs +1 -1
- package/esm2020/lib/ui-kit.module.mjs +41 -1
- package/esm2020/public-api.mjs +10 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +3408 -178
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +3388 -176
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/assets/images/image-assets.constants.d.ts +1 -0
- package/lib/segment-control/segment-control.component.d.ts +2 -1
- package/lib/table/dynamic-table/dynamic-table.component.d.ts +43 -1
- package/lib/templates/modular-table-template/dialogs/delete-folder-dialog.component.d.ts +34 -0
- package/lib/templates/modular-table-template/dialogs/move-to-folder-dialog.component.d.ts +57 -0
- package/lib/templates/modular-table-template/dialogs/new-folder-dialog.component.d.ts +79 -0
- package/lib/templates/modular-table-template/directives/folder-drag.directive.d.ts +10 -0
- package/lib/templates/modular-table-template/directives/folder-drop.directive.d.ts +22 -0
- package/lib/templates/modular-table-template/directives/row-drag.directive.d.ts +10 -0
- package/lib/templates/modular-table-template/folder-sidebar/folder-sidebar.component.d.ts +149 -0
- package/lib/templates/modular-table-template/modular-table-template.component.d.ts +453 -0
- package/lib/templates/modular-table-template/modular-table-template.models.d.ts +150 -0
- package/lib/templates/table-template.component.d.ts +40 -2
- package/lib/ui-kit.module.d.ts +153 -145
- package/package.json +1 -1
- package/public-api.d.ts +9 -0
- package/src/lib/assets/images/EmptyFolderState.png +0 -0
- package/src/lib/assets/images/image-assets.constants.ts +3 -0
- package/styles.css +1 -1
|
@@ -0,0 +1,352 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, } from '@angular/core';
|
|
2
|
+
import { FormControl, FormGroup } from '@angular/forms';
|
|
3
|
+
import { DEFAULT_MODULAR_LABELS } from '../modular-table-template.models';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "../../../custom-input/custom-input.component";
|
|
6
|
+
import * as i2 from "../../../dynamic-select/dynamic-select-field.component";
|
|
7
|
+
import * as i3 from "@angular/common";
|
|
8
|
+
export const DEFAULT_FOLDER_COLOR = '#99999E';
|
|
9
|
+
const PRESET_FOLDER_COLORS = [
|
|
10
|
+
DEFAULT_FOLDER_COLOR,
|
|
11
|
+
'#4F46E5',
|
|
12
|
+
'#06B6D4',
|
|
13
|
+
'#10B981',
|
|
14
|
+
'#F59E0B',
|
|
15
|
+
'#EF4444',
|
|
16
|
+
'#EC4899',
|
|
17
|
+
'#8B5CF6',
|
|
18
|
+
];
|
|
19
|
+
/**
|
|
20
|
+
* Body of the "New Folder" dialog.
|
|
21
|
+
*
|
|
22
|
+
* Uses reusable library components (`cqa-custom-input` for the name field,
|
|
23
|
+
* `cqa-dynamic-select` for the parent dropdown) so hosts can drop this dialog
|
|
24
|
+
* into their own modal shell without worrying about styling drift.
|
|
25
|
+
*
|
|
26
|
+
* Two integration paths:
|
|
27
|
+
*
|
|
28
|
+
* 1. **From within this library** via `DialogService.open(...)`. The outer
|
|
29
|
+
* `cqa-dialog` supplies title/description/Cancel/Create-folder buttons; the
|
|
30
|
+
* Create button reads `name` + `parentId` via `getComponentInstance()`.
|
|
31
|
+
*
|
|
32
|
+
* 2. **Host-driven** — host renders `<cqa-new-folder-dialog>` inside its own
|
|
33
|
+
* modal and listens to `(submitted)` / `(cancelled)` / `(nameChange)` /
|
|
34
|
+
* `(parentIdChange)`.
|
|
35
|
+
*/
|
|
36
|
+
export class NewFolderDialogComponent {
|
|
37
|
+
constructor(cdr) {
|
|
38
|
+
this.cdr = cdr;
|
|
39
|
+
// NOTE: These inputs are declared as setters (instead of plain @Input fields)
|
|
40
|
+
// because `DialogService.open(...)` assigns the `inputs` map via direct
|
|
41
|
+
// property assignment *after* `attachComponent` has already run the
|
|
42
|
+
// component's `ngOnInit`. Direct assignment bypasses `ngOnChanges`, so without
|
|
43
|
+
// setters the `parentSelectConfig` would be built from a default empty folders
|
|
44
|
+
// array and the parent-folder dropdown would render empty / appear broken.
|
|
45
|
+
this._folders = [];
|
|
46
|
+
this._labels = { ...DEFAULT_MODULAR_LABELS };
|
|
47
|
+
this.name = '';
|
|
48
|
+
this._parentId = null;
|
|
49
|
+
this.color = DEFAULT_FOLDER_COLOR;
|
|
50
|
+
this.presetColors = PRESET_FOLDER_COLORS;
|
|
51
|
+
this.rainbowBorder = 'conic-gradient(from 0deg, #ef4444, #f59e0b, #eab308, #10b981, #06b6d4, #4f46e5, #8b5cf6, #ec4899, #ef4444)';
|
|
52
|
+
this.nameChange = new EventEmitter();
|
|
53
|
+
this.parentIdChange = new EventEmitter();
|
|
54
|
+
this.colorChange = new EventEmitter();
|
|
55
|
+
this.submitted = new EventEmitter();
|
|
56
|
+
this.cancelled = new EventEmitter();
|
|
57
|
+
/** Reactive form used by cqa-dynamic-select. Untyped to match the project's Angular 13.4 forms API. */
|
|
58
|
+
this.parentForm = new FormGroup({ parentId: new FormControl(null) });
|
|
59
|
+
this.parentOptions = [];
|
|
60
|
+
this.rebuildSelectConfig(); // ensure config exists even before first input set
|
|
61
|
+
}
|
|
62
|
+
set folders(value) {
|
|
63
|
+
this._folders = value || [];
|
|
64
|
+
this.rebuildOptions();
|
|
65
|
+
this.rebuildSelectConfig();
|
|
66
|
+
}
|
|
67
|
+
get folders() { return this._folders; }
|
|
68
|
+
set labels(value) {
|
|
69
|
+
this._labels = value || { ...DEFAULT_MODULAR_LABELS };
|
|
70
|
+
this.rebuildSelectConfig();
|
|
71
|
+
}
|
|
72
|
+
get labels() { return this._labels; }
|
|
73
|
+
set parentId(value) {
|
|
74
|
+
this._parentId = value ?? null;
|
|
75
|
+
this.syncParentControl();
|
|
76
|
+
}
|
|
77
|
+
get parentId() { return this._parentId; }
|
|
78
|
+
ngOnInit() {
|
|
79
|
+
// Setters above already rebuilt config once folders/labels were assigned. This
|
|
80
|
+
// just ensures the form control matches the current parentId value in case
|
|
81
|
+
// ordering of input assignment differs.
|
|
82
|
+
this.syncParentControl();
|
|
83
|
+
}
|
|
84
|
+
rebuildSelectConfig() {
|
|
85
|
+
this.parentSelectConfig = {
|
|
86
|
+
key: 'parentId',
|
|
87
|
+
label: this.labels.newFolderDialogParentLabel,
|
|
88
|
+
placeholder: this.labels.newFolderDialogParentNone,
|
|
89
|
+
searchable: false,
|
|
90
|
+
multiple: false,
|
|
91
|
+
closeOnSelect: true,
|
|
92
|
+
options: [
|
|
93
|
+
// `getOptionValue` in cqa-dynamic-select returns `id ?? value`, so using
|
|
94
|
+
// `value: null` (with no id) yields a stable `null` resolved value that
|
|
95
|
+
// matches the FormControl's null state.
|
|
96
|
+
{ value: null, name: this.labels.newFolderDialogParentNone },
|
|
97
|
+
...this.parentOptions.map(o => ({ id: o.id, name: o.label })),
|
|
98
|
+
],
|
|
99
|
+
};
|
|
100
|
+
}
|
|
101
|
+
rebuildOptions() {
|
|
102
|
+
const opts = [];
|
|
103
|
+
const walk = (nodes, trail) => {
|
|
104
|
+
for (const n of nodes || []) {
|
|
105
|
+
const label = [...trail, n.name].join(' › ');
|
|
106
|
+
opts.push({ id: n.id, label });
|
|
107
|
+
if (n.children?.length)
|
|
108
|
+
walk(n.children, [...trail, n.name]);
|
|
109
|
+
}
|
|
110
|
+
};
|
|
111
|
+
walk(this.folders, []);
|
|
112
|
+
this.parentOptions = opts;
|
|
113
|
+
this.cdr.markForCheck();
|
|
114
|
+
}
|
|
115
|
+
syncParentControl() {
|
|
116
|
+
// Keep the reactive form in sync with the `parentId` input without bouncing events.
|
|
117
|
+
this.parentForm.get('parentId')?.setValue(this.parentId ?? null, { emitEvent: false });
|
|
118
|
+
}
|
|
119
|
+
onNameChange(value) {
|
|
120
|
+
this.name = value;
|
|
121
|
+
this.nameChange.emit(value);
|
|
122
|
+
}
|
|
123
|
+
onParentSelectionChange(event) {
|
|
124
|
+
// cqa-dynamic-select resolves options to `id ?? value`; option ids in the
|
|
125
|
+
// parent tree are numeric so we coerce defensively.
|
|
126
|
+
const raw = event?.value;
|
|
127
|
+
let next = null;
|
|
128
|
+
if (raw != null) {
|
|
129
|
+
const coerced = typeof raw === 'number' ? raw : Number(raw);
|
|
130
|
+
if (Number.isFinite(coerced))
|
|
131
|
+
next = coerced;
|
|
132
|
+
}
|
|
133
|
+
this.parentId = next;
|
|
134
|
+
this.parentIdChange.emit(next);
|
|
135
|
+
}
|
|
136
|
+
isColor(c) {
|
|
137
|
+
return (this.color || '').toLowerCase() === (c || '').toLowerCase();
|
|
138
|
+
}
|
|
139
|
+
get isCustomColor() {
|
|
140
|
+
const c = (this.color || '').toLowerCase();
|
|
141
|
+
return !!c && !this.presetColors.some(p => p.toLowerCase() === c);
|
|
142
|
+
}
|
|
143
|
+
onColorChange(value) {
|
|
144
|
+
const next = value || DEFAULT_FOLDER_COLOR;
|
|
145
|
+
if (this.color === next)
|
|
146
|
+
return;
|
|
147
|
+
this.color = next;
|
|
148
|
+
this.colorChange.emit(next);
|
|
149
|
+
this.cdr.markForCheck();
|
|
150
|
+
}
|
|
151
|
+
onColorInputEvent(event) {
|
|
152
|
+
const target = event.target;
|
|
153
|
+
if (target)
|
|
154
|
+
this.onColorChange(target.value);
|
|
155
|
+
}
|
|
156
|
+
/** Returns true when the current state is a valid submission (non-empty, trimmed name). */
|
|
157
|
+
get isValid() {
|
|
158
|
+
return !!(this.name || '').trim();
|
|
159
|
+
}
|
|
160
|
+
submit() {
|
|
161
|
+
if (!this.isValid)
|
|
162
|
+
return;
|
|
163
|
+
this.submitted.emit({
|
|
164
|
+
name: (this.name || '').trim(),
|
|
165
|
+
parentId: this.parentId,
|
|
166
|
+
color: this.color || DEFAULT_FOLDER_COLOR,
|
|
167
|
+
});
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
NewFolderDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NewFolderDialogComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
171
|
+
NewFolderDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: NewFolderDialogComponent, selector: "cqa-new-folder-dialog", inputs: { folders: "folders", labels: "labels", name: "name", parentId: "parentId", color: "color" }, outputs: { nameChange: "nameChange", parentIdChange: "parentIdChange", colorChange: "colorChange", submitted: "submitted", cancelled: "cancelled" }, host: { classAttribute: "cqa-ui-root" }, ngImport: i0, template: `
|
|
172
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-5 cqa-w-full">
|
|
173
|
+
<!-- Folder name -->
|
|
174
|
+
<cqa-custom-input
|
|
175
|
+
[label]="labels.newFolderDialogNameLabel"
|
|
176
|
+
[required]="true"
|
|
177
|
+
[placeholder]="labels.newFolderDialogNamePlaceholder"
|
|
178
|
+
[value]="name"
|
|
179
|
+
[fullWidth]="true"
|
|
180
|
+
(valueChange)="onNameChange($event)"
|
|
181
|
+
(enterPressed)="submit()"
|
|
182
|
+
></cqa-custom-input>
|
|
183
|
+
|
|
184
|
+
<!-- Parent folder -->
|
|
185
|
+
<cqa-dynamic-select
|
|
186
|
+
class="cqa-block cqa-w-full"
|
|
187
|
+
[form]="parentForm"
|
|
188
|
+
[config]="parentSelectConfig"
|
|
189
|
+
(selectionChange)="onParentSelectionChange($event)"
|
|
190
|
+
></cqa-dynamic-select>
|
|
191
|
+
|
|
192
|
+
<!-- Folder color (optional) -->
|
|
193
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-2">
|
|
194
|
+
<label class="cqa-text-sm cqa-font-medium cqa-text-neutral-700">
|
|
195
|
+
{{ labels.newFolderDialogColorLabel }}
|
|
196
|
+
</label>
|
|
197
|
+
<div class="cqa-flex cqa-items-center cqa-gap-2.5 cqa-flex-wrap">
|
|
198
|
+
<button
|
|
199
|
+
*ngFor="let c of presetColors"
|
|
200
|
+
type="button"
|
|
201
|
+
class="cqa-relative cqa-w-7 cqa-h-7 cqa-rounded-full cqa-cursor-pointer cqa-border-0 cqa-p-0 cqa-transition-transform"
|
|
202
|
+
[style.backgroundColor]="c"
|
|
203
|
+
[style.boxShadow]="isColor(c) ? ('0 0 0 2px #ffffff, 0 0 0 4px ' + c) : '0 0 0 1px rgba(15, 23, 42, 0.08)'"
|
|
204
|
+
(click)="onColorChange(c)"
|
|
205
|
+
[attr.aria-label]="'Set folder color ' + c"
|
|
206
|
+
[attr.aria-pressed]="isColor(c)"
|
|
207
|
+
[title]="c"
|
|
208
|
+
>
|
|
209
|
+
<svg
|
|
210
|
+
*ngIf="isColor(c)"
|
|
211
|
+
class="cqa-absolute cqa-inset-0 cqa-m-auto"
|
|
212
|
+
width="14" height="14" viewBox="0 0 24 24" fill="none" aria-hidden="true"
|
|
213
|
+
>
|
|
214
|
+
<path d="M5 12l4.5 4.5L19 7" stroke="#ffffff" stroke-width="3" stroke-linecap="round" stroke-linejoin="round"/>
|
|
215
|
+
</svg>
|
|
216
|
+
</button>
|
|
217
|
+
|
|
218
|
+
<label
|
|
219
|
+
class="cqa-relative cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-rounded-full cqa-cursor-pointer cqa-transition-transform"
|
|
220
|
+
[style.background]="rainbowBorder"
|
|
221
|
+
[title]="isCustomColor ? ('Custom color (' + color + ')') : 'Pick a custom color'"
|
|
222
|
+
>
|
|
223
|
+
<span
|
|
224
|
+
class="cqa-absolute cqa-flex cqa-items-center cqa-justify-center cqa-rounded-full"
|
|
225
|
+
[style.top.px]="2"
|
|
226
|
+
[style.right.px]="2"
|
|
227
|
+
[style.bottom.px]="2"
|
|
228
|
+
[style.left.px]="2"
|
|
229
|
+
[style.backgroundColor]="isCustomColor ? color : '#ffffff'"
|
|
230
|
+
>
|
|
231
|
+
<svg *ngIf="!isCustomColor" width="14" height="14" viewBox="0 0 24 24" fill="none" aria-hidden="true">
|
|
232
|
+
<path d="M12 5v14M5 12h14" stroke="#525258" stroke-width="2" stroke-linecap="round"/>
|
|
233
|
+
</svg>
|
|
234
|
+
<svg *ngIf="isCustomColor" width="14" height="14" viewBox="0 0 24 24" fill="none" aria-hidden="true">
|
|
235
|
+
<path d="M5 12l4.5 4.5L19 7" stroke="#ffffff" stroke-width="3" stroke-linecap="round" stroke-linejoin="round"/>
|
|
236
|
+
</svg>
|
|
237
|
+
</span>
|
|
238
|
+
<input
|
|
239
|
+
type="color"
|
|
240
|
+
[value]="color"
|
|
241
|
+
(input)="onColorInputEvent($event)"
|
|
242
|
+
class="cqa-absolute cqa-inset-0 cqa-w-full cqa-h-full cqa-opacity-0 cqa-cursor-pointer cqa-border-0 cqa-p-0"
|
|
243
|
+
aria-label="Pick custom folder color"
|
|
244
|
+
/>
|
|
245
|
+
</label>
|
|
246
|
+
</div>
|
|
247
|
+
</div>
|
|
248
|
+
</div>
|
|
249
|
+
`, isInline: true, components: [{ type: i1.CustomInputComponent, selector: "cqa-custom-input", inputs: ["inputId", "label", "type", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "inputInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused", "enterPressed"] }, { type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }], directives: [{ type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
250
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NewFolderDialogComponent, decorators: [{
|
|
251
|
+
type: Component,
|
|
252
|
+
args: [{ selector: 'cqa-new-folder-dialog', template: `
|
|
253
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-5 cqa-w-full">
|
|
254
|
+
<!-- Folder name -->
|
|
255
|
+
<cqa-custom-input
|
|
256
|
+
[label]="labels.newFolderDialogNameLabel"
|
|
257
|
+
[required]="true"
|
|
258
|
+
[placeholder]="labels.newFolderDialogNamePlaceholder"
|
|
259
|
+
[value]="name"
|
|
260
|
+
[fullWidth]="true"
|
|
261
|
+
(valueChange)="onNameChange($event)"
|
|
262
|
+
(enterPressed)="submit()"
|
|
263
|
+
></cqa-custom-input>
|
|
264
|
+
|
|
265
|
+
<!-- Parent folder -->
|
|
266
|
+
<cqa-dynamic-select
|
|
267
|
+
class="cqa-block cqa-w-full"
|
|
268
|
+
[form]="parentForm"
|
|
269
|
+
[config]="parentSelectConfig"
|
|
270
|
+
(selectionChange)="onParentSelectionChange($event)"
|
|
271
|
+
></cqa-dynamic-select>
|
|
272
|
+
|
|
273
|
+
<!-- Folder color (optional) -->
|
|
274
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-2">
|
|
275
|
+
<label class="cqa-text-sm cqa-font-medium cqa-text-neutral-700">
|
|
276
|
+
{{ labels.newFolderDialogColorLabel }}
|
|
277
|
+
</label>
|
|
278
|
+
<div class="cqa-flex cqa-items-center cqa-gap-2.5 cqa-flex-wrap">
|
|
279
|
+
<button
|
|
280
|
+
*ngFor="let c of presetColors"
|
|
281
|
+
type="button"
|
|
282
|
+
class="cqa-relative cqa-w-7 cqa-h-7 cqa-rounded-full cqa-cursor-pointer cqa-border-0 cqa-p-0 cqa-transition-transform"
|
|
283
|
+
[style.backgroundColor]="c"
|
|
284
|
+
[style.boxShadow]="isColor(c) ? ('0 0 0 2px #ffffff, 0 0 0 4px ' + c) : '0 0 0 1px rgba(15, 23, 42, 0.08)'"
|
|
285
|
+
(click)="onColorChange(c)"
|
|
286
|
+
[attr.aria-label]="'Set folder color ' + c"
|
|
287
|
+
[attr.aria-pressed]="isColor(c)"
|
|
288
|
+
[title]="c"
|
|
289
|
+
>
|
|
290
|
+
<svg
|
|
291
|
+
*ngIf="isColor(c)"
|
|
292
|
+
class="cqa-absolute cqa-inset-0 cqa-m-auto"
|
|
293
|
+
width="14" height="14" viewBox="0 0 24 24" fill="none" aria-hidden="true"
|
|
294
|
+
>
|
|
295
|
+
<path d="M5 12l4.5 4.5L19 7" stroke="#ffffff" stroke-width="3" stroke-linecap="round" stroke-linejoin="round"/>
|
|
296
|
+
</svg>
|
|
297
|
+
</button>
|
|
298
|
+
|
|
299
|
+
<label
|
|
300
|
+
class="cqa-relative cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-rounded-full cqa-cursor-pointer cqa-transition-transform"
|
|
301
|
+
[style.background]="rainbowBorder"
|
|
302
|
+
[title]="isCustomColor ? ('Custom color (' + color + ')') : 'Pick a custom color'"
|
|
303
|
+
>
|
|
304
|
+
<span
|
|
305
|
+
class="cqa-absolute cqa-flex cqa-items-center cqa-justify-center cqa-rounded-full"
|
|
306
|
+
[style.top.px]="2"
|
|
307
|
+
[style.right.px]="2"
|
|
308
|
+
[style.bottom.px]="2"
|
|
309
|
+
[style.left.px]="2"
|
|
310
|
+
[style.backgroundColor]="isCustomColor ? color : '#ffffff'"
|
|
311
|
+
>
|
|
312
|
+
<svg *ngIf="!isCustomColor" width="14" height="14" viewBox="0 0 24 24" fill="none" aria-hidden="true">
|
|
313
|
+
<path d="M12 5v14M5 12h14" stroke="#525258" stroke-width="2" stroke-linecap="round"/>
|
|
314
|
+
</svg>
|
|
315
|
+
<svg *ngIf="isCustomColor" width="14" height="14" viewBox="0 0 24 24" fill="none" aria-hidden="true">
|
|
316
|
+
<path d="M5 12l4.5 4.5L19 7" stroke="#ffffff" stroke-width="3" stroke-linecap="round" stroke-linejoin="round"/>
|
|
317
|
+
</svg>
|
|
318
|
+
</span>
|
|
319
|
+
<input
|
|
320
|
+
type="color"
|
|
321
|
+
[value]="color"
|
|
322
|
+
(input)="onColorInputEvent($event)"
|
|
323
|
+
class="cqa-absolute cqa-inset-0 cqa-w-full cqa-h-full cqa-opacity-0 cqa-cursor-pointer cqa-border-0 cqa-p-0"
|
|
324
|
+
aria-label="Pick custom folder color"
|
|
325
|
+
/>
|
|
326
|
+
</label>
|
|
327
|
+
</div>
|
|
328
|
+
</div>
|
|
329
|
+
</div>
|
|
330
|
+
`, host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.OnPush, styles: [] }]
|
|
331
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { folders: [{
|
|
332
|
+
type: Input
|
|
333
|
+
}], labels: [{
|
|
334
|
+
type: Input
|
|
335
|
+
}], name: [{
|
|
336
|
+
type: Input
|
|
337
|
+
}], parentId: [{
|
|
338
|
+
type: Input
|
|
339
|
+
}], color: [{
|
|
340
|
+
type: Input
|
|
341
|
+
}], nameChange: [{
|
|
342
|
+
type: Output
|
|
343
|
+
}], parentIdChange: [{
|
|
344
|
+
type: Output
|
|
345
|
+
}], colorChange: [{
|
|
346
|
+
type: Output
|
|
347
|
+
}], submitted: [{
|
|
348
|
+
type: Output
|
|
349
|
+
}], cancelled: [{
|
|
350
|
+
type: Output
|
|
351
|
+
}] } });
|
|
352
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"new-folder-dialog.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/templates/modular-table-template/dialogs/new-folder-dialog.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  OnInit,\n  Output,\n} from '@angular/core';\nimport { FormControl, FormGroup } from '@angular/forms';\nimport { DynamicSelectFieldConfig } from '../../../dynamic-select/dynamic-select-field.component';\nimport { DEFAULT_MODULAR_LABELS, FolderNode, ModularLabels } from '../modular-table-template.models';\n\ninterface ParentOption {\n  id: number;\n  label: string;\n}\n\nexport const DEFAULT_FOLDER_COLOR = '#99999E';\n\nconst PRESET_FOLDER_COLORS: string[] = [\n  DEFAULT_FOLDER_COLOR,\n  '#4F46E5',\n  '#06B6D4',\n  '#10B981',\n  '#F59E0B',\n  '#EF4444',\n  '#EC4899',\n  '#8B5CF6',\n];\n\n/**\n * Body of the \"New Folder\" dialog.\n *\n * Uses reusable library components (`cqa-custom-input` for the name field,\n * `cqa-dynamic-select` for the parent dropdown) so hosts can drop this dialog\n * into their own modal shell without worrying about styling drift.\n *\n * Two integration paths:\n *\n *  1. **From within this library** via `DialogService.open(...)`. The outer\n *     `cqa-dialog` supplies title/description/Cancel/Create-folder buttons; the\n *     Create button reads `name` + `parentId` via `getComponentInstance()`.\n *\n *  2. **Host-driven** — host renders `<cqa-new-folder-dialog>` inside its own\n *     modal and listens to `(submitted)` / `(cancelled)` / `(nameChange)` /\n *     `(parentIdChange)`.\n */\n@Component({\n  selector: 'cqa-new-folder-dialog',\n  template: `\n    <div class=\"cqa-flex cqa-flex-col cqa-gap-5 cqa-w-full\">\n      <!-- Folder name -->\n      <cqa-custom-input\n        [label]=\"labels.newFolderDialogNameLabel\"\n        [required]=\"true\"\n        [placeholder]=\"labels.newFolderDialogNamePlaceholder\"\n        [value]=\"name\"\n        [fullWidth]=\"true\"\n        (valueChange)=\"onNameChange($event)\"\n        (enterPressed)=\"submit()\"\n      ></cqa-custom-input>\n\n      <!-- Parent folder -->\n      <cqa-dynamic-select\n        class=\"cqa-block cqa-w-full\"\n        [form]=\"parentForm\"\n        [config]=\"parentSelectConfig\"\n        (selectionChange)=\"onParentSelectionChange($event)\"\n      ></cqa-dynamic-select>\n\n      <!-- Folder color (optional) -->\n      <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n        <label class=\"cqa-text-sm cqa-font-medium cqa-text-neutral-700\">\n          {{ labels.newFolderDialogColorLabel }}\n        </label>\n        <div class=\"cqa-flex cqa-items-center cqa-gap-2.5 cqa-flex-wrap\">\n          <button\n            *ngFor=\"let c of presetColors\"\n            type=\"button\"\n            class=\"cqa-relative cqa-w-7 cqa-h-7 cqa-rounded-full cqa-cursor-pointer cqa-border-0 cqa-p-0 cqa-transition-transform\"\n            [style.backgroundColor]=\"c\"\n            [style.boxShadow]=\"isColor(c) ? ('0 0 0 2px #ffffff, 0 0 0 4px ' + c) : '0 0 0 1px rgba(15, 23, 42, 0.08)'\"\n            (click)=\"onColorChange(c)\"\n            [attr.aria-label]=\"'Set folder color ' + c\"\n            [attr.aria-pressed]=\"isColor(c)\"\n            [title]=\"c\"\n          >\n            <svg\n              *ngIf=\"isColor(c)\"\n              class=\"cqa-absolute cqa-inset-0 cqa-m-auto\"\n              width=\"14\" height=\"14\" viewBox=\"0 0 24 24\" fill=\"none\" aria-hidden=\"true\"\n            >\n              <path d=\"M5 12l4.5 4.5L19 7\" stroke=\"#ffffff\" stroke-width=\"3\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n            </svg>\n          </button>\n\n          <label\n            class=\"cqa-relative cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-rounded-full cqa-cursor-pointer cqa-transition-transform\"\n            [style.background]=\"rainbowBorder\"\n            [title]=\"isCustomColor ? ('Custom color (' + color + ')') : 'Pick a custom color'\"\n          >\n            <span\n              class=\"cqa-absolute cqa-flex cqa-items-center cqa-justify-center cqa-rounded-full\"\n              [style.top.px]=\"2\"\n              [style.right.px]=\"2\"\n              [style.bottom.px]=\"2\"\n              [style.left.px]=\"2\"\n              [style.backgroundColor]=\"isCustomColor ? color : '#ffffff'\"\n            >\n              <svg *ngIf=\"!isCustomColor\" width=\"14\" height=\"14\" viewBox=\"0 0 24 24\" fill=\"none\" aria-hidden=\"true\">\n                <path d=\"M12 5v14M5 12h14\" stroke=\"#525258\" stroke-width=\"2\" stroke-linecap=\"round\"/>\n              </svg>\n              <svg *ngIf=\"isCustomColor\" width=\"14\" height=\"14\" viewBox=\"0 0 24 24\" fill=\"none\" aria-hidden=\"true\">\n                <path d=\"M5 12l4.5 4.5L19 7\" stroke=\"#ffffff\" stroke-width=\"3\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n              </svg>\n            </span>\n            <input\n              type=\"color\"\n              [value]=\"color\"\n              (input)=\"onColorInputEvent($event)\"\n              class=\"cqa-absolute cqa-inset-0 cqa-w-full cqa-h-full cqa-opacity-0 cqa-cursor-pointer cqa-border-0 cqa-p-0\"\n              aria-label=\"Pick custom folder color\"\n            />\n          </label>\n        </div>\n      </div>\n    </div>\n  `,\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class NewFolderDialogComponent implements OnInit {\n  // NOTE: These inputs are declared as setters (instead of plain @Input fields)\n  // because `DialogService.open(...)` assigns the `inputs` map via direct\n  // property assignment *after* `attachComponent` has already run the\n  // component's `ngOnInit`. Direct assignment bypasses `ngOnChanges`, so without\n  // setters the `parentSelectConfig` would be built from a default empty folders\n  // array and the parent-folder dropdown would render empty / appear broken.\n\n  private _folders: FolderNode[] = [];\n  @Input() set folders(value: FolderNode[]) {\n    this._folders = value || [];\n    this.rebuildOptions();\n    this.rebuildSelectConfig();\n  }\n  get folders(): FolderNode[] { return this._folders; }\n\n  private _labels: ModularLabels = { ...DEFAULT_MODULAR_LABELS };\n  @Input() set labels(value: ModularLabels) {\n    this._labels = value || { ...DEFAULT_MODULAR_LABELS };\n    this.rebuildSelectConfig();\n  }\n  get labels(): ModularLabels { return this._labels; }\n\n  @Input() name: string = '';\n\n  private _parentId: number | null = null;\n  @Input() set parentId(value: number | null) {\n    this._parentId = value ?? null;\n    this.syncParentControl();\n  }\n  get parentId(): number | null { return this._parentId; }\n\n  @Input() color: string = DEFAULT_FOLDER_COLOR;\n\n  readonly presetColors: string[] = PRESET_FOLDER_COLORS;\n  readonly rainbowBorder: string =\n    'conic-gradient(from 0deg, #ef4444, #f59e0b, #eab308, #10b981, #06b6d4, #4f46e5, #8b5cf6, #ec4899, #ef4444)';\n\n  @Output() nameChange = new EventEmitter<string>();\n  @Output() parentIdChange = new EventEmitter<number | null>();\n  @Output() colorChange = new EventEmitter<string>();\n  @Output() submitted = new EventEmitter<{ name: string; parentId: number | null; color: string }>();\n  @Output() cancelled = new EventEmitter<void>();\n\n  /** Reactive form used by cqa-dynamic-select. Untyped to match the project's Angular 13.4 forms API. */\n  parentForm = new FormGroup({ parentId: new FormControl(null) });\n\n  /**\n   * Config passed to `cqa-dynamic-select`. Stored as a field (not a getter) so\n   * the object reference is stable across change-detection passes — a getter\n   * returns a fresh object every CD tick and causes\n   * `ExpressionChangedAfterItHasBeenCheckedError` + repeated `ngOnChanges` on\n   * the child select. Other consumers in this lib (e.g. `loop-step`) follow\n   * the same pattern.\n   */\n  parentSelectConfig!: DynamicSelectFieldConfig;\n\n  private parentOptions: ParentOption[] = [];\n\n  constructor(private cdr: ChangeDetectorRef) {\n    this.rebuildSelectConfig(); // ensure config exists even before first input set\n  }\n\n  ngOnInit(): void {\n    // Setters above already rebuilt config once folders/labels were assigned. This\n    // just ensures the form control matches the current parentId value in case\n    // ordering of input assignment differs.\n    this.syncParentControl();\n  }\n\n  private rebuildSelectConfig(): void {\n    this.parentSelectConfig = {\n      key: 'parentId',\n      label: this.labels.newFolderDialogParentLabel,\n      placeholder: this.labels.newFolderDialogParentNone,\n      searchable: false,\n      multiple: false,\n      closeOnSelect: true,\n      options: [\n        // `getOptionValue` in cqa-dynamic-select returns `id ?? value`, so using\n        // `value: null` (with no id) yields a stable `null` resolved value that\n        // matches the FormControl's null state.\n        { value: null, name: this.labels.newFolderDialogParentNone },\n        ...this.parentOptions.map(o => ({ id: o.id, name: o.label })),\n      ],\n    };\n  }\n\n  private rebuildOptions(): void {\n    const opts: ParentOption[] = [];\n    const walk = (nodes: FolderNode[], trail: string[]): void => {\n      for (const n of nodes || []) {\n        const label = [...trail, n.name].join(' › ');\n        opts.push({ id: n.id, label });\n        if (n.children?.length) walk(n.children, [...trail, n.name]);\n      }\n    };\n    walk(this.folders, []);\n    this.parentOptions = opts;\n    this.cdr.markForCheck();\n  }\n\n  private syncParentControl(): void {\n    // Keep the reactive form in sync with the `parentId` input without bouncing events.\n    this.parentForm.get('parentId')?.setValue(this.parentId ?? null, { emitEvent: false });\n  }\n\n  onNameChange(value: string): void {\n    this.name = value;\n    this.nameChange.emit(value);\n  }\n\n  onParentSelectionChange(event: { key: string; value: any }): void {\n    // cqa-dynamic-select resolves options to `id ?? value`; option ids in the\n    // parent tree are numeric so we coerce defensively.\n    const raw = event?.value;\n    let next: number | null = null;\n    if (raw != null) {\n      const coerced = typeof raw === 'number' ? raw : Number(raw);\n      if (Number.isFinite(coerced)) next = coerced;\n    }\n    this.parentId = next;\n    this.parentIdChange.emit(next);\n  }\n\n  isColor(c: string): boolean {\n    return (this.color || '').toLowerCase() === (c || '').toLowerCase();\n  }\n\n  get isCustomColor(): boolean {\n    const c = (this.color || '').toLowerCase();\n    return !!c && !this.presetColors.some(p => p.toLowerCase() === c);\n  }\n\n  onColorChange(value: string): void {\n    const next = value || DEFAULT_FOLDER_COLOR;\n    if (this.color === next) return;\n    this.color = next;\n    this.colorChange.emit(next);\n    this.cdr.markForCheck();\n  }\n\n  onColorInputEvent(event: Event): void {\n    const target = event.target as HTMLInputElement | null;\n    if (target) this.onColorChange(target.value);\n  }\n\n  /** Returns true when the current state is a valid submission (non-empty, trimmed name). */\n  get isValid(): boolean {\n    return !!(this.name || '').trim();\n  }\n\n  submit(): void {\n    if (!this.isValid) return;\n    this.submitted.emit({\n      name: (this.name || '').trim(),\n      parentId: this.parentId,\n      color: this.color || DEFAULT_FOLDER_COLOR,\n    });\n  }\n}\n"]}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { Directive, HostBinding, HostListener, Input } from '@angular/core';
|
|
2
|
+
import { FOLDER_DRAG_MIME } from '../modular-table-template.models';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class FolderDragDirective {
|
|
5
|
+
constructor() {
|
|
6
|
+
this.dragEnabled = true;
|
|
7
|
+
}
|
|
8
|
+
get draggable() {
|
|
9
|
+
return this.dragEnabled && this.folderId != null ? 'true' : null;
|
|
10
|
+
}
|
|
11
|
+
onDragStart(event) {
|
|
12
|
+
if (!this.dragEnabled || this.folderId == null || !event.dataTransfer)
|
|
13
|
+
return;
|
|
14
|
+
event.stopPropagation();
|
|
15
|
+
const payload = JSON.stringify(this.folderId);
|
|
16
|
+
event.dataTransfer.effectAllowed = 'move';
|
|
17
|
+
try {
|
|
18
|
+
event.dataTransfer.setData(FOLDER_DRAG_MIME, payload);
|
|
19
|
+
}
|
|
20
|
+
catch {
|
|
21
|
+
/* some browsers reject custom MIME — fall through */
|
|
22
|
+
}
|
|
23
|
+
event.dataTransfer.setData('text/plain', payload);
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
FolderDragDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: FolderDragDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
27
|
+
FolderDragDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.4.0", type: FolderDragDirective, selector: "[cqaFolderDrag]", inputs: { folderId: ["cqaFolderDrag", "folderId"], dragEnabled: "dragEnabled" }, host: { listeners: { "dragstart": "onDragStart($event)" }, properties: { "attr.draggable": "this.draggable" } }, ngImport: i0 });
|
|
28
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: FolderDragDirective, decorators: [{
|
|
29
|
+
type: Directive,
|
|
30
|
+
args: [{
|
|
31
|
+
selector: '[cqaFolderDrag]',
|
|
32
|
+
}]
|
|
33
|
+
}], propDecorators: { folderId: [{
|
|
34
|
+
type: Input,
|
|
35
|
+
args: ['cqaFolderDrag']
|
|
36
|
+
}], dragEnabled: [{
|
|
37
|
+
type: Input
|
|
38
|
+
}], draggable: [{
|
|
39
|
+
type: HostBinding,
|
|
40
|
+
args: ['attr.draggable']
|
|
41
|
+
}], onDragStart: [{
|
|
42
|
+
type: HostListener,
|
|
43
|
+
args: ['dragstart', ['$event']]
|
|
44
|
+
}] } });
|
|
45
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9sZGVyLWRyYWcuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi90ZW1wbGF0ZXMvbW9kdWxhci10YWJsZS10ZW1wbGF0ZS9kaXJlY3RpdmVzL2ZvbGRlci1kcmFnLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLFdBQVcsRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzVFLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLGtDQUFrQyxDQUFDOztBQUtwRSxNQUFNLE9BQU8sbUJBQW1CO0lBSGhDO1FBTVcsZ0JBQVcsR0FBWSxJQUFJLENBQUM7S0FrQnRDO0lBaEJDLElBQW1DLFNBQVM7UUFDMUMsT0FBTyxJQUFJLENBQUMsV0FBVyxJQUFJLElBQUksQ0FBQyxRQUFRLElBQUksSUFBSSxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQztJQUNuRSxDQUFDO0lBRXNDLFdBQVcsQ0FBQyxLQUFnQjtRQUNqRSxJQUFJLENBQUMsSUFBSSxDQUFDLFdBQVcsSUFBSSxJQUFJLENBQUMsUUFBUSxJQUFJLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxZQUFZO1lBQUUsT0FBTztRQUM5RSxLQUFLLENBQUMsZUFBZSxFQUFFLENBQUM7UUFDeEIsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUM7UUFDOUMsS0FBSyxDQUFDLFlBQVksQ0FBQyxhQUFhLEdBQUcsTUFBTSxDQUFDO1FBQzFDLElBQUk7WUFDRixLQUFLLENBQUMsWUFBWSxDQUFDLE9BQU8sQ0FBQyxnQkFBZ0IsRUFBRSxPQUFPLENBQUMsQ0FBQztTQUN2RDtRQUFDLE1BQU07WUFDTixxREFBcUQ7U0FDdEQ7UUFDRCxLQUFLLENBQUMsWUFBWSxDQUFDLE9BQU8sQ0FBQyxZQUFZLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDcEQsQ0FBQzs7Z0hBcEJVLG1CQUFtQjtvR0FBbkIsbUJBQW1COzJGQUFuQixtQkFBbUI7a0JBSC9CLFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLGlCQUFpQjtpQkFDNUI7OEJBR3lCLFFBQVE7c0JBQS9CLEtBQUs7dUJBQUMsZUFBZTtnQkFDYixXQUFXO3NCQUFuQixLQUFLO2dCQUU2QixTQUFTO3NCQUEzQyxXQUFXO3VCQUFDLGdCQUFnQjtnQkFJVSxXQUFXO3NCQUFqRCxZQUFZO3VCQUFDLFdBQVcsRUFBRSxDQUFDLFFBQVEsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERpcmVjdGl2ZSwgSG9zdEJpbmRpbmcsIEhvc3RMaXN0ZW5lciwgSW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEZPTERFUl9EUkFHX01JTUUgfSBmcm9tICcuLi9tb2R1bGFyLXRhYmxlLXRlbXBsYXRlLm1vZGVscyc7XG5cbkBEaXJlY3RpdmUoe1xuICBzZWxlY3RvcjogJ1tjcWFGb2xkZXJEcmFnXScsXG59KVxuZXhwb3J0IGNsYXNzIEZvbGRlckRyYWdEaXJlY3RpdmUge1xuICAvKiogRm9sZGVyIGlkIGJlaW5nIGRyYWdnZWQuICovXG4gIEBJbnB1dCgnY3FhRm9sZGVyRHJhZycpIGZvbGRlcklkITogbnVtYmVyIHwgbnVsbDtcbiAgQElucHV0KCkgZHJhZ0VuYWJsZWQ6IGJvb2xlYW4gPSB0cnVlO1xuXG4gIEBIb3N0QmluZGluZygnYXR0ci5kcmFnZ2FibGUnKSBnZXQgZHJhZ2dhYmxlKCkge1xuICAgIHJldHVybiB0aGlzLmRyYWdFbmFibGVkICYmIHRoaXMuZm9sZGVySWQgIT0gbnVsbCA/ICd0cnVlJyA6IG51bGw7XG4gIH1cblxuICBASG9zdExpc3RlbmVyKCdkcmFnc3RhcnQnLCBbJyRldmVudCddKSBvbkRyYWdTdGFydChldmVudDogRHJhZ0V2ZW50KSB7XG4gICAgaWYgKCF0aGlzLmRyYWdFbmFibGVkIHx8IHRoaXMuZm9sZGVySWQgPT0gbnVsbCB8fCAhZXZlbnQuZGF0YVRyYW5zZmVyKSByZXR1cm47XG4gICAgZXZlbnQuc3RvcFByb3BhZ2F0aW9uKCk7XG4gICAgY29uc3QgcGF5bG9hZCA9IEpTT04uc3RyaW5naWZ5KHRoaXMuZm9sZGVySWQpO1xuICAgIGV2ZW50LmRhdGFUcmFuc2Zlci5lZmZlY3RBbGxvd2VkID0gJ21vdmUnO1xuICAgIHRyeSB7XG4gICAgICBldmVudC5kYXRhVHJhbnNmZXIuc2V0RGF0YShGT0xERVJfRFJBR19NSU1FLCBwYXlsb2FkKTtcbiAgICB9IGNhdGNoIHtcbiAgICAgIC8qIHNvbWUgYnJvd3NlcnMgcmVqZWN0IGN1c3RvbSBNSU1FIOKAlCBmYWxsIHRocm91Z2ggKi9cbiAgICB9XG4gICAgZXZlbnQuZGF0YVRyYW5zZmVyLnNldERhdGEoJ3RleHQvcGxhaW4nLCBwYXlsb2FkKTtcbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
import { Directive, EventEmitter, HostBinding, HostListener, Input, Output } from '@angular/core';
|
|
2
|
+
import { FOLDER_DRAG_MIME, ROW_DRAG_MIME } from '../modular-table-template.models';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class FolderDropDirective {
|
|
5
|
+
constructor() {
|
|
6
|
+
/** Folder id this drop target represents. `null` means "unorganised / root". */
|
|
7
|
+
this.targetFolderId = null;
|
|
8
|
+
this.dropEnabled = true;
|
|
9
|
+
this.testsDropped = new EventEmitter();
|
|
10
|
+
/** Fires when a folder row is dropped here. `folderId` is the dragged folder's id. */
|
|
11
|
+
this.folderDropped = new EventEmitter();
|
|
12
|
+
this.isOver = false;
|
|
13
|
+
}
|
|
14
|
+
onDragOver(event) {
|
|
15
|
+
if (!this.dropEnabled)
|
|
16
|
+
return;
|
|
17
|
+
event.preventDefault();
|
|
18
|
+
if (event.dataTransfer) {
|
|
19
|
+
event.dataTransfer.dropEffect = 'move';
|
|
20
|
+
}
|
|
21
|
+
this.isOver = true;
|
|
22
|
+
}
|
|
23
|
+
onDragLeave() {
|
|
24
|
+
this.isOver = false;
|
|
25
|
+
}
|
|
26
|
+
onDrop(event) {
|
|
27
|
+
this.isOver = false;
|
|
28
|
+
if (!this.dropEnabled || !event.dataTransfer)
|
|
29
|
+
return;
|
|
30
|
+
event.preventDefault();
|
|
31
|
+
// Folder drops take precedence — the sidebar writes the folder MIME on dragstart.
|
|
32
|
+
const folderRaw = event.dataTransfer.getData(FOLDER_DRAG_MIME);
|
|
33
|
+
if (folderRaw) {
|
|
34
|
+
let folderId = null;
|
|
35
|
+
try {
|
|
36
|
+
const parsed = JSON.parse(folderRaw);
|
|
37
|
+
if (typeof parsed === 'number' && Number.isFinite(parsed))
|
|
38
|
+
folderId = parsed;
|
|
39
|
+
}
|
|
40
|
+
catch {
|
|
41
|
+
return;
|
|
42
|
+
}
|
|
43
|
+
if (folderId == null)
|
|
44
|
+
return;
|
|
45
|
+
this.folderDropped.emit({ folderId, targetFolderId: this.targetFolderId });
|
|
46
|
+
return;
|
|
47
|
+
}
|
|
48
|
+
// Fall through to the test-ids path (unchanged).
|
|
49
|
+
const raw = event.dataTransfer.getData(ROW_DRAG_MIME) || event.dataTransfer.getData('text/plain');
|
|
50
|
+
if (!raw)
|
|
51
|
+
return;
|
|
52
|
+
let ids = [];
|
|
53
|
+
try {
|
|
54
|
+
const parsed = JSON.parse(raw);
|
|
55
|
+
if (Array.isArray(parsed))
|
|
56
|
+
ids = parsed;
|
|
57
|
+
}
|
|
58
|
+
catch {
|
|
59
|
+
return;
|
|
60
|
+
}
|
|
61
|
+
if (!ids.length)
|
|
62
|
+
return;
|
|
63
|
+
this.testsDropped.emit({ testIds: ids, targetFolderId: this.targetFolderId });
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
FolderDropDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: FolderDropDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
67
|
+
FolderDropDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.4.0", type: FolderDropDirective, selector: "[cqaFolderDrop]", inputs: { targetFolderId: ["cqaFolderDrop", "targetFolderId"], dropEnabled: "dropEnabled" }, outputs: { testsDropped: "testsDropped", folderDropped: "folderDropped" }, host: { listeners: { "dragover": "onDragOver($event)", "dragleave": "onDragLeave()", "drop": "onDrop($event)" }, properties: { "class.cqa-folder-drop-over": "this.isOver" } }, ngImport: i0 });
|
|
68
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: FolderDropDirective, decorators: [{
|
|
69
|
+
type: Directive,
|
|
70
|
+
args: [{
|
|
71
|
+
selector: '[cqaFolderDrop]',
|
|
72
|
+
}]
|
|
73
|
+
}], propDecorators: { targetFolderId: [{
|
|
74
|
+
type: Input,
|
|
75
|
+
args: ['cqaFolderDrop']
|
|
76
|
+
}], dropEnabled: [{
|
|
77
|
+
type: Input
|
|
78
|
+
}], testsDropped: [{
|
|
79
|
+
type: Output
|
|
80
|
+
}], folderDropped: [{
|
|
81
|
+
type: Output
|
|
82
|
+
}], isOver: [{
|
|
83
|
+
type: HostBinding,
|
|
84
|
+
args: ['class.cqa-folder-drop-over']
|
|
85
|
+
}], onDragOver: [{
|
|
86
|
+
type: HostListener,
|
|
87
|
+
args: ['dragover', ['$event']]
|
|
88
|
+
}], onDragLeave: [{
|
|
89
|
+
type: HostListener,
|
|
90
|
+
args: ['dragleave']
|
|
91
|
+
}], onDrop: [{
|
|
92
|
+
type: HostListener,
|
|
93
|
+
args: ['drop', ['$event']]
|
|
94
|
+
}] } });
|
|
95
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9sZGVyLWRyb3AuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi90ZW1wbGF0ZXMvbW9kdWxhci10YWJsZS10ZW1wbGF0ZS9kaXJlY3RpdmVzL2ZvbGRlci1kcm9wLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLFlBQVksRUFBRSxXQUFXLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDbEcsT0FBTyxFQUFFLGdCQUFnQixFQUFFLGFBQWEsRUFBRSxNQUFNLGtDQUFrQyxDQUFDOztBQUtuRixNQUFNLE9BQU8sbUJBQW1CO0lBSGhDO1FBSUUsZ0ZBQWdGO1FBQ3hELG1CQUFjLEdBQWtCLElBQUksQ0FBQztRQUNwRCxnQkFBVyxHQUFZLElBQUksQ0FBQztRQUUzQixpQkFBWSxHQUFHLElBQUksWUFBWSxFQUFzRSxDQUFDO1FBQ2hILHNGQUFzRjtRQUM1RSxrQkFBYSxHQUFHLElBQUksWUFBWSxFQUF1RCxDQUFDO1FBRXZELFdBQU0sR0FBRyxLQUFLLENBQUM7S0FnRDNEO0lBOUN1QyxVQUFVLENBQUMsS0FBZ0I7UUFDL0QsSUFBSSxDQUFDLElBQUksQ0FBQyxXQUFXO1lBQUUsT0FBTztRQUM5QixLQUFLLENBQUMsY0FBYyxFQUFFLENBQUM7UUFDdkIsSUFBSSxLQUFLLENBQUMsWUFBWSxFQUFFO1lBQ3RCLEtBQUssQ0FBQyxZQUFZLENBQUMsVUFBVSxHQUFHLE1BQU0sQ0FBQztTQUN4QztRQUNELElBQUksQ0FBQyxNQUFNLEdBQUcsSUFBSSxDQUFDO0lBQ3JCLENBQUM7SUFFMEIsV0FBVztRQUNwQyxJQUFJLENBQUMsTUFBTSxHQUFHLEtBQUssQ0FBQztJQUN0QixDQUFDO0lBRWlDLE1BQU0sQ0FBQyxLQUFnQjtRQUN2RCxJQUFJLENBQUMsTUFBTSxHQUFHLEtBQUssQ0FBQztRQUNwQixJQUFJLENBQUMsSUFBSSxDQUFDLFdBQVcsSUFBSSxDQUFDLEtBQUssQ0FBQyxZQUFZO1lBQUUsT0FBTztRQUNyRCxLQUFLLENBQUMsY0FBYyxFQUFFLENBQUM7UUFFdkIsa0ZBQWtGO1FBQ2xGLE1BQU0sU0FBUyxHQUFHLEtBQUssQ0FBQyxZQUFZLENBQUMsT0FBTyxDQUFDLGdCQUFnQixDQUFDLENBQUM7UUFDL0QsSUFBSSxTQUFTLEVBQUU7WUFDYixJQUFJLFFBQVEsR0FBa0IsSUFBSSxDQUFDO1lBQ25DLElBQUk7Z0JBQ0YsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsQ0FBQztnQkFDckMsSUFBSSxPQUFPLE1BQU0sS0FBSyxRQUFRLElBQUksTUFBTSxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUM7b0JBQUUsUUFBUSxHQUFHLE1BQU0sQ0FBQzthQUM5RTtZQUFDLE1BQU07Z0JBQ04sT0FBTzthQUNSO1lBQ0QsSUFBSSxRQUFRLElBQUksSUFBSTtnQkFBRSxPQUFPO1lBQzdCLElBQUksQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLEVBQUUsUUFBUSxFQUFFLGNBQWMsRUFBRSxJQUFJLENBQUMsY0FBYyxFQUFFLENBQUMsQ0FBQztZQUMzRSxPQUFPO1NBQ1I7UUFFRCxpREFBaUQ7UUFDakQsTUFBTSxHQUFHLEdBQUcsS0FBSyxDQUFDLFlBQVksQ0FBQyxPQUFPLENBQUMsYUFBYSxDQUFDLElBQUksS0FBSyxDQUFDLFlBQVksQ0FBQyxPQUFPLENBQUMsWUFBWSxDQUFDLENBQUM7UUFDbEcsSUFBSSxDQUFDLEdBQUc7WUFBRSxPQUFPO1FBQ2pCLElBQUksR0FBRyxHQUEyQixFQUFFLENBQUM7UUFDckMsSUFBSTtZQUNGLE1BQU0sTUFBTSxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUM7WUFDL0IsSUFBSSxLQUFLLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQztnQkFBRSxHQUFHLEdBQUcsTUFBTSxDQUFDO1NBQ3pDO1FBQUMsTUFBTTtZQUNOLE9BQU87U0FDUjtRQUNELElBQUksQ0FBQyxHQUFHLENBQUMsTUFBTTtZQUFFLE9BQU87UUFDeEIsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsRUFBRSxPQUFPLEVBQUUsR0FBRyxFQUFFLGNBQWMsRUFBRSxJQUFJLENBQUMsY0FBYyxFQUFFLENBQUMsQ0FBQztJQUNoRixDQUFDOztnSEF4RFUsbUJBQW1CO29HQUFuQixtQkFBbUI7MkZBQW5CLG1CQUFtQjtrQkFIL0IsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsaUJBQWlCO2lCQUM1Qjs4QkFHeUIsY0FBYztzQkFBckMsS0FBSzt1QkFBQyxlQUFlO2dCQUNiLFdBQVc7c0JBQW5CLEtBQUs7Z0JBRUksWUFBWTtzQkFBckIsTUFBTTtnQkFFRyxhQUFhO3NCQUF0QixNQUFNO2dCQUVvQyxNQUFNO3NCQUFoRCxXQUFXO3VCQUFDLDRCQUE0QjtnQkFFSCxVQUFVO3NCQUEvQyxZQUFZO3VCQUFDLFVBQVUsRUFBRSxDQUFDLFFBQVEsQ0FBQztnQkFTVCxXQUFXO3NCQUFyQyxZQUFZO3VCQUFDLFdBQVc7Z0JBSVMsTUFBTTtzQkFBdkMsWUFBWTt1QkFBQyxNQUFNLEVBQUUsQ0FBQyxRQUFRLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEaXJlY3RpdmUsIEV2ZW50RW1pdHRlciwgSG9zdEJpbmRpbmcsIEhvc3RMaXN0ZW5lciwgSW5wdXQsIE91dHB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgRk9MREVSX0RSQUdfTUlNRSwgUk9XX0RSQUdfTUlNRSB9IGZyb20gJy4uL21vZHVsYXItdGFibGUtdGVtcGxhdGUubW9kZWxzJztcblxuQERpcmVjdGl2ZSh7XG4gIHNlbGVjdG9yOiAnW2NxYUZvbGRlckRyb3BdJyxcbn0pXG5leHBvcnQgY2xhc3MgRm9sZGVyRHJvcERpcmVjdGl2ZSB7XG4gIC8qKiBGb2xkZXIgaWQgdGhpcyBkcm9wIHRhcmdldCByZXByZXNlbnRzLiBgbnVsbGAgbWVhbnMgXCJ1bm9yZ2FuaXNlZCAvIHJvb3RcIi4gKi9cbiAgQElucHV0KCdjcWFGb2xkZXJEcm9wJykgdGFyZ2V0Rm9sZGVySWQ6IG51bWJlciB8IG51bGwgPSBudWxsO1xuICBASW5wdXQoKSBkcm9wRW5hYmxlZDogYm9vbGVhbiA9IHRydWU7XG5cbiAgQE91dHB1dCgpIHRlc3RzRHJvcHBlZCA9IG5ldyBFdmVudEVtaXR0ZXI8eyB0ZXN0SWRzOiBBcnJheTxzdHJpbmcgfCBudW1iZXI+OyB0YXJnZXRGb2xkZXJJZDogbnVtYmVyIHwgbnVsbCB9PigpO1xuICAvKiogRmlyZXMgd2hlbiBhIGZvbGRlciByb3cgaXMgZHJvcHBlZCBoZXJlLiBgZm9sZGVySWRgIGlzIHRoZSBkcmFnZ2VkIGZvbGRlcidzIGlkLiAqL1xuICBAT3V0cHV0KCkgZm9sZGVyRHJvcHBlZCA9IG5ldyBFdmVudEVtaXR0ZXI8eyBmb2xkZXJJZDogbnVtYmVyOyB0YXJnZXRGb2xkZXJJZDogbnVtYmVyIHwgbnVsbCB9PigpO1xuXG4gIEBIb3N0QmluZGluZygnY2xhc3MuY3FhLWZvbGRlci1kcm9wLW92ZXInKSBpc092ZXIgPSBmYWxzZTtcblxuICBASG9zdExpc3RlbmVyKCdkcmFnb3ZlcicsIFsnJGV2ZW50J10pIG9uRHJhZ092ZXIoZXZlbnQ6IERyYWdFdmVudCkge1xuICAgIGlmICghdGhpcy5kcm9wRW5hYmxlZCkgcmV0dXJuO1xuICAgIGV2ZW50LnByZXZlbnREZWZhdWx0KCk7XG4gICAgaWYgKGV2ZW50LmRhdGFUcmFuc2Zlcikge1xuICAgICAgZXZlbnQuZGF0YVRyYW5zZmVyLmRyb3BFZmZlY3QgPSAnbW92ZSc7XG4gICAgfVxuICAgIHRoaXMuaXNPdmVyID0gdHJ1ZTtcbiAgfVxuXG4gIEBIb3N0TGlzdGVuZXIoJ2RyYWdsZWF2ZScpIG9uRHJhZ0xlYXZlKCkge1xuICAgIHRoaXMuaXNPdmVyID0gZmFsc2U7XG4gIH1cblxuICBASG9zdExpc3RlbmVyKCdkcm9wJywgWyckZXZlbnQnXSkgb25Ecm9wKGV2ZW50OiBEcmFnRXZlbnQpIHtcbiAgICB0aGlzLmlzT3ZlciA9IGZhbHNlO1xuICAgIGlmICghdGhpcy5kcm9wRW5hYmxlZCB8fCAhZXZlbnQuZGF0YVRyYW5zZmVyKSByZXR1cm47XG4gICAgZXZlbnQucHJldmVudERlZmF1bHQoKTtcblxuICAgIC8vIEZvbGRlciBkcm9wcyB0YWtlIHByZWNlZGVuY2Ug4oCUIHRoZSBzaWRlYmFyIHdyaXRlcyB0aGUgZm9sZGVyIE1JTUUgb24gZHJhZ3N0YXJ0LlxuICAgIGNvbnN0IGZvbGRlclJhdyA9IGV2ZW50LmRhdGFUcmFuc2Zlci5nZXREYXRhKEZPTERFUl9EUkFHX01JTUUpO1xuICAgIGlmIChmb2xkZXJSYXcpIHtcbiAgICAgIGxldCBmb2xkZXJJZDogbnVtYmVyIHwgbnVsbCA9IG51bGw7XG4gICAgICB0cnkge1xuICAgICAgICBjb25zdCBwYXJzZWQgPSBKU09OLnBhcnNlKGZvbGRlclJhdyk7XG4gICAgICAgIGlmICh0eXBlb2YgcGFyc2VkID09PSAnbnVtYmVyJyAmJiBOdW1iZXIuaXNGaW5pdGUocGFyc2VkKSkgZm9sZGVySWQgPSBwYXJzZWQ7XG4gICAgICB9IGNhdGNoIHtcbiAgICAgICAgcmV0dXJuO1xuICAgICAgfVxuICAgICAgaWYgKGZvbGRlcklkID09IG51bGwpIHJldHVybjtcbiAgICAgIHRoaXMuZm9sZGVyRHJvcHBlZC5lbWl0KHsgZm9sZGVySWQsIHRhcmdldEZvbGRlcklkOiB0aGlzLnRhcmdldEZvbGRlcklkIH0pO1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIC8vIEZhbGwgdGhyb3VnaCB0byB0aGUgdGVzdC1pZHMgcGF0aCAodW5jaGFuZ2VkKS5cbiAgICBjb25zdCByYXcgPSBldmVudC5kYXRhVHJhbnNmZXIuZ2V0RGF0YShST1dfRFJBR19NSU1FKSB8fCBldmVudC5kYXRhVHJhbnNmZXIuZ2V0RGF0YSgndGV4dC9wbGFpbicpO1xuICAgIGlmICghcmF3KSByZXR1cm47XG4gICAgbGV0IGlkczogQXJyYXk8c3RyaW5nIHwgbnVtYmVyPiA9IFtdO1xuICAgIHRyeSB7XG4gICAgICBjb25zdCBwYXJzZWQgPSBKU09OLnBhcnNlKHJhdyk7XG4gICAgICBpZiAoQXJyYXkuaXNBcnJheShwYXJzZWQpKSBpZHMgPSBwYXJzZWQ7XG4gICAgfSBjYXRjaCB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGlmICghaWRzLmxlbmd0aCkgcmV0dXJuO1xuICAgIHRoaXMudGVzdHNEcm9wcGVkLmVtaXQoeyB0ZXN0SWRzOiBpZHMsIHRhcmdldEZvbGRlcklkOiB0aGlzLnRhcmdldEZvbGRlcklkIH0pO1xuICB9XG59XG4iXX0=
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { Directive, HostBinding, HostListener, Input } from '@angular/core';
|
|
2
|
+
import { ROW_DRAG_MIME } from '../modular-table-template.models';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class RowDragDirective {
|
|
5
|
+
constructor() {
|
|
6
|
+
this.dragEnabled = true;
|
|
7
|
+
}
|
|
8
|
+
get draggable() {
|
|
9
|
+
return this.dragEnabled ? 'true' : null;
|
|
10
|
+
}
|
|
11
|
+
onDragStart(event) {
|
|
12
|
+
if (!this.dragEnabled || !event.dataTransfer || !this.buildPayload)
|
|
13
|
+
return;
|
|
14
|
+
const ids = this.buildPayload() ?? [];
|
|
15
|
+
event.dataTransfer.effectAllowed = 'move';
|
|
16
|
+
try {
|
|
17
|
+
event.dataTransfer.setData(ROW_DRAG_MIME, JSON.stringify(ids));
|
|
18
|
+
}
|
|
19
|
+
catch {
|
|
20
|
+
/* some browsers reject custom MIME — fall through */
|
|
21
|
+
}
|
|
22
|
+
event.dataTransfer.setData('text/plain', JSON.stringify(ids));
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
RowDragDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: RowDragDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
26
|
+
RowDragDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.4.0", type: RowDragDirective, selector: "[cqaRowDrag]", inputs: { buildPayload: ["cqaRowDrag", "buildPayload"], dragEnabled: "dragEnabled" }, host: { listeners: { "dragstart": "onDragStart($event)" }, properties: { "attr.draggable": "this.draggable" } }, ngImport: i0 });
|
|
27
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: RowDragDirective, decorators: [{
|
|
28
|
+
type: Directive,
|
|
29
|
+
args: [{
|
|
30
|
+
selector: '[cqaRowDrag]',
|
|
31
|
+
}]
|
|
32
|
+
}], propDecorators: { buildPayload: [{
|
|
33
|
+
type: Input,
|
|
34
|
+
args: ['cqaRowDrag']
|
|
35
|
+
}], dragEnabled: [{
|
|
36
|
+
type: Input
|
|
37
|
+
}], draggable: [{
|
|
38
|
+
type: HostBinding,
|
|
39
|
+
args: ['attr.draggable']
|
|
40
|
+
}], onDragStart: [{
|
|
41
|
+
type: HostListener,
|
|
42
|
+
args: ['dragstart', ['$event']]
|
|
43
|
+
}] } });
|
|
44
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicm93LWRyYWcuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi90ZW1wbGF0ZXMvbW9kdWxhci10YWJsZS10ZW1wbGF0ZS9kaXJlY3RpdmVzL3Jvdy1kcmFnLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLFdBQVcsRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzVFLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQzs7QUFLakUsTUFBTSxPQUFPLGdCQUFnQjtJQUg3QjtRQU1XLGdCQUFXLEdBQVksSUFBSSxDQUFDO0tBaUJ0QztJQWZDLElBQW1DLFNBQVM7UUFDMUMsT0FBTyxJQUFJLENBQUMsV0FBVyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQztJQUMxQyxDQUFDO0lBRXNDLFdBQVcsQ0FBQyxLQUFnQjtRQUNqRSxJQUFJLENBQUMsSUFBSSxDQUFDLFdBQVcsSUFBSSxDQUFDLEtBQUssQ0FBQyxZQUFZLElBQUksQ0FBQyxJQUFJLENBQUMsWUFBWTtZQUFFLE9BQU87UUFDM0UsTUFBTSxHQUFHLEdBQUcsSUFBSSxDQUFDLFlBQVksRUFBRSxJQUFJLEVBQUUsQ0FBQztRQUN0QyxLQUFLLENBQUMsWUFBWSxDQUFDLGFBQWEsR0FBRyxNQUFNLENBQUM7UUFDMUMsSUFBSTtZQUNGLEtBQUssQ0FBQyxZQUFZLENBQUMsT0FBTyxDQUFDLGFBQWEsRUFBRSxJQUFJLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUM7U0FDaEU7UUFBQyxNQUFNO1lBQ04scURBQXFEO1NBQ3REO1FBQ0QsS0FBSyxDQUFDLFlBQVksQ0FBQyxPQUFPLENBQUMsWUFBWSxFQUFFLElBQUksQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztJQUNoRSxDQUFDOzs2R0FuQlUsZ0JBQWdCO2lHQUFoQixnQkFBZ0I7MkZBQWhCLGdCQUFnQjtrQkFINUIsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsY0FBYztpQkFDekI7OEJBR3NCLFlBQVk7c0JBQWhDLEtBQUs7dUJBQUMsWUFBWTtnQkFDVixXQUFXO3NCQUFuQixLQUFLO2dCQUU2QixTQUFTO3NCQUEzQyxXQUFXO3VCQUFDLGdCQUFnQjtnQkFJVSxXQUFXO3NCQUFqRCxZQUFZO3VCQUFDLFdBQVcsRUFBRSxDQUFDLFFBQVEsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERpcmVjdGl2ZSwgSG9zdEJpbmRpbmcsIEhvc3RMaXN0ZW5lciwgSW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFJPV19EUkFHX01JTUUgfSBmcm9tICcuLi9tb2R1bGFyLXRhYmxlLXRlbXBsYXRlLm1vZGVscyc7XG5cbkBEaXJlY3RpdmUoe1xuICBzZWxlY3RvcjogJ1tjcWFSb3dEcmFnXScsXG59KVxuZXhwb3J0IGNsYXNzIFJvd0RyYWdEaXJlY3RpdmUge1xuICAvKiogRnVuY3Rpb24gdGhhdCByZXR1cm5zIHRoZSB0ZXN0IGlkIHBheWxvYWQgZm9yIHRoaXMgcm93IChhcnJheSBvZiBpZHMpLiAqL1xuICBASW5wdXQoJ2NxYVJvd0RyYWcnKSBidWlsZFBheWxvYWQhOiAoKSA9PiBBcnJheTxzdHJpbmcgfCBudW1iZXI+O1xuICBASW5wdXQoKSBkcmFnRW5hYmxlZDogYm9vbGVhbiA9IHRydWU7XG5cbiAgQEhvc3RCaW5kaW5nKCdhdHRyLmRyYWdnYWJsZScpIGdldCBkcmFnZ2FibGUoKSB7XG4gICAgcmV0dXJuIHRoaXMuZHJhZ0VuYWJsZWQgPyAndHJ1ZScgOiBudWxsO1xuICB9XG5cbiAgQEhvc3RMaXN0ZW5lcignZHJhZ3N0YXJ0JywgWyckZXZlbnQnXSkgb25EcmFnU3RhcnQoZXZlbnQ6IERyYWdFdmVudCkge1xuICAgIGlmICghdGhpcy5kcmFnRW5hYmxlZCB8fCAhZXZlbnQuZGF0YVRyYW5zZmVyIHx8ICF0aGlzLmJ1aWxkUGF5bG9hZCkgcmV0dXJuO1xuICAgIGNvbnN0IGlkcyA9IHRoaXMuYnVpbGRQYXlsb2FkKCkgPz8gW107XG4gICAgZXZlbnQuZGF0YVRyYW5zZmVyLmVmZmVjdEFsbG93ZWQgPSAnbW92ZSc7XG4gICAgdHJ5IHtcbiAgICAgIGV2ZW50LmRhdGFUcmFuc2Zlci5zZXREYXRhKFJPV19EUkFHX01JTUUsIEpTT04uc3RyaW5naWZ5KGlkcykpO1xuICAgIH0gY2F0Y2gge1xuICAgICAgLyogc29tZSBicm93c2VycyByZWplY3QgY3VzdG9tIE1JTUUg4oCUIGZhbGwgdGhyb3VnaCAqL1xuICAgIH1cbiAgICBldmVudC5kYXRhVHJhbnNmZXIuc2V0RGF0YSgndGV4dC9wbGFpbicsIEpTT04uc3RyaW5naWZ5KGlkcykpO1xuICB9XG59XG4iXX0=
|