@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
package/esm2020/lib/templates/modular-table-template/folder-sidebar/folder-sidebar.component.mjs
ADDED
|
@@ -0,0 +1,479 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, HostListener, Input, Output, } from '@angular/core';
|
|
2
|
+
import { DEFAULT_MODULAR_LABELS, } from '../modular-table-template.models';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/material/icon";
|
|
5
|
+
import * as i2 from "../../../search-bar/search-bar.component";
|
|
6
|
+
import * as i3 from "@angular/common";
|
|
7
|
+
import * as i4 from "../directives/folder-drop.directive";
|
|
8
|
+
import * as i5 from "../directives/folder-drag.directive";
|
|
9
|
+
import * as i6 from "@angular/forms";
|
|
10
|
+
export class FolderSidebarComponent {
|
|
11
|
+
constructor(cdr) {
|
|
12
|
+
this.cdr = cdr;
|
|
13
|
+
this.folders = [];
|
|
14
|
+
this.selectedFolderId = null;
|
|
15
|
+
this.expandedFolderIds = [];
|
|
16
|
+
this.unorganisedCount = 0;
|
|
17
|
+
this.allowCreate = true;
|
|
18
|
+
this.allowRename = true;
|
|
19
|
+
this.allowDelete = true;
|
|
20
|
+
this.allowMove = true;
|
|
21
|
+
this.allowDuplicate = true;
|
|
22
|
+
this.allowDrop = true;
|
|
23
|
+
this.showCounts = true;
|
|
24
|
+
this.collapsed = false;
|
|
25
|
+
this.labels = { ...DEFAULT_MODULAR_LABELS };
|
|
26
|
+
this.folderSelected = new EventEmitter();
|
|
27
|
+
this.folderExpansionToggled = new EventEmitter();
|
|
28
|
+
/** Emitted after the host completes folder creation (e.g., inline rename flow). Reserved for future use. */
|
|
29
|
+
this.folderCreated = new EventEmitter();
|
|
30
|
+
/** Emitted when the user clicks the "+" header button. Host should open a creation modal. */
|
|
31
|
+
this.folderCreateRequested = new EventEmitter();
|
|
32
|
+
/** Carries the full renamed folder node + parent id so hosts can issue complete update requests. */
|
|
33
|
+
this.folderRenamed = new EventEmitter();
|
|
34
|
+
this.folderDeleted = new EventEmitter();
|
|
35
|
+
/** User picked "Move folder" from the context menu. Host opens the folder picker. */
|
|
36
|
+
this.folderMoveRequested = new EventEmitter();
|
|
37
|
+
/** User picked "Duplicate folder" from the context menu. Host clones the subtree (tests referenced, not copied). */
|
|
38
|
+
this.folderDuplicateRequested = new EventEmitter();
|
|
39
|
+
this.testsDropped = new EventEmitter();
|
|
40
|
+
/** Fires when a folder row is dropped onto another folder (or the Unorganised row). */
|
|
41
|
+
this.folderDropped = new EventEmitter();
|
|
42
|
+
this.collapsedChange = new EventEmitter();
|
|
43
|
+
this.searchValue = '';
|
|
44
|
+
this.renamingId = null;
|
|
45
|
+
this.renameDraft = '';
|
|
46
|
+
/** Id of the folder whose context menu is open, or null when closed. */
|
|
47
|
+
this.contextMenuFolderId = null;
|
|
48
|
+
/** Viewport-anchored position (client coordinates) for the floating menu. */
|
|
49
|
+
this.contextMenuPosition = { x: 0, y: 0 };
|
|
50
|
+
/** Pending auto-expand timers while a folder drag is hovered over collapsed rows. */
|
|
51
|
+
this.dragExpandTimers = new Map();
|
|
52
|
+
this.DRAG_EXPAND_DELAY_MS = 600;
|
|
53
|
+
/** Id of the folder currently being dragged; used to skip cycle targets and source-hover expansion. */
|
|
54
|
+
this.activeFolderDragId = null;
|
|
55
|
+
this.trackByRow = (_, row) => row.node.id;
|
|
56
|
+
}
|
|
57
|
+
get expandedSet() {
|
|
58
|
+
return new Set(this.expandedFolderIds || []);
|
|
59
|
+
}
|
|
60
|
+
/** Flattened, depth-aware list of rows to render in order. Ancestors of matches are kept. */
|
|
61
|
+
get rows() {
|
|
62
|
+
const search = (this.searchValue || '').trim().toLowerCase();
|
|
63
|
+
const expanded = this.expandedSet;
|
|
64
|
+
const keepSet = new Set();
|
|
65
|
+
if (search) {
|
|
66
|
+
const mark = (nodes, ancestors) => {
|
|
67
|
+
for (const n of nodes || []) {
|
|
68
|
+
const matched = (n.name || '').toLowerCase().includes(search);
|
|
69
|
+
const childAncestors = [...ancestors, n.id];
|
|
70
|
+
if (matched) {
|
|
71
|
+
for (const a of ancestors)
|
|
72
|
+
keepSet.add(a);
|
|
73
|
+
keepSet.add(n.id);
|
|
74
|
+
}
|
|
75
|
+
if (n.children?.length)
|
|
76
|
+
mark(n.children, childAncestors);
|
|
77
|
+
}
|
|
78
|
+
};
|
|
79
|
+
mark(this.folders, []);
|
|
80
|
+
}
|
|
81
|
+
const out = [];
|
|
82
|
+
const walk = (nodes, depth, parentVisible) => {
|
|
83
|
+
for (const n of nodes || []) {
|
|
84
|
+
const searchVisible = !search || keepSet.has(n.id);
|
|
85
|
+
const visible = parentVisible && searchVisible;
|
|
86
|
+
const hasChildren = !!(n.children && n.children.length);
|
|
87
|
+
out.push({ node: n, depth, visible, hasChildren });
|
|
88
|
+
const expandedBySearch = !!search && keepSet.has(n.id);
|
|
89
|
+
const isNodeExpanded = expanded.has(n.id) || expandedBySearch;
|
|
90
|
+
if (hasChildren && isNodeExpanded) {
|
|
91
|
+
walk(n.children, depth + 1, visible);
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
};
|
|
95
|
+
walk(this.folders, 0, true);
|
|
96
|
+
return out.filter(r => r.visible);
|
|
97
|
+
}
|
|
98
|
+
isExpanded(id) {
|
|
99
|
+
return this.expandedSet.has(id);
|
|
100
|
+
}
|
|
101
|
+
isSelected(id) {
|
|
102
|
+
if (id == null && this.selectedFolderId == null)
|
|
103
|
+
return true;
|
|
104
|
+
return id != null && this.selectedFolderId != null && id === this.selectedFolderId;
|
|
105
|
+
}
|
|
106
|
+
onToggle(n, event) {
|
|
107
|
+
event.stopPropagation();
|
|
108
|
+
const expanded = !this.isExpanded(n.id);
|
|
109
|
+
this.folderExpansionToggled.emit({ id: n.id, expanded });
|
|
110
|
+
}
|
|
111
|
+
onSelect(n) {
|
|
112
|
+
if (this.renamingId != null)
|
|
113
|
+
return;
|
|
114
|
+
this.folderSelected.emit(n == null ? null : n.id);
|
|
115
|
+
}
|
|
116
|
+
onSelectUnorganised() {
|
|
117
|
+
// Sentinel: hosts that keep an "Unorganised" bucket can treat a null selection
|
|
118
|
+
// as "show unorganised". If there is already a root (null) selection pattern,
|
|
119
|
+
// this matches it. Host decides semantics.
|
|
120
|
+
this.folderSelected.emit(null);
|
|
121
|
+
}
|
|
122
|
+
beginRename(n, event) {
|
|
123
|
+
if (!this.allowRename)
|
|
124
|
+
return;
|
|
125
|
+
event.stopPropagation();
|
|
126
|
+
this.renamingId = n.id;
|
|
127
|
+
this.renameDraft = n.name;
|
|
128
|
+
this.cdr.markForCheck();
|
|
129
|
+
this.focusRenameInput(n.id);
|
|
130
|
+
}
|
|
131
|
+
commitRename(n) {
|
|
132
|
+
const name = (this.renameDraft || '').trim();
|
|
133
|
+
if (name && name !== n.name) {
|
|
134
|
+
// Emit the full renamed node (with the new name) + parent context so hosts can
|
|
135
|
+
// call `PUT /test_case_folders/{id}` without re-resolving from their cached tree.
|
|
136
|
+
const folder = { ...n, name };
|
|
137
|
+
this.folderRenamed.emit({
|
|
138
|
+
folder,
|
|
139
|
+
parentId: this.parentIdOf(n.id),
|
|
140
|
+
previousName: n.name,
|
|
141
|
+
});
|
|
142
|
+
}
|
|
143
|
+
this.cancelRename();
|
|
144
|
+
}
|
|
145
|
+
cancelRename() {
|
|
146
|
+
this.renamingId = null;
|
|
147
|
+
this.renameDraft = '';
|
|
148
|
+
this.cdr.markForCheck();
|
|
149
|
+
}
|
|
150
|
+
onRenameKey(event, n) {
|
|
151
|
+
// Stop bubbling so the row's keydown handler (which treats Space/Enter/Arrows as
|
|
152
|
+
// selection / tree navigation) can't swallow characters typed into the rename input.
|
|
153
|
+
event.stopPropagation();
|
|
154
|
+
if (event.key === 'Enter') {
|
|
155
|
+
event.preventDefault();
|
|
156
|
+
this.commitRename(n);
|
|
157
|
+
}
|
|
158
|
+
else if (event.key === 'Escape') {
|
|
159
|
+
event.preventDefault();
|
|
160
|
+
this.cancelRename();
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
deleteFolder(n, event) {
|
|
164
|
+
if (!this.allowDelete)
|
|
165
|
+
return;
|
|
166
|
+
event.stopPropagation();
|
|
167
|
+
this.folderDeleted.emit({ id: n.id });
|
|
168
|
+
}
|
|
169
|
+
/**
|
|
170
|
+
* Opens the folder context menu anchored at the given client coordinates.
|
|
171
|
+
* Called from both (contextmenu) on the row and (click) on the ellipsis button.
|
|
172
|
+
* Menu visibility is clamped in the template via CSS transforms so it never leaves the viewport.
|
|
173
|
+
*/
|
|
174
|
+
openContextMenu(n, event) {
|
|
175
|
+
event.preventDefault();
|
|
176
|
+
event.stopPropagation();
|
|
177
|
+
if (!this.hasAnyContextAction)
|
|
178
|
+
return;
|
|
179
|
+
this.contextMenuFolderId = n.id;
|
|
180
|
+
this.contextMenuPosition = { x: event.clientX, y: event.clientY };
|
|
181
|
+
this.cdr.markForCheck();
|
|
182
|
+
}
|
|
183
|
+
closeContextMenu() {
|
|
184
|
+
if (this.contextMenuFolderId == null)
|
|
185
|
+
return;
|
|
186
|
+
this.contextMenuFolderId = null;
|
|
187
|
+
this.cdr.markForCheck();
|
|
188
|
+
}
|
|
189
|
+
/** True when at least one menu entry is enabled — otherwise the trigger is suppressed entirely. */
|
|
190
|
+
get hasAnyContextAction() {
|
|
191
|
+
return this.allowCreate || this.allowRename || this.allowMove || this.allowDuplicate || this.allowDelete;
|
|
192
|
+
}
|
|
193
|
+
onContextCreateSubfolder(n) {
|
|
194
|
+
this.closeContextMenu();
|
|
195
|
+
if (!this.allowCreate)
|
|
196
|
+
return;
|
|
197
|
+
this.folderCreateRequested.emit({ parentId: n.id });
|
|
198
|
+
}
|
|
199
|
+
onContextRename(n) {
|
|
200
|
+
this.closeContextMenu();
|
|
201
|
+
if (!this.allowRename)
|
|
202
|
+
return;
|
|
203
|
+
this.renamingId = n.id;
|
|
204
|
+
this.renameDraft = n.name;
|
|
205
|
+
this.cdr.markForCheck();
|
|
206
|
+
this.focusRenameInput(n.id);
|
|
207
|
+
}
|
|
208
|
+
onContextMove(n) {
|
|
209
|
+
this.closeContextMenu();
|
|
210
|
+
if (!this.allowMove)
|
|
211
|
+
return;
|
|
212
|
+
this.folderMoveRequested.emit({ id: n.id });
|
|
213
|
+
}
|
|
214
|
+
onContextDuplicate(n) {
|
|
215
|
+
this.closeContextMenu();
|
|
216
|
+
if (!this.allowDuplicate)
|
|
217
|
+
return;
|
|
218
|
+
this.folderDuplicateRequested.emit({ id: n.id });
|
|
219
|
+
}
|
|
220
|
+
onContextDelete(n) {
|
|
221
|
+
this.closeContextMenu();
|
|
222
|
+
if (!this.allowDelete)
|
|
223
|
+
return;
|
|
224
|
+
this.folderDeleted.emit({ id: n.id });
|
|
225
|
+
}
|
|
226
|
+
onDocumentClick() {
|
|
227
|
+
this.closeContextMenu();
|
|
228
|
+
}
|
|
229
|
+
onEscape() {
|
|
230
|
+
this.closeContextMenu();
|
|
231
|
+
}
|
|
232
|
+
/** Template helper: resolve a folder node by id from the input tree. */
|
|
233
|
+
folderById(id) {
|
|
234
|
+
if (id == null)
|
|
235
|
+
return null;
|
|
236
|
+
const find = (nodes) => {
|
|
237
|
+
for (const n of nodes || []) {
|
|
238
|
+
if (n.id === id)
|
|
239
|
+
return n;
|
|
240
|
+
if (n.children?.length) {
|
|
241
|
+
const hit = find(n.children);
|
|
242
|
+
if (hit)
|
|
243
|
+
return hit;
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
return null;
|
|
247
|
+
};
|
|
248
|
+
return find(this.folders);
|
|
249
|
+
}
|
|
250
|
+
/** Returns the parent id of `id`, or null when `id` is a root folder / not found. */
|
|
251
|
+
parentIdOf(id) {
|
|
252
|
+
let result = null;
|
|
253
|
+
const walk = (nodes, parentId) => {
|
|
254
|
+
for (const n of nodes || []) {
|
|
255
|
+
if (n.id === id) {
|
|
256
|
+
result = parentId;
|
|
257
|
+
return true;
|
|
258
|
+
}
|
|
259
|
+
if (n.children?.length && walk(n.children, n.id))
|
|
260
|
+
return true;
|
|
261
|
+
}
|
|
262
|
+
return false;
|
|
263
|
+
};
|
|
264
|
+
walk(this.folders, null);
|
|
265
|
+
return result;
|
|
266
|
+
}
|
|
267
|
+
requestCreate(parentId = null) {
|
|
268
|
+
if (!this.allowCreate)
|
|
269
|
+
return;
|
|
270
|
+
this.folderCreateRequested.emit({ parentId });
|
|
271
|
+
}
|
|
272
|
+
togglePanel() {
|
|
273
|
+
this.collapsed = !this.collapsed;
|
|
274
|
+
this.collapsedChange.emit(this.collapsed);
|
|
275
|
+
}
|
|
276
|
+
rowDropped(targetId, event) {
|
|
277
|
+
// Directive already emits with the bound folder id, but normalise in case callers
|
|
278
|
+
// wire raw events.
|
|
279
|
+
this.testsDropped.emit({ testIds: event.testIds, targetFolderId: targetId });
|
|
280
|
+
}
|
|
281
|
+
/**
|
|
282
|
+
* Drop target received a folder drag. Validate against self/descendant cycles and
|
|
283
|
+
* emit `folderDropped` if the move is legal. Called from the directive's
|
|
284
|
+
* `(folderDropped)` output on every row and on the Unorganised row.
|
|
285
|
+
*/
|
|
286
|
+
onFolderRowDropped(targetId, event) {
|
|
287
|
+
this.clearAllExpandTimers();
|
|
288
|
+
this.activeFolderDragId = null;
|
|
289
|
+
const sourceId = event.folderId;
|
|
290
|
+
if (targetId != null && targetId === sourceId)
|
|
291
|
+
return;
|
|
292
|
+
if (targetId != null && this.isDescendantOf(sourceId, targetId))
|
|
293
|
+
return;
|
|
294
|
+
this.folderDropped.emit({ id: sourceId, newParentId: targetId });
|
|
295
|
+
}
|
|
296
|
+
/** Remember the dragged folder so we can block self-hover expansion and cycle drops. */
|
|
297
|
+
onFolderDragStart(n) {
|
|
298
|
+
this.activeFolderDragId = n.id;
|
|
299
|
+
}
|
|
300
|
+
onFolderDragEnd() {
|
|
301
|
+
this.activeFolderDragId = null;
|
|
302
|
+
this.clearAllExpandTimers();
|
|
303
|
+
}
|
|
304
|
+
/** Schedule an auto-expand for a collapsed folder that has children during drag-hover. */
|
|
305
|
+
onFolderRowDragOver(row) {
|
|
306
|
+
const id = row.node.id;
|
|
307
|
+
if (!row.hasChildren)
|
|
308
|
+
return;
|
|
309
|
+
if (this.isExpanded(id))
|
|
310
|
+
return;
|
|
311
|
+
if (this.activeFolderDragId === id)
|
|
312
|
+
return; // don't expand the source
|
|
313
|
+
if (this.dragExpandTimers.has(id))
|
|
314
|
+
return;
|
|
315
|
+
const timer = setTimeout(() => {
|
|
316
|
+
this.dragExpandTimers.delete(id);
|
|
317
|
+
if (!this.isExpanded(id)) {
|
|
318
|
+
this.folderExpansionToggled.emit({ id, expanded: true });
|
|
319
|
+
}
|
|
320
|
+
}, this.DRAG_EXPAND_DELAY_MS);
|
|
321
|
+
this.dragExpandTimers.set(id, timer);
|
|
322
|
+
}
|
|
323
|
+
onFolderRowDragLeave(row) {
|
|
324
|
+
const id = row.node.id;
|
|
325
|
+
const timer = this.dragExpandTimers.get(id);
|
|
326
|
+
if (timer != null) {
|
|
327
|
+
clearTimeout(timer);
|
|
328
|
+
this.dragExpandTimers.delete(id);
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
onDocumentDragEnd() {
|
|
332
|
+
this.onFolderDragEnd();
|
|
333
|
+
}
|
|
334
|
+
/** True when `candidateId` is `sourceId` itself or any descendant of it. */
|
|
335
|
+
isDescendantOf(sourceId, candidateId) {
|
|
336
|
+
const source = this.folderById(sourceId);
|
|
337
|
+
if (!source)
|
|
338
|
+
return false;
|
|
339
|
+
const stack = [source];
|
|
340
|
+
while (stack.length) {
|
|
341
|
+
const n = stack.pop();
|
|
342
|
+
if (n.id === candidateId)
|
|
343
|
+
return true;
|
|
344
|
+
if (n.children?.length)
|
|
345
|
+
stack.push(...n.children);
|
|
346
|
+
}
|
|
347
|
+
return false;
|
|
348
|
+
}
|
|
349
|
+
clearAllExpandTimers() {
|
|
350
|
+
this.dragExpandTimers.forEach(t => clearTimeout(t));
|
|
351
|
+
this.dragExpandTimers.clear();
|
|
352
|
+
}
|
|
353
|
+
ngOnDestroy() {
|
|
354
|
+
this.clearAllExpandTimers();
|
|
355
|
+
}
|
|
356
|
+
onRowKeydown(event, row, index) {
|
|
357
|
+
const rows = this.rows;
|
|
358
|
+
switch (event.key) {
|
|
359
|
+
case 'ArrowDown': {
|
|
360
|
+
event.preventDefault();
|
|
361
|
+
const next = rows[index + 1];
|
|
362
|
+
if (next)
|
|
363
|
+
this.focusRowElement(next.node.id);
|
|
364
|
+
break;
|
|
365
|
+
}
|
|
366
|
+
case 'ArrowUp': {
|
|
367
|
+
event.preventDefault();
|
|
368
|
+
const prev = rows[index - 1];
|
|
369
|
+
if (prev)
|
|
370
|
+
this.focusRowElement(prev.node.id);
|
|
371
|
+
break;
|
|
372
|
+
}
|
|
373
|
+
case 'ArrowRight': {
|
|
374
|
+
if (row.hasChildren && !this.isExpanded(row.node.id)) {
|
|
375
|
+
event.preventDefault();
|
|
376
|
+
this.folderExpansionToggled.emit({ id: row.node.id, expanded: true });
|
|
377
|
+
}
|
|
378
|
+
break;
|
|
379
|
+
}
|
|
380
|
+
case 'ArrowLeft': {
|
|
381
|
+
if (row.hasChildren && this.isExpanded(row.node.id)) {
|
|
382
|
+
event.preventDefault();
|
|
383
|
+
this.folderExpansionToggled.emit({ id: row.node.id, expanded: false });
|
|
384
|
+
}
|
|
385
|
+
break;
|
|
386
|
+
}
|
|
387
|
+
case 'Enter':
|
|
388
|
+
case ' ': {
|
|
389
|
+
event.preventDefault();
|
|
390
|
+
this.onSelect(row.node);
|
|
391
|
+
break;
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
}
|
|
395
|
+
focusRowElement(id) {
|
|
396
|
+
queueMicrotask(() => {
|
|
397
|
+
const el = document.querySelector(`[data-folder-row-id="${id}"]`);
|
|
398
|
+
el?.focus();
|
|
399
|
+
});
|
|
400
|
+
}
|
|
401
|
+
/**
|
|
402
|
+
* Focuses the inline rename input for the given folder. Uses `setTimeout` (not
|
|
403
|
+
* microtask) because the input is gated by `*ngIf="renamingId === ..."` and
|
|
404
|
+
* must survive the next OnPush change-detection cycle before it's in the DOM.
|
|
405
|
+
*/
|
|
406
|
+
focusRenameInput(id) {
|
|
407
|
+
setTimeout(() => {
|
|
408
|
+
const el = document.querySelector(`[data-folder-rename-input="${id}"]`);
|
|
409
|
+
if (!el)
|
|
410
|
+
return;
|
|
411
|
+
el.focus();
|
|
412
|
+
el.select();
|
|
413
|
+
}, 0);
|
|
414
|
+
}
|
|
415
|
+
}
|
|
416
|
+
FolderSidebarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: FolderSidebarComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
417
|
+
FolderSidebarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: FolderSidebarComponent, selector: "cqa-folder-sidebar", inputs: { folders: "folders", selectedFolderId: "selectedFolderId", expandedFolderIds: "expandedFolderIds", unorganisedCount: "unorganisedCount", allowCreate: "allowCreate", allowRename: "allowRename", allowDelete: "allowDelete", allowMove: "allowMove", allowDuplicate: "allowDuplicate", allowDrop: "allowDrop", showCounts: "showCounts", collapsed: "collapsed", labels: "labels" }, outputs: { folderSelected: "folderSelected", folderExpansionToggled: "folderExpansionToggled", folderCreated: "folderCreated", folderCreateRequested: "folderCreateRequested", folderRenamed: "folderRenamed", folderDeleted: "folderDeleted", folderMoveRequested: "folderMoveRequested", folderDuplicateRequested: "folderDuplicateRequested", testsDropped: "testsDropped", folderDropped: "folderDropped", collapsedChange: "collapsedChange" }, host: { listeners: { "document:click": "onDocumentClick()", "document:keydown.escape": "onEscape()", "document:dragend": "onDocumentDragEnd()" }, classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<!-- Reusable folder icon. Render via <ng-container *ngTemplateOutlet=\"folderIcon; context: { color: node.color }\"></ng-container>. -->\n<ng-template #folderIcon let-color=\"color\">\n <span class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-4 cqa-h-4 cqa-shrink-0\">\n <svg width=\"13\" height=\"12\" viewBox=\"0 0 13 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path d=\"M12.1916 9.85824C12.1916 10.1677 12.0687 10.4644 11.8499 10.6832C11.6311 10.902 11.3343 11.0249 11.0249 11.0249H1.69157C1.38215 11.0249 1.0854 10.902 0.866611 10.6832C0.647819 10.4644 0.524902 10.1677 0.524902 9.85824V1.69157C0.524902 1.38215 0.647819 1.0854 0.866611 0.866611C1.0854 0.647819 1.38215 0.524902 1.69157 0.524902H4.60824L5.7749 2.2749H11.0249C11.3343 2.2749 11.6311 2.39782 11.8499 2.61661C12.0687 2.8354 12.1916 3.13215 12.1916 3.44157V9.85824Z\" [attr.stroke]=\"color || '#99999E'\" stroke-width=\"1.05\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </svg>\n </span>\n</ng-template>\n\n<aside\n class=\"cqa-flex cqa-flex-col cqa-bg-white cqa-border cqa-border-neutral-200 cqa-rounded-lg cqa-h-full cqa-min-h-0\"\n [class.cqa-w-[240px]]=\"!collapsed\"\n [class.cqa-w-[48px]]=\"collapsed\"\n style=\"transition: width 150ms ease;\"\n>\n <!-- Header -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-px-3 cqa-py-3\">\n <span *ngIf=\"!collapsed\" class=\"cqa-text-sm cqa-font-semibold cqa-text-neutral-900\">\n {{ labels.folders }}\n </span>\n <div class=\"cqa-flex cqa-items-center cqa-gap-1\" [class.cqa-w-full]=\"collapsed\" [class.cqa-justify-center]=\"collapsed\">\n <button\n type=\"button\"\n class=\"cqa-p-1 cqa-rounded hover:cqa-bg-neutral-100 cqa-text-neutral-500\"\n (click)=\"togglePanel()\"\n [attr.aria-label]=\"collapsed ? 'Expand folders panel' : 'Collapse folders panel'\"\n >\n <mat-icon style=\"font-size:18px;width:18px;height:18px\">\n {{ collapsed ? 'chevron_right' : 'keyboard_double_arrow_left' }}\n </mat-icon>\n </button>\n <button\n *ngIf=\"!collapsed && allowCreate\"\n type=\"button\"\n class=\"cqa-p-1 cqa-rounded hover:cqa-bg-neutral-100 cqa-text-neutral-500\"\n (click)=\"requestCreate(null)\"\n [attr.aria-label]=\"labels.newFolder\"\n [title]=\"labels.newFolder\"\n >\n <mat-icon style=\"font-size:18px;width:18px;height:18px\">add</mat-icon>\n </button>\n </div>\n </div>\n\n <ng-container *ngIf=\"!collapsed\">\n <!-- Search -->\n <div class=\"cqa-px-3 cqa-pb-2\">\n <cqa-search-bar\n size=\"sm\"\n [fullWidth]=\"true\"\n [value]=\"searchValue\"\n [placeholder]=\"labels.searchFoldersPlaceholder\"\n [showClear]=\"true\"\n (valueChange)=\"searchValue = $event\"\n (cleared)=\"searchValue = ''\"\n ></cqa-search-bar>\n </div>\n\n <!-- Tree (folders + Unorganised tab) -->\n <div role=\"tree\" class=\"cqa-flex-1 cqa-overflow-y-auto cqa-py-1\">\n <ng-container *ngFor=\"let row of rows; let i = index; trackBy: trackByRow\">\n <div\n role=\"treeitem\"\n tabindex=\"0\"\n [attr.aria-expanded]=\"row.hasChildren ? isExpanded(row.node.id) : null\"\n [attr.aria-selected]=\"isSelected(row.node.id)\"\n [attr.data-folder-row-id]=\"row.node.id\"\n [cqaFolderDrop]=\"row.node.id\"\n [dropEnabled]=\"allowDrop\"\n (testsDropped)=\"rowDropped(row.node.id, $event)\"\n (folderDropped)=\"onFolderRowDropped(row.node.id, $event)\"\n [cqaFolderDrag]=\"row.node.id\"\n [dragEnabled]=\"allowDrop\"\n (dragstart)=\"onFolderDragStart(row.node)\"\n (dragend)=\"onFolderDragEnd()\"\n (dragover)=\"onFolderRowDragOver(row)\"\n (dragleave)=\"onFolderRowDragLeave(row)\"\n (click)=\"onSelect(row.node)\"\n (contextmenu)=\"openContextMenu(row.node, $event)\"\n (keydown)=\"onRowKeydown($event, row, i)\"\n class=\"cqa-group cqa-flex cqa-items-center cqa-gap-1 cqa-pr-3 cqa-py-1.5 cqa-cursor-pointer hover:cqa-bg-neutral-50 focus:cqa-outline-none focus:cqa-bg-neutral-100 cqa-relative\"\n [class.cqa-bg-indigo-50]=\"isSelected(row.node.id) || contextMenuFolderId === row.node.id\"\n [style.paddingLeft.px]=\"8 + row.depth * 16\"\n [attr.title]=\"row.node.totalCount != null ? (row.node.totalCount + ' total including subfolders') : null\"\n >\n <span\n *ngIf=\"isSelected(row.node.id)\"\n class=\"cqa-absolute cqa-left-0 cqa-top-0 cqa-bottom-0 cqa-w-[3px] cqa-bg-indigo-600\"\n ></span>\n <button\n type=\"button\"\n class=\"cqa-p-0 cqa-text-neutral-400\"\n [style.visibility]=\"row.hasChildren ? 'visible' : 'hidden'\"\n (click)=\"onToggle(row.node, $event)\"\n [attr.aria-label]=\"isExpanded(row.node.id) ? 'Collapse' : 'Expand'\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">\n {{ isExpanded(row.node.id) ? 'expand_more' : 'chevron_right' }}\n </mat-icon>\n </button>\n <ng-container *ngTemplateOutlet=\"folderIcon; context: { color: row.node.color }\"></ng-container>\n\n <ng-container *ngIf=\"renamingId === row.node.id; else nameTpl\">\n <input\n type=\"text\"\n size=\"1\"\n [attr.data-folder-rename-input]=\"row.node.id\"\n class=\"cqa-flex-1 cqa-min-w-0 cqa-w-full cqa-px-2 cqa-py-0.5 cqa-rounded cqa-border cqa-border-neutral-300 cqa-bg-white cqa-text-sm cqa-shadow-sm focus:cqa-outline-none focus:cqa-border-indigo-400 focus:cqa-ring-1 focus:cqa-ring-indigo-200\"\n [(ngModel)]=\"renameDraft\"\n (keydown)=\"onRenameKey($event, row.node)\"\n (keypress)=\"$event.stopPropagation()\"\n (keyup)=\"$event.stopPropagation()\"\n (blur)=\"commitRename(row.node)\"\n (click)=\"$event.stopPropagation()\"\n />\n </ng-container>\n <ng-template #nameTpl>\n <span\n class=\"cqa-flex-1 cqa-text-sm cqa-text-neutral-800 cqa-truncate\"\n (dblclick)=\"beginRename(row.node, $event)\"\n >{{ row.node.name }}</span>\n </ng-template>\n\n <!-- Count shown at rest; hidden on row hover when delete is allowed -->\n <span\n *ngIf=\"showCounts && row.node.count != null\"\n class=\"cqa-text-xs cqa-text-neutral-400 cqa-tabular-nums cqa-ml-1\"\n [class.group-hover:cqa-hidden]=\"allowDelete\"\n >{{ row.node.count }}</span>\n\n <button\n *ngIf=\"hasAnyContextAction\"\n type=\"button\"\n class=\"cqa-p-0.5 cqa-rounded hover:cqa-bg-neutral-200 cqa-text-neutral-500\"\n [class.cqa-hidden]=\"contextMenuFolderId !== row.node.id\"\n [class.group-hover:cqa-inline-flex]=\"true\"\n [class.cqa-inline-flex]=\"contextMenuFolderId === row.node.id\"\n (click)=\"openContextMenu(row.node, $event)\"\n [attr.aria-label]=\"'Open actions for ' + row.node.name\"\n [attr.aria-haspopup]=\"'menu'\"\n [attr.aria-expanded]=\"contextMenuFolderId === row.node.id\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">more_horiz</mat-icon>\n </button>\n </div>\n </ng-container>\n\n <!-- Divider between folder tree and Unorganised tab -->\n <div\n aria-hidden=\"true\"\n class=\"cqa-mx-3\"\n style=\"height: 1px; background-color: #E5E7EB; margin-top: 6px; margin-bottom: 6px;\"\n ></div>\n\n <!-- Unorganised \u2014 flows with the tree, behaves like a tab item -->\n <div\n role=\"treeitem\"\n tabindex=\"0\"\n [attr.aria-selected]=\"isSelected(null)\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-px-3 cqa-py-1.5 cqa-cursor-pointer hover:cqa-bg-neutral-50 focus:cqa-outline-none focus:cqa-bg-neutral-100 cqa-relative\"\n [class.cqa-bg-indigo-50]=\"isSelected(null)\"\n [cqaFolderDrop]=\"null\"\n [dropEnabled]=\"allowDrop\"\n (testsDropped)=\"rowDropped(null, $event)\"\n (folderDropped)=\"onFolderRowDropped(null, $event)\"\n (click)=\"onSelectUnorganised()\"\n >\n <span\n *ngIf=\"isSelected(null)\"\n class=\"cqa-absolute cqa-left-0 cqa-top-0 cqa-bottom-0 cqa-w-[3px] cqa-bg-indigo-600\"\n ></span>\n <mat-icon class=\"cqa-text-neutral-500\" style=\"font-size:16px;width:16px;height:16px\">inbox</mat-icon>\n <span class=\"cqa-flex-1 cqa-text-sm cqa-text-neutral-700\">{{ labels.unorganised }}</span>\n <span *ngIf=\"showCounts\" class=\"cqa-text-xs cqa-text-neutral-400 cqa-tabular-nums\">{{ unorganisedCount }}</span>\n </div>\n </div>\n\n <!-- Folder context menu (right-click / ellipsis). Anchored to viewport coords. -->\n <div\n *ngIf=\"contextMenuFolderId !== null\"\n role=\"menu\"\n class=\"cqa-fixed cqa-z-50 cqa-min-w-[180px] cqa-bg-white cqa-border cqa-border-neutral-200 cqa-rounded-md cqa-shadow-lg cqa-py-1\"\n [style.left.px]=\"contextMenuPosition.x\"\n [style.top.px]=\"contextMenuPosition.y\"\n (click)=\"$event.stopPropagation()\"\n (contextmenu)=\"$event.preventDefault(); $event.stopPropagation()\"\n >\n <ng-container *ngIf=\"folderById(contextMenuFolderId) as menuNode\">\n <button\n *ngIf=\"allowCreate\"\n type=\"button\"\n role=\"menuitem\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-neutral-800 hover:cqa-bg-neutral-100 cqa-text-left\"\n (click)=\"onContextCreateSubfolder(menuNode)\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">create_new_folder</mat-icon>\n <span>{{ labels.folderMenuCreateSubfolder }}</span>\n </button>\n <button\n *ngIf=\"allowRename\"\n type=\"button\"\n role=\"menuitem\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-neutral-800 hover:cqa-bg-neutral-100 cqa-text-left\"\n (click)=\"onContextRename(menuNode)\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">edit</mat-icon>\n <span>{{ labels.folderMenuRename }}</span>\n </button>\n <button\n *ngIf=\"allowMove\"\n type=\"button\"\n role=\"menuitem\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-neutral-800 hover:cqa-bg-neutral-100 cqa-text-left\"\n (click)=\"onContextMove(menuNode)\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">drive_file_move</mat-icon>\n <span>{{ labels.folderMenuMove }}</span>\n </button>\n <button\n *ngIf=\"allowDuplicate\"\n type=\"button\"\n role=\"menuitem\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-neutral-800 hover:cqa-bg-neutral-100 cqa-text-left\"\n (click)=\"onContextDuplicate(menuNode)\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">content_copy</mat-icon>\n <span>{{ labels.folderMenuDuplicate }}</span>\n </button>\n <div *ngIf=\"allowDelete && (allowCreate || allowRename || allowMove || allowDuplicate)\" class=\"cqa-h-px cqa-bg-neutral-200 cqa-my-1\"></div>\n <button\n *ngIf=\"allowDelete\"\n type=\"button\"\n role=\"menuitem\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-red-600 hover:cqa-bg-red-50 cqa-text-left\"\n (click)=\"onContextDelete(menuNode)\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">delete_outline</mat-icon>\n <span>{{ labels.folderMenuDelete }}</span>\n </button>\n </ng-container>\n </div>\n\n </ng-container>\n</aside>\n", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i2.SearchBarComponent, selector: "cqa-search-bar", inputs: ["placeholder", "value", "disabled", "showClear", "ariaLabel", "autoFocus", "size", "fullWidth"], outputs: ["valueChange", "search", "cleared"] }], directives: [{ type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i4.FolderDropDirective, selector: "[cqaFolderDrop]", inputs: ["cqaFolderDrop", "dropEnabled"], outputs: ["testsDropped", "folderDropped"] }, { type: i5.FolderDragDirective, selector: "[cqaFolderDrag]", inputs: ["cqaFolderDrag", "dragEnabled"] }, { type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i6.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i6.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i6.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
418
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: FolderSidebarComponent, decorators: [{
|
|
419
|
+
type: Component,
|
|
420
|
+
args: [{ selector: 'cqa-folder-sidebar', host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "<!-- Reusable folder icon. Render via <ng-container *ngTemplateOutlet=\"folderIcon; context: { color: node.color }\"></ng-container>. -->\n<ng-template #folderIcon let-color=\"color\">\n <span class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-4 cqa-h-4 cqa-shrink-0\">\n <svg width=\"13\" height=\"12\" viewBox=\"0 0 13 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path d=\"M12.1916 9.85824C12.1916 10.1677 12.0687 10.4644 11.8499 10.6832C11.6311 10.902 11.3343 11.0249 11.0249 11.0249H1.69157C1.38215 11.0249 1.0854 10.902 0.866611 10.6832C0.647819 10.4644 0.524902 10.1677 0.524902 9.85824V1.69157C0.524902 1.38215 0.647819 1.0854 0.866611 0.866611C1.0854 0.647819 1.38215 0.524902 1.69157 0.524902H4.60824L5.7749 2.2749H11.0249C11.3343 2.2749 11.6311 2.39782 11.8499 2.61661C12.0687 2.8354 12.1916 3.13215 12.1916 3.44157V9.85824Z\" [attr.stroke]=\"color || '#99999E'\" stroke-width=\"1.05\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </svg>\n </span>\n</ng-template>\n\n<aside\n class=\"cqa-flex cqa-flex-col cqa-bg-white cqa-border cqa-border-neutral-200 cqa-rounded-lg cqa-h-full cqa-min-h-0\"\n [class.cqa-w-[240px]]=\"!collapsed\"\n [class.cqa-w-[48px]]=\"collapsed\"\n style=\"transition: width 150ms ease;\"\n>\n <!-- Header -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-px-3 cqa-py-3\">\n <span *ngIf=\"!collapsed\" class=\"cqa-text-sm cqa-font-semibold cqa-text-neutral-900\">\n {{ labels.folders }}\n </span>\n <div class=\"cqa-flex cqa-items-center cqa-gap-1\" [class.cqa-w-full]=\"collapsed\" [class.cqa-justify-center]=\"collapsed\">\n <button\n type=\"button\"\n class=\"cqa-p-1 cqa-rounded hover:cqa-bg-neutral-100 cqa-text-neutral-500\"\n (click)=\"togglePanel()\"\n [attr.aria-label]=\"collapsed ? 'Expand folders panel' : 'Collapse folders panel'\"\n >\n <mat-icon style=\"font-size:18px;width:18px;height:18px\">\n {{ collapsed ? 'chevron_right' : 'keyboard_double_arrow_left' }}\n </mat-icon>\n </button>\n <button\n *ngIf=\"!collapsed && allowCreate\"\n type=\"button\"\n class=\"cqa-p-1 cqa-rounded hover:cqa-bg-neutral-100 cqa-text-neutral-500\"\n (click)=\"requestCreate(null)\"\n [attr.aria-label]=\"labels.newFolder\"\n [title]=\"labels.newFolder\"\n >\n <mat-icon style=\"font-size:18px;width:18px;height:18px\">add</mat-icon>\n </button>\n </div>\n </div>\n\n <ng-container *ngIf=\"!collapsed\">\n <!-- Search -->\n <div class=\"cqa-px-3 cqa-pb-2\">\n <cqa-search-bar\n size=\"sm\"\n [fullWidth]=\"true\"\n [value]=\"searchValue\"\n [placeholder]=\"labels.searchFoldersPlaceholder\"\n [showClear]=\"true\"\n (valueChange)=\"searchValue = $event\"\n (cleared)=\"searchValue = ''\"\n ></cqa-search-bar>\n </div>\n\n <!-- Tree (folders + Unorganised tab) -->\n <div role=\"tree\" class=\"cqa-flex-1 cqa-overflow-y-auto cqa-py-1\">\n <ng-container *ngFor=\"let row of rows; let i = index; trackBy: trackByRow\">\n <div\n role=\"treeitem\"\n tabindex=\"0\"\n [attr.aria-expanded]=\"row.hasChildren ? isExpanded(row.node.id) : null\"\n [attr.aria-selected]=\"isSelected(row.node.id)\"\n [attr.data-folder-row-id]=\"row.node.id\"\n [cqaFolderDrop]=\"row.node.id\"\n [dropEnabled]=\"allowDrop\"\n (testsDropped)=\"rowDropped(row.node.id, $event)\"\n (folderDropped)=\"onFolderRowDropped(row.node.id, $event)\"\n [cqaFolderDrag]=\"row.node.id\"\n [dragEnabled]=\"allowDrop\"\n (dragstart)=\"onFolderDragStart(row.node)\"\n (dragend)=\"onFolderDragEnd()\"\n (dragover)=\"onFolderRowDragOver(row)\"\n (dragleave)=\"onFolderRowDragLeave(row)\"\n (click)=\"onSelect(row.node)\"\n (contextmenu)=\"openContextMenu(row.node, $event)\"\n (keydown)=\"onRowKeydown($event, row, i)\"\n class=\"cqa-group cqa-flex cqa-items-center cqa-gap-1 cqa-pr-3 cqa-py-1.5 cqa-cursor-pointer hover:cqa-bg-neutral-50 focus:cqa-outline-none focus:cqa-bg-neutral-100 cqa-relative\"\n [class.cqa-bg-indigo-50]=\"isSelected(row.node.id) || contextMenuFolderId === row.node.id\"\n [style.paddingLeft.px]=\"8 + row.depth * 16\"\n [attr.title]=\"row.node.totalCount != null ? (row.node.totalCount + ' total including subfolders') : null\"\n >\n <span\n *ngIf=\"isSelected(row.node.id)\"\n class=\"cqa-absolute cqa-left-0 cqa-top-0 cqa-bottom-0 cqa-w-[3px] cqa-bg-indigo-600\"\n ></span>\n <button\n type=\"button\"\n class=\"cqa-p-0 cqa-text-neutral-400\"\n [style.visibility]=\"row.hasChildren ? 'visible' : 'hidden'\"\n (click)=\"onToggle(row.node, $event)\"\n [attr.aria-label]=\"isExpanded(row.node.id) ? 'Collapse' : 'Expand'\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">\n {{ isExpanded(row.node.id) ? 'expand_more' : 'chevron_right' }}\n </mat-icon>\n </button>\n <ng-container *ngTemplateOutlet=\"folderIcon; context: { color: row.node.color }\"></ng-container>\n\n <ng-container *ngIf=\"renamingId === row.node.id; else nameTpl\">\n <input\n type=\"text\"\n size=\"1\"\n [attr.data-folder-rename-input]=\"row.node.id\"\n class=\"cqa-flex-1 cqa-min-w-0 cqa-w-full cqa-px-2 cqa-py-0.5 cqa-rounded cqa-border cqa-border-neutral-300 cqa-bg-white cqa-text-sm cqa-shadow-sm focus:cqa-outline-none focus:cqa-border-indigo-400 focus:cqa-ring-1 focus:cqa-ring-indigo-200\"\n [(ngModel)]=\"renameDraft\"\n (keydown)=\"onRenameKey($event, row.node)\"\n (keypress)=\"$event.stopPropagation()\"\n (keyup)=\"$event.stopPropagation()\"\n (blur)=\"commitRename(row.node)\"\n (click)=\"$event.stopPropagation()\"\n />\n </ng-container>\n <ng-template #nameTpl>\n <span\n class=\"cqa-flex-1 cqa-text-sm cqa-text-neutral-800 cqa-truncate\"\n (dblclick)=\"beginRename(row.node, $event)\"\n >{{ row.node.name }}</span>\n </ng-template>\n\n <!-- Count shown at rest; hidden on row hover when delete is allowed -->\n <span\n *ngIf=\"showCounts && row.node.count != null\"\n class=\"cqa-text-xs cqa-text-neutral-400 cqa-tabular-nums cqa-ml-1\"\n [class.group-hover:cqa-hidden]=\"allowDelete\"\n >{{ row.node.count }}</span>\n\n <button\n *ngIf=\"hasAnyContextAction\"\n type=\"button\"\n class=\"cqa-p-0.5 cqa-rounded hover:cqa-bg-neutral-200 cqa-text-neutral-500\"\n [class.cqa-hidden]=\"contextMenuFolderId !== row.node.id\"\n [class.group-hover:cqa-inline-flex]=\"true\"\n [class.cqa-inline-flex]=\"contextMenuFolderId === row.node.id\"\n (click)=\"openContextMenu(row.node, $event)\"\n [attr.aria-label]=\"'Open actions for ' + row.node.name\"\n [attr.aria-haspopup]=\"'menu'\"\n [attr.aria-expanded]=\"contextMenuFolderId === row.node.id\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">more_horiz</mat-icon>\n </button>\n </div>\n </ng-container>\n\n <!-- Divider between folder tree and Unorganised tab -->\n <div\n aria-hidden=\"true\"\n class=\"cqa-mx-3\"\n style=\"height: 1px; background-color: #E5E7EB; margin-top: 6px; margin-bottom: 6px;\"\n ></div>\n\n <!-- Unorganised \u2014 flows with the tree, behaves like a tab item -->\n <div\n role=\"treeitem\"\n tabindex=\"0\"\n [attr.aria-selected]=\"isSelected(null)\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-px-3 cqa-py-1.5 cqa-cursor-pointer hover:cqa-bg-neutral-50 focus:cqa-outline-none focus:cqa-bg-neutral-100 cqa-relative\"\n [class.cqa-bg-indigo-50]=\"isSelected(null)\"\n [cqaFolderDrop]=\"null\"\n [dropEnabled]=\"allowDrop\"\n (testsDropped)=\"rowDropped(null, $event)\"\n (folderDropped)=\"onFolderRowDropped(null, $event)\"\n (click)=\"onSelectUnorganised()\"\n >\n <span\n *ngIf=\"isSelected(null)\"\n class=\"cqa-absolute cqa-left-0 cqa-top-0 cqa-bottom-0 cqa-w-[3px] cqa-bg-indigo-600\"\n ></span>\n <mat-icon class=\"cqa-text-neutral-500\" style=\"font-size:16px;width:16px;height:16px\">inbox</mat-icon>\n <span class=\"cqa-flex-1 cqa-text-sm cqa-text-neutral-700\">{{ labels.unorganised }}</span>\n <span *ngIf=\"showCounts\" class=\"cqa-text-xs cqa-text-neutral-400 cqa-tabular-nums\">{{ unorganisedCount }}</span>\n </div>\n </div>\n\n <!-- Folder context menu (right-click / ellipsis). Anchored to viewport coords. -->\n <div\n *ngIf=\"contextMenuFolderId !== null\"\n role=\"menu\"\n class=\"cqa-fixed cqa-z-50 cqa-min-w-[180px] cqa-bg-white cqa-border cqa-border-neutral-200 cqa-rounded-md cqa-shadow-lg cqa-py-1\"\n [style.left.px]=\"contextMenuPosition.x\"\n [style.top.px]=\"contextMenuPosition.y\"\n (click)=\"$event.stopPropagation()\"\n (contextmenu)=\"$event.preventDefault(); $event.stopPropagation()\"\n >\n <ng-container *ngIf=\"folderById(contextMenuFolderId) as menuNode\">\n <button\n *ngIf=\"allowCreate\"\n type=\"button\"\n role=\"menuitem\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-neutral-800 hover:cqa-bg-neutral-100 cqa-text-left\"\n (click)=\"onContextCreateSubfolder(menuNode)\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">create_new_folder</mat-icon>\n <span>{{ labels.folderMenuCreateSubfolder }}</span>\n </button>\n <button\n *ngIf=\"allowRename\"\n type=\"button\"\n role=\"menuitem\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-neutral-800 hover:cqa-bg-neutral-100 cqa-text-left\"\n (click)=\"onContextRename(menuNode)\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">edit</mat-icon>\n <span>{{ labels.folderMenuRename }}</span>\n </button>\n <button\n *ngIf=\"allowMove\"\n type=\"button\"\n role=\"menuitem\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-neutral-800 hover:cqa-bg-neutral-100 cqa-text-left\"\n (click)=\"onContextMove(menuNode)\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">drive_file_move</mat-icon>\n <span>{{ labels.folderMenuMove }}</span>\n </button>\n <button\n *ngIf=\"allowDuplicate\"\n type=\"button\"\n role=\"menuitem\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-neutral-800 hover:cqa-bg-neutral-100 cqa-text-left\"\n (click)=\"onContextDuplicate(menuNode)\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">content_copy</mat-icon>\n <span>{{ labels.folderMenuDuplicate }}</span>\n </button>\n <div *ngIf=\"allowDelete && (allowCreate || allowRename || allowMove || allowDuplicate)\" class=\"cqa-h-px cqa-bg-neutral-200 cqa-my-1\"></div>\n <button\n *ngIf=\"allowDelete\"\n type=\"button\"\n role=\"menuitem\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-red-600 hover:cqa-bg-red-50 cqa-text-left\"\n (click)=\"onContextDelete(menuNode)\"\n >\n <mat-icon style=\"font-size:16px;width:16px;height:16px\">delete_outline</mat-icon>\n <span>{{ labels.folderMenuDelete }}</span>\n </button>\n </ng-container>\n </div>\n\n </ng-container>\n</aside>\n", styles: [] }]
|
|
421
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { folders: [{
|
|
422
|
+
type: Input
|
|
423
|
+
}], selectedFolderId: [{
|
|
424
|
+
type: Input
|
|
425
|
+
}], expandedFolderIds: [{
|
|
426
|
+
type: Input
|
|
427
|
+
}], unorganisedCount: [{
|
|
428
|
+
type: Input
|
|
429
|
+
}], allowCreate: [{
|
|
430
|
+
type: Input
|
|
431
|
+
}], allowRename: [{
|
|
432
|
+
type: Input
|
|
433
|
+
}], allowDelete: [{
|
|
434
|
+
type: Input
|
|
435
|
+
}], allowMove: [{
|
|
436
|
+
type: Input
|
|
437
|
+
}], allowDuplicate: [{
|
|
438
|
+
type: Input
|
|
439
|
+
}], allowDrop: [{
|
|
440
|
+
type: Input
|
|
441
|
+
}], showCounts: [{
|
|
442
|
+
type: Input
|
|
443
|
+
}], collapsed: [{
|
|
444
|
+
type: Input
|
|
445
|
+
}], labels: [{
|
|
446
|
+
type: Input
|
|
447
|
+
}], folderSelected: [{
|
|
448
|
+
type: Output
|
|
449
|
+
}], folderExpansionToggled: [{
|
|
450
|
+
type: Output
|
|
451
|
+
}], folderCreated: [{
|
|
452
|
+
type: Output
|
|
453
|
+
}], folderCreateRequested: [{
|
|
454
|
+
type: Output
|
|
455
|
+
}], folderRenamed: [{
|
|
456
|
+
type: Output
|
|
457
|
+
}], folderDeleted: [{
|
|
458
|
+
type: Output
|
|
459
|
+
}], folderMoveRequested: [{
|
|
460
|
+
type: Output
|
|
461
|
+
}], folderDuplicateRequested: [{
|
|
462
|
+
type: Output
|
|
463
|
+
}], testsDropped: [{
|
|
464
|
+
type: Output
|
|
465
|
+
}], folderDropped: [{
|
|
466
|
+
type: Output
|
|
467
|
+
}], collapsedChange: [{
|
|
468
|
+
type: Output
|
|
469
|
+
}], onDocumentClick: [{
|
|
470
|
+
type: HostListener,
|
|
471
|
+
args: ['document:click']
|
|
472
|
+
}], onEscape: [{
|
|
473
|
+
type: HostListener,
|
|
474
|
+
args: ['document:keydown.escape']
|
|
475
|
+
}], onDocumentDragEnd: [{
|
|
476
|
+
type: HostListener,
|
|
477
|
+
args: ['document:dragend']
|
|
478
|
+
}] } });
|
|
479
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"folder-sidebar.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/templates/modular-table-template/folder-sidebar/folder-sidebar.component.ts","../../../../../../../src/lib/templates/modular-table-template/folder-sidebar/folder-sidebar.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  HostListener,\n  Input,\n  OnDestroy,\n  Output,\n} from '@angular/core';\nimport {\n  DEFAULT_MODULAR_LABELS,\n  FolderNode,\n  FolderRenamedPayload,\n  ModularLabels,\n} from '../modular-table-template.models';\n\ninterface RenderNode {\n  node: FolderNode;\n  depth: number;\n  visible: boolean;\n  hasChildren: boolean;\n}\n\n@Component({\n  selector: 'cqa-folder-sidebar',\n  templateUrl: './folder-sidebar.component.html',\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class FolderSidebarComponent implements OnDestroy {\n  @Input() folders: FolderNode[] = [];\n  @Input() selectedFolderId: number | null = null;\n  @Input() expandedFolderIds: number[] = [];\n  @Input() unorganisedCount: number = 0;\n  @Input() allowCreate: boolean = true;\n  @Input() allowRename: boolean = true;\n  @Input() allowDelete: boolean = true;\n  @Input() allowMove: boolean = true;\n  @Input() allowDuplicate: boolean = true;\n  @Input() allowDrop: boolean = true;\n  @Input() showCounts: boolean = true;\n  @Input() collapsed: boolean = false;\n  @Input() labels: ModularLabels = { ...DEFAULT_MODULAR_LABELS };\n\n  @Output() folderSelected = new EventEmitter<number | null>();\n  @Output() folderExpansionToggled = new EventEmitter<{ id: number; expanded: boolean }>();\n  /** Emitted after the host completes folder creation (e.g., inline rename flow). Reserved for future use. */\n  @Output() folderCreated = new EventEmitter<{ parentId: number | null; name: string }>();\n  /** Emitted when the user clicks the \"+\" header button. Host should open a creation modal. */\n  @Output() folderCreateRequested = new EventEmitter<{ parentId: number | null }>();\n  /** Carries the full renamed folder node + parent id so hosts can issue complete update requests. */\n  @Output() folderRenamed = new EventEmitter<FolderRenamedPayload>();\n  @Output() folderDeleted = new EventEmitter<{ id: number }>();\n  /** User picked \"Move folder\" from the context menu. Host opens the folder picker. */\n  @Output() folderMoveRequested = new EventEmitter<{ id: number }>();\n  /** User picked \"Duplicate folder\" from the context menu. Host clones the subtree (tests referenced, not copied). */\n  @Output() folderDuplicateRequested = new EventEmitter<{ id: number }>();\n  @Output() testsDropped = new EventEmitter<{ testIds: Array<string | number>; targetFolderId: number | null }>();\n  /** Fires when a folder row is dropped onto another folder (or the Unorganised row). */\n  @Output() folderDropped = new EventEmitter<{ id: number; newParentId: number | null }>();\n  @Output() collapsedChange = new EventEmitter<boolean>();\n\n  searchValue: string = '';\n  renamingId: number | null = null;\n  renameDraft: string = '';\n\n  /** Id of the folder whose context menu is open, or null when closed. */\n  contextMenuFolderId: number | null = null;\n  /** Viewport-anchored position (client coordinates) for the floating menu. */\n  contextMenuPosition: { x: number; y: number } = { x: 0, y: 0 };\n\n  /** Pending auto-expand timers while a folder drag is hovered over collapsed rows. */\n  private dragExpandTimers = new Map<number, ReturnType<typeof setTimeout>>();\n  private readonly DRAG_EXPAND_DELAY_MS = 600;\n  /** Id of the folder currently being dragged; used to skip cycle targets and source-hover expansion. */\n  private activeFolderDragId: number | null = null;\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  get expandedSet(): Set<number> {\n    return new Set(this.expandedFolderIds || []);\n  }\n\n  /** Flattened, depth-aware list of rows to render in order. Ancestors of matches are kept. */\n  get rows(): RenderNode[] {\n    const search = (this.searchValue || '').trim().toLowerCase();\n    const expanded = this.expandedSet;\n    const keepSet = new Set<number>();\n    if (search) {\n      const mark = (nodes: FolderNode[], ancestors: number[]): void => {\n        for (const n of nodes || []) {\n          const matched = (n.name || '').toLowerCase().includes(search);\n          const childAncestors = [...ancestors, n.id];\n          if (matched) {\n            for (const a of ancestors) keepSet.add(a);\n            keepSet.add(n.id);\n          }\n          if (n.children?.length) mark(n.children, childAncestors);\n        }\n      };\n      mark(this.folders, []);\n    }\n\n    const out: RenderNode[] = [];\n    const walk = (nodes: FolderNode[], depth: number, parentVisible: boolean) => {\n      for (const n of nodes || []) {\n        const searchVisible = !search || keepSet.has(n.id);\n        const visible = parentVisible && searchVisible;\n        const hasChildren = !!(n.children && n.children.length);\n        out.push({ node: n, depth, visible, hasChildren });\n        const expandedBySearch = !!search && keepSet.has(n.id);\n        const isNodeExpanded = expanded.has(n.id) || expandedBySearch;\n        if (hasChildren && isNodeExpanded) {\n          walk(n.children!, depth + 1, visible);\n        }\n      }\n    };\n    walk(this.folders, 0, true);\n    return out.filter(r => r.visible);\n  }\n\n  isExpanded(id: number): boolean {\n    return this.expandedSet.has(id);\n  }\n\n  isSelected(id: number | null): boolean {\n    if (id == null && this.selectedFolderId == null) return true;\n    return id != null && this.selectedFolderId != null && id === this.selectedFolderId;\n  }\n\n  onToggle(n: FolderNode, event: Event): void {\n    event.stopPropagation();\n    const expanded = !this.isExpanded(n.id);\n    this.folderExpansionToggled.emit({ id: n.id, expanded });\n  }\n\n  onSelect(n: FolderNode | null): void {\n    if (this.renamingId != null) return;\n    this.folderSelected.emit(n == null ? null : n.id);\n  }\n\n  onSelectUnorganised(): void {\n    // Sentinel: hosts that keep an \"Unorganised\" bucket can treat a null selection\n    // as \"show unorganised\". If there is already a root (null) selection pattern,\n    // this matches it. Host decides semantics.\n    this.folderSelected.emit(null);\n  }\n\n  beginRename(n: FolderNode, event: Event): void {\n    if (!this.allowRename) return;\n    event.stopPropagation();\n    this.renamingId = n.id;\n    this.renameDraft = n.name;\n    this.cdr.markForCheck();\n    this.focusRenameInput(n.id);\n  }\n\n  commitRename(n: FolderNode): void {\n    const name = (this.renameDraft || '').trim();\n    if (name && name !== n.name) {\n      // Emit the full renamed node (with the new name) + parent context so hosts can\n      // call `PUT /test_case_folders/{id}` without re-resolving from their cached tree.\n      const folder: FolderNode = { ...n, name };\n      this.folderRenamed.emit({\n        folder,\n        parentId: this.parentIdOf(n.id),\n        previousName: n.name,\n      });\n    }\n    this.cancelRename();\n  }\n\n  cancelRename(): void {\n    this.renamingId = null;\n    this.renameDraft = '';\n    this.cdr.markForCheck();\n  }\n\n  onRenameKey(event: KeyboardEvent, n: FolderNode): void {\n    // Stop bubbling so the row's keydown handler (which treats Space/Enter/Arrows as\n    // selection / tree navigation) can't swallow characters typed into the rename input.\n    event.stopPropagation();\n    if (event.key === 'Enter') {\n      event.preventDefault();\n      this.commitRename(n);\n    } else if (event.key === 'Escape') {\n      event.preventDefault();\n      this.cancelRename();\n    }\n  }\n\n  deleteFolder(n: FolderNode, event: Event): void {\n    if (!this.allowDelete) return;\n    event.stopPropagation();\n    this.folderDeleted.emit({ id: n.id });\n  }\n\n  /**\n   * Opens the folder context menu anchored at the given client coordinates.\n   * Called from both (contextmenu) on the row and (click) on the ellipsis button.\n   * Menu visibility is clamped in the template via CSS transforms so it never leaves the viewport.\n   */\n  openContextMenu(n: FolderNode, event: MouseEvent): void {\n    event.preventDefault();\n    event.stopPropagation();\n    if (!this.hasAnyContextAction) return;\n    this.contextMenuFolderId = n.id;\n    this.contextMenuPosition = { x: event.clientX, y: event.clientY };\n    this.cdr.markForCheck();\n  }\n\n  closeContextMenu(): void {\n    if (this.contextMenuFolderId == null) return;\n    this.contextMenuFolderId = null;\n    this.cdr.markForCheck();\n  }\n\n  /** True when at least one menu entry is enabled — otherwise the trigger is suppressed entirely. */\n  get hasAnyContextAction(): boolean {\n    return this.allowCreate || this.allowRename || this.allowMove || this.allowDuplicate || this.allowDelete;\n  }\n\n  onContextCreateSubfolder(n: FolderNode): void {\n    this.closeContextMenu();\n    if (!this.allowCreate) return;\n    this.folderCreateRequested.emit({ parentId: n.id });\n  }\n\n  onContextRename(n: FolderNode): void {\n    this.closeContextMenu();\n    if (!this.allowRename) return;\n    this.renamingId = n.id;\n    this.renameDraft = n.name;\n    this.cdr.markForCheck();\n    this.focusRenameInput(n.id);\n  }\n\n  onContextMove(n: FolderNode): void {\n    this.closeContextMenu();\n    if (!this.allowMove) return;\n    this.folderMoveRequested.emit({ id: n.id });\n  }\n\n  onContextDuplicate(n: FolderNode): void {\n    this.closeContextMenu();\n    if (!this.allowDuplicate) return;\n    this.folderDuplicateRequested.emit({ id: n.id });\n  }\n\n  onContextDelete(n: FolderNode): void {\n    this.closeContextMenu();\n    if (!this.allowDelete) return;\n    this.folderDeleted.emit({ id: n.id });\n  }\n\n  @HostListener('document:click')\n  onDocumentClick(): void {\n    this.closeContextMenu();\n  }\n\n  @HostListener('document:keydown.escape')\n  onEscape(): void {\n    this.closeContextMenu();\n  }\n\n  /** Template helper: resolve a folder node by id from the input tree. */\n  folderById(id: number | null): FolderNode | null {\n    if (id == null) return null;\n    const find = (nodes: FolderNode[]): FolderNode | null => {\n      for (const n of nodes || []) {\n        if (n.id === id) return n;\n        if (n.children?.length) {\n          const hit = find(n.children);\n          if (hit) return hit;\n        }\n      }\n      return null;\n    };\n    return find(this.folders);\n  }\n\n  /** Returns the parent id of `id`, or null when `id` is a root folder / not found. */\n  private parentIdOf(id: number): number | null {\n    let result: number | null = null;\n    const walk = (nodes: FolderNode[], parentId: number | null): boolean => {\n      for (const n of nodes || []) {\n        if (n.id === id) {\n          result = parentId;\n          return true;\n        }\n        if (n.children?.length && walk(n.children, n.id)) return true;\n      }\n      return false;\n    };\n    walk(this.folders, null);\n    return result;\n  }\n\n  requestCreate(parentId: number | null = null): void {\n    if (!this.allowCreate) return;\n    this.folderCreateRequested.emit({ parentId });\n  }\n\n  togglePanel(): void {\n    this.collapsed = !this.collapsed;\n    this.collapsedChange.emit(this.collapsed);\n  }\n\n  rowDropped(targetId: number | null, event: { testIds: Array<string | number>; targetFolderId: number | null }): void {\n    // Directive already emits with the bound folder id, but normalise in case callers\n    // wire raw events.\n    this.testsDropped.emit({ testIds: event.testIds, targetFolderId: targetId });\n  }\n\n  /**\n   * Drop target received a folder drag. Validate against self/descendant cycles and\n   * emit `folderDropped` if the move is legal. Called from the directive's\n   * `(folderDropped)` output on every row and on the Unorganised row.\n   */\n  onFolderRowDropped(targetId: number | null, event: { folderId: number; targetFolderId: number | null }): void {\n    this.clearAllExpandTimers();\n    this.activeFolderDragId = null;\n    const sourceId = event.folderId;\n    if (targetId != null && targetId === sourceId) return;\n    if (targetId != null && this.isDescendantOf(sourceId, targetId)) return;\n    this.folderDropped.emit({ id: sourceId, newParentId: targetId });\n  }\n\n  /** Remember the dragged folder so we can block self-hover expansion and cycle drops. */\n  onFolderDragStart(n: FolderNode): void {\n    this.activeFolderDragId = n.id;\n  }\n\n  onFolderDragEnd(): void {\n    this.activeFolderDragId = null;\n    this.clearAllExpandTimers();\n  }\n\n  /** Schedule an auto-expand for a collapsed folder that has children during drag-hover. */\n  onFolderRowDragOver(row: RenderNode): void {\n    const id = row.node.id;\n    if (!row.hasChildren) return;\n    if (this.isExpanded(id)) return;\n    if (this.activeFolderDragId === id) return; // don't expand the source\n    if (this.dragExpandTimers.has(id)) return;\n    const timer = setTimeout(() => {\n      this.dragExpandTimers.delete(id);\n      if (!this.isExpanded(id)) {\n        this.folderExpansionToggled.emit({ id, expanded: true });\n      }\n    }, this.DRAG_EXPAND_DELAY_MS);\n    this.dragExpandTimers.set(id, timer);\n  }\n\n  onFolderRowDragLeave(row: RenderNode): void {\n    const id = row.node.id;\n    const timer = this.dragExpandTimers.get(id);\n    if (timer != null) {\n      clearTimeout(timer);\n      this.dragExpandTimers.delete(id);\n    }\n  }\n\n  @HostListener('document:dragend')\n  onDocumentDragEnd(): void {\n    this.onFolderDragEnd();\n  }\n\n  /** True when `candidateId` is `sourceId` itself or any descendant of it. */\n  private isDescendantOf(sourceId: number, candidateId: number): boolean {\n    const source = this.folderById(sourceId);\n    if (!source) return false;\n    const stack: FolderNode[] = [source];\n    while (stack.length) {\n      const n = stack.pop()!;\n      if (n.id === candidateId) return true;\n      if (n.children?.length) stack.push(...n.children);\n    }\n    return false;\n  }\n\n  private clearAllExpandTimers(): void {\n    this.dragExpandTimers.forEach(t => clearTimeout(t));\n    this.dragExpandTimers.clear();\n  }\n\n  ngOnDestroy(): void {\n    this.clearAllExpandTimers();\n  }\n\n  trackByRow = (_: number, row: RenderNode) => row.node.id;\n\n  onRowKeydown(event: KeyboardEvent, row: RenderNode, index: number): void {\n    const rows = this.rows;\n    switch (event.key) {\n      case 'ArrowDown': {\n        event.preventDefault();\n        const next = rows[index + 1];\n        if (next) this.focusRowElement(next.node.id);\n        break;\n      }\n      case 'ArrowUp': {\n        event.preventDefault();\n        const prev = rows[index - 1];\n        if (prev) this.focusRowElement(prev.node.id);\n        break;\n      }\n      case 'ArrowRight': {\n        if (row.hasChildren && !this.isExpanded(row.node.id)) {\n          event.preventDefault();\n          this.folderExpansionToggled.emit({ id: row.node.id, expanded: true });\n        }\n        break;\n      }\n      case 'ArrowLeft': {\n        if (row.hasChildren && this.isExpanded(row.node.id)) {\n          event.preventDefault();\n          this.folderExpansionToggled.emit({ id: row.node.id, expanded: false });\n        }\n        break;\n      }\n      case 'Enter':\n      case ' ': {\n        event.preventDefault();\n        this.onSelect(row.node);\n        break;\n      }\n    }\n  }\n\n  private focusRowElement(id: number): void {\n    queueMicrotask(() => {\n      const el = document.querySelector<HTMLElement>(`[data-folder-row-id=\"${id}\"]`);\n      el?.focus();\n    });\n  }\n\n  /**\n   * Focuses the inline rename input for the given folder. Uses `setTimeout` (not\n   * microtask) because the input is gated by `*ngIf=\"renamingId === ...\"` and\n   * must survive the next OnPush change-detection cycle before it's in the DOM.\n   */\n  private focusRenameInput(id: number): void {\n    setTimeout(() => {\n      const el = document.querySelector<HTMLInputElement>(`[data-folder-rename-input=\"${id}\"]`);\n      if (!el) return;\n      el.focus();\n      el.select();\n    }, 0);\n  }\n}\n","<!-- Reusable folder icon. Render via <ng-container *ngTemplateOutlet=\"folderIcon; context: { color: node.color }\"></ng-container>. -->\n<ng-template #folderIcon let-color=\"color\">\n  <span class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-4 cqa-h-4 cqa-shrink-0\">\n    <svg width=\"13\" height=\"12\" viewBox=\"0 0 13 12\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n      <path d=\"M12.1916 9.85824C12.1916 10.1677 12.0687 10.4644 11.8499 10.6832C11.6311 10.902 11.3343 11.0249 11.0249 11.0249H1.69157C1.38215 11.0249 1.0854 10.902 0.866611 10.6832C0.647819 10.4644 0.524902 10.1677 0.524902 9.85824V1.69157C0.524902 1.38215 0.647819 1.0854 0.866611 0.866611C1.0854 0.647819 1.38215 0.524902 1.69157 0.524902H4.60824L5.7749 2.2749H11.0249C11.3343 2.2749 11.6311 2.39782 11.8499 2.61661C12.0687 2.8354 12.1916 3.13215 12.1916 3.44157V9.85824Z\" [attr.stroke]=\"color || '#99999E'\" stroke-width=\"1.05\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n    </svg>\n  </span>\n</ng-template>\n\n<aside\n  class=\"cqa-flex cqa-flex-col cqa-bg-white cqa-border cqa-border-neutral-200 cqa-rounded-lg cqa-h-full cqa-min-h-0\"\n  [class.cqa-w-[240px]]=\"!collapsed\"\n  [class.cqa-w-[48px]]=\"collapsed\"\n  style=\"transition: width 150ms ease;\"\n>\n  <!-- Header -->\n  <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-px-3 cqa-py-3\">\n    <span *ngIf=\"!collapsed\" class=\"cqa-text-sm cqa-font-semibold cqa-text-neutral-900\">\n      {{ labels.folders }}\n    </span>\n    <div class=\"cqa-flex cqa-items-center cqa-gap-1\" [class.cqa-w-full]=\"collapsed\" [class.cqa-justify-center]=\"collapsed\">\n      <button\n        type=\"button\"\n        class=\"cqa-p-1 cqa-rounded hover:cqa-bg-neutral-100 cqa-text-neutral-500\"\n        (click)=\"togglePanel()\"\n        [attr.aria-label]=\"collapsed ? 'Expand folders panel' : 'Collapse folders panel'\"\n      >\n        <mat-icon style=\"font-size:18px;width:18px;height:18px\">\n          {{ collapsed ? 'chevron_right' : 'keyboard_double_arrow_left' }}\n        </mat-icon>\n      </button>\n      <button\n        *ngIf=\"!collapsed && allowCreate\"\n        type=\"button\"\n        class=\"cqa-p-1 cqa-rounded hover:cqa-bg-neutral-100 cqa-text-neutral-500\"\n        (click)=\"requestCreate(null)\"\n        [attr.aria-label]=\"labels.newFolder\"\n        [title]=\"labels.newFolder\"\n      >\n        <mat-icon style=\"font-size:18px;width:18px;height:18px\">add</mat-icon>\n      </button>\n    </div>\n  </div>\n\n  <ng-container *ngIf=\"!collapsed\">\n    <!-- Search -->\n    <div class=\"cqa-px-3 cqa-pb-2\">\n      <cqa-search-bar\n        size=\"sm\"\n        [fullWidth]=\"true\"\n        [value]=\"searchValue\"\n        [placeholder]=\"labels.searchFoldersPlaceholder\"\n        [showClear]=\"true\"\n        (valueChange)=\"searchValue = $event\"\n        (cleared)=\"searchValue = ''\"\n      ></cqa-search-bar>\n    </div>\n\n    <!-- Tree (folders + Unorganised tab) -->\n    <div role=\"tree\" class=\"cqa-flex-1 cqa-overflow-y-auto cqa-py-1\">\n      <ng-container *ngFor=\"let row of rows; let i = index; trackBy: trackByRow\">\n        <div\n          role=\"treeitem\"\n          tabindex=\"0\"\n          [attr.aria-expanded]=\"row.hasChildren ? isExpanded(row.node.id) : null\"\n          [attr.aria-selected]=\"isSelected(row.node.id)\"\n          [attr.data-folder-row-id]=\"row.node.id\"\n          [cqaFolderDrop]=\"row.node.id\"\n          [dropEnabled]=\"allowDrop\"\n          (testsDropped)=\"rowDropped(row.node.id, $event)\"\n          (folderDropped)=\"onFolderRowDropped(row.node.id, $event)\"\n          [cqaFolderDrag]=\"row.node.id\"\n          [dragEnabled]=\"allowDrop\"\n          (dragstart)=\"onFolderDragStart(row.node)\"\n          (dragend)=\"onFolderDragEnd()\"\n          (dragover)=\"onFolderRowDragOver(row)\"\n          (dragleave)=\"onFolderRowDragLeave(row)\"\n          (click)=\"onSelect(row.node)\"\n          (contextmenu)=\"openContextMenu(row.node, $event)\"\n          (keydown)=\"onRowKeydown($event, row, i)\"\n          class=\"cqa-group cqa-flex cqa-items-center cqa-gap-1 cqa-pr-3 cqa-py-1.5 cqa-cursor-pointer hover:cqa-bg-neutral-50 focus:cqa-outline-none focus:cqa-bg-neutral-100 cqa-relative\"\n          [class.cqa-bg-indigo-50]=\"isSelected(row.node.id) || contextMenuFolderId === row.node.id\"\n          [style.paddingLeft.px]=\"8 + row.depth * 16\"\n          [attr.title]=\"row.node.totalCount != null ? (row.node.totalCount + ' total including subfolders') : null\"\n        >\n          <span\n            *ngIf=\"isSelected(row.node.id)\"\n            class=\"cqa-absolute cqa-left-0 cqa-top-0 cqa-bottom-0 cqa-w-[3px] cqa-bg-indigo-600\"\n          ></span>\n          <button\n            type=\"button\"\n            class=\"cqa-p-0 cqa-text-neutral-400\"\n            [style.visibility]=\"row.hasChildren ? 'visible' : 'hidden'\"\n            (click)=\"onToggle(row.node, $event)\"\n            [attr.aria-label]=\"isExpanded(row.node.id) ? 'Collapse' : 'Expand'\"\n          >\n            <mat-icon style=\"font-size:16px;width:16px;height:16px\">\n              {{ isExpanded(row.node.id) ? 'expand_more' : 'chevron_right' }}\n            </mat-icon>\n          </button>\n          <ng-container *ngTemplateOutlet=\"folderIcon; context: { color: row.node.color }\"></ng-container>\n\n          <ng-container *ngIf=\"renamingId === row.node.id; else nameTpl\">\n            <input\n              type=\"text\"\n              size=\"1\"\n              [attr.data-folder-rename-input]=\"row.node.id\"\n              class=\"cqa-flex-1 cqa-min-w-0 cqa-w-full cqa-px-2 cqa-py-0.5 cqa-rounded cqa-border cqa-border-neutral-300 cqa-bg-white cqa-text-sm cqa-shadow-sm focus:cqa-outline-none focus:cqa-border-indigo-400 focus:cqa-ring-1 focus:cqa-ring-indigo-200\"\n              [(ngModel)]=\"renameDraft\"\n              (keydown)=\"onRenameKey($event, row.node)\"\n              (keypress)=\"$event.stopPropagation()\"\n              (keyup)=\"$event.stopPropagation()\"\n              (blur)=\"commitRename(row.node)\"\n              (click)=\"$event.stopPropagation()\"\n            />\n          </ng-container>\n          <ng-template #nameTpl>\n            <span\n              class=\"cqa-flex-1 cqa-text-sm cqa-text-neutral-800 cqa-truncate\"\n              (dblclick)=\"beginRename(row.node, $event)\"\n            >{{ row.node.name }}</span>\n          </ng-template>\n\n          <!-- Count shown at rest; hidden on row hover when delete is allowed -->\n          <span\n            *ngIf=\"showCounts && row.node.count != null\"\n            class=\"cqa-text-xs cqa-text-neutral-400 cqa-tabular-nums cqa-ml-1\"\n            [class.group-hover:cqa-hidden]=\"allowDelete\"\n          >{{ row.node.count }}</span>\n\n          <button\n            *ngIf=\"hasAnyContextAction\"\n            type=\"button\"\n            class=\"cqa-p-0.5 cqa-rounded hover:cqa-bg-neutral-200 cqa-text-neutral-500\"\n            [class.cqa-hidden]=\"contextMenuFolderId !== row.node.id\"\n            [class.group-hover:cqa-inline-flex]=\"true\"\n            [class.cqa-inline-flex]=\"contextMenuFolderId === row.node.id\"\n            (click)=\"openContextMenu(row.node, $event)\"\n            [attr.aria-label]=\"'Open actions for ' + row.node.name\"\n            [attr.aria-haspopup]=\"'menu'\"\n            [attr.aria-expanded]=\"contextMenuFolderId === row.node.id\"\n          >\n            <mat-icon style=\"font-size:16px;width:16px;height:16px\">more_horiz</mat-icon>\n          </button>\n        </div>\n      </ng-container>\n\n      <!-- Divider between folder tree and Unorganised tab -->\n      <div\n        aria-hidden=\"true\"\n        class=\"cqa-mx-3\"\n        style=\"height: 1px; background-color: #E5E7EB; margin-top: 6px; margin-bottom: 6px;\"\n      ></div>\n\n      <!-- Unorganised — flows with the tree, behaves like a tab item -->\n      <div\n        role=\"treeitem\"\n        tabindex=\"0\"\n        [attr.aria-selected]=\"isSelected(null)\"\n        class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-px-3 cqa-py-1.5 cqa-cursor-pointer hover:cqa-bg-neutral-50 focus:cqa-outline-none focus:cqa-bg-neutral-100 cqa-relative\"\n        [class.cqa-bg-indigo-50]=\"isSelected(null)\"\n        [cqaFolderDrop]=\"null\"\n        [dropEnabled]=\"allowDrop\"\n        (testsDropped)=\"rowDropped(null, $event)\"\n        (folderDropped)=\"onFolderRowDropped(null, $event)\"\n        (click)=\"onSelectUnorganised()\"\n      >\n        <span\n          *ngIf=\"isSelected(null)\"\n          class=\"cqa-absolute cqa-left-0 cqa-top-0 cqa-bottom-0 cqa-w-[3px] cqa-bg-indigo-600\"\n        ></span>\n        <mat-icon class=\"cqa-text-neutral-500\" style=\"font-size:16px;width:16px;height:16px\">inbox</mat-icon>\n        <span class=\"cqa-flex-1 cqa-text-sm cqa-text-neutral-700\">{{ labels.unorganised }}</span>\n        <span *ngIf=\"showCounts\" class=\"cqa-text-xs cqa-text-neutral-400 cqa-tabular-nums\">{{ unorganisedCount }}</span>\n      </div>\n    </div>\n\n    <!-- Folder context menu (right-click / ellipsis). Anchored to viewport coords. -->\n    <div\n      *ngIf=\"contextMenuFolderId !== null\"\n      role=\"menu\"\n      class=\"cqa-fixed cqa-z-50 cqa-min-w-[180px] cqa-bg-white cqa-border cqa-border-neutral-200 cqa-rounded-md cqa-shadow-lg cqa-py-1\"\n      [style.left.px]=\"contextMenuPosition.x\"\n      [style.top.px]=\"contextMenuPosition.y\"\n      (click)=\"$event.stopPropagation()\"\n      (contextmenu)=\"$event.preventDefault(); $event.stopPropagation()\"\n    >\n      <ng-container *ngIf=\"folderById(contextMenuFolderId) as menuNode\">\n        <button\n          *ngIf=\"allowCreate\"\n          type=\"button\"\n          role=\"menuitem\"\n          class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-neutral-800 hover:cqa-bg-neutral-100 cqa-text-left\"\n          (click)=\"onContextCreateSubfolder(menuNode)\"\n        >\n          <mat-icon style=\"font-size:16px;width:16px;height:16px\">create_new_folder</mat-icon>\n          <span>{{ labels.folderMenuCreateSubfolder }}</span>\n        </button>\n        <button\n          *ngIf=\"allowRename\"\n          type=\"button\"\n          role=\"menuitem\"\n          class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-neutral-800 hover:cqa-bg-neutral-100 cqa-text-left\"\n          (click)=\"onContextRename(menuNode)\"\n        >\n          <mat-icon style=\"font-size:16px;width:16px;height:16px\">edit</mat-icon>\n          <span>{{ labels.folderMenuRename }}</span>\n        </button>\n        <button\n          *ngIf=\"allowMove\"\n          type=\"button\"\n          role=\"menuitem\"\n          class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-neutral-800 hover:cqa-bg-neutral-100 cqa-text-left\"\n          (click)=\"onContextMove(menuNode)\"\n        >\n          <mat-icon style=\"font-size:16px;width:16px;height:16px\">drive_file_move</mat-icon>\n          <span>{{ labels.folderMenuMove }}</span>\n        </button>\n        <button\n          *ngIf=\"allowDuplicate\"\n          type=\"button\"\n          role=\"menuitem\"\n          class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-neutral-800 hover:cqa-bg-neutral-100 cqa-text-left\"\n          (click)=\"onContextDuplicate(menuNode)\"\n        >\n          <mat-icon style=\"font-size:16px;width:16px;height:16px\">content_copy</mat-icon>\n          <span>{{ labels.folderMenuDuplicate }}</span>\n        </button>\n        <div *ngIf=\"allowDelete && (allowCreate || allowRename || allowMove || allowDuplicate)\" class=\"cqa-h-px cqa-bg-neutral-200 cqa-my-1\"></div>\n        <button\n          *ngIf=\"allowDelete\"\n          type=\"button\"\n          role=\"menuitem\"\n          class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-w-full cqa-px-3 cqa-py-1.5 cqa-text-sm cqa-text-red-600 hover:cqa-bg-red-50 cqa-text-left\"\n          (click)=\"onContextDelete(menuNode)\"\n        >\n          <mat-icon style=\"font-size:16px;width:16px;height:16px\">delete_outline</mat-icon>\n          <span>{{ labels.folderMenuDelete }}</span>\n        </button>\n      </ng-container>\n    </div>\n\n  </ng-container>\n</aside>\n"]}
|