@cqa-lib/cqa-ui 1.1.525 → 1.1.527
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 +3 -3
- 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 +3409 -179
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +3389 -177
- 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/dialogs/delete-folder-dialog.component.mjs
ADDED
|
@@ -0,0 +1,181 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, 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/common";
|
|
5
|
+
/**
|
|
6
|
+
* Body of the "Delete folder" dialog — a radio-card picker for what happens to
|
|
7
|
+
* the folder's contained test cases.
|
|
8
|
+
*
|
|
9
|
+
* Designed to be usable two ways:
|
|
10
|
+
*
|
|
11
|
+
* 1. **From within this library** via `DialogService.open(...)`. The outer
|
|
12
|
+
* `cqa-dialog` supplies the title/Cancel/Delete-folder buttons; the Delete
|
|
13
|
+
* button reads `strategy` via `getComponentInstance()`.
|
|
14
|
+
*
|
|
15
|
+
* 2. **From the host/UI layer directly** — hosts render this inside their own
|
|
16
|
+
* modal and observe `(strategyChange)`.
|
|
17
|
+
*/
|
|
18
|
+
export class DeleteFolderDialogComponent {
|
|
19
|
+
constructor(cdr) {
|
|
20
|
+
this.cdr = cdr;
|
|
21
|
+
this.folderName = '';
|
|
22
|
+
this.testCount = 0;
|
|
23
|
+
this.hasParent = false;
|
|
24
|
+
this.labels = { ...DEFAULT_MODULAR_LABELS };
|
|
25
|
+
this.strategy = 'MOVE_TO_PARENT';
|
|
26
|
+
this.strategyChange = new EventEmitter();
|
|
27
|
+
}
|
|
28
|
+
ngOnInit() {
|
|
29
|
+
this.normaliseStrategy();
|
|
30
|
+
}
|
|
31
|
+
ngOnChanges(changes) {
|
|
32
|
+
if (changes['hasParent'])
|
|
33
|
+
this.normaliseStrategy();
|
|
34
|
+
}
|
|
35
|
+
/** If the folder has no parent, the "move to parent" option isn't available — fall back. */
|
|
36
|
+
normaliseStrategy() {
|
|
37
|
+
if (!this.hasParent && this.strategy === 'MOVE_TO_PARENT') {
|
|
38
|
+
this.strategy = 'MOVE_TO_UNORGANISED';
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
get bodyText() {
|
|
42
|
+
const template = this.testCount === 1
|
|
43
|
+
? this.labels.deleteFolderDialogBodySingular
|
|
44
|
+
: this.labels.deleteFolderDialogBodyPlural;
|
|
45
|
+
return template.replace('{n}', '' + this.testCount);
|
|
46
|
+
}
|
|
47
|
+
setStrategy(value) {
|
|
48
|
+
if (this.strategy === value)
|
|
49
|
+
return;
|
|
50
|
+
this.strategy = value;
|
|
51
|
+
this.strategyChange.emit(value);
|
|
52
|
+
this.cdr.markForCheck();
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
DeleteFolderDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DeleteFolderDialogComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
56
|
+
DeleteFolderDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DeleteFolderDialogComponent, selector: "cqa-delete-folder-dialog", inputs: { folderName: "folderName", testCount: "testCount", hasParent: "hasParent", labels: "labels", strategy: "strategy" }, outputs: { strategyChange: "strategyChange" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: `
|
|
57
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full">
|
|
58
|
+
<p class="cqa-text-sm cqa-text-neutral-800 cqa-m-0">{{ bodyText }}</p>
|
|
59
|
+
|
|
60
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-3">
|
|
61
|
+
<!-- Option 1: Move to parent folder (hidden when folder has no parent) -->
|
|
62
|
+
<button
|
|
63
|
+
*ngIf="hasParent"
|
|
64
|
+
type="button"
|
|
65
|
+
class="cqa-flex cqa-items-start cqa-gap-3 cqa-text-left cqa-p-4 cqa-rounded-[10px] cqa-border-solid cqa-border cqa-transition-colors cqa-w-full"
|
|
66
|
+
[ngClass]="strategy === 'MOVE_TO_PARENT' ? 'cqa-border-indigo-500 cqa-bg-indigo-50' : 'cqa-border-[#E2E2E3] hover:cqa-border-neutral-300 cqa-bg-white'"
|
|
67
|
+
(click)="setStrategy('MOVE_TO_PARENT')"
|
|
68
|
+
>
|
|
69
|
+
<ng-container *ngTemplateOutlet="radio; context: { checked: strategy === 'MOVE_TO_PARENT' }"></ng-container>
|
|
70
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0">
|
|
71
|
+
<span class="cqa-text-sm cqa-font-semibold cqa-text-neutral-900">
|
|
72
|
+
{{ labels.deleteFolderDialogMoveToParentTitle }}
|
|
73
|
+
</span>
|
|
74
|
+
<span class="cqa-text-sm cqa-text-neutral-500">
|
|
75
|
+
{{ labels.deleteFolderDialogMoveToParentDescription }}
|
|
76
|
+
</span>
|
|
77
|
+
</div>
|
|
78
|
+
</button>
|
|
79
|
+
|
|
80
|
+
<!-- Option 2: Move to Unorganised -->
|
|
81
|
+
<button
|
|
82
|
+
type="button"
|
|
83
|
+
class="cqa-flex cqa-items-start cqa-gap-3 cqa-text-left cqa-p-4 cqa-rounded-[10px] cqa-border-solid cqa-border cqa-transition-colors cqa-w-full"
|
|
84
|
+
[ngClass]="strategy === 'MOVE_TO_UNORGANISED' ? 'cqa-border-indigo-500 cqa-bg-indigo-50' : 'cqa-border-[#E2E2E3] hover:cqa-border-neutral-300 cqa-bg-white'"
|
|
85
|
+
(click)="setStrategy('MOVE_TO_UNORGANISED')"
|
|
86
|
+
>
|
|
87
|
+
<ng-container *ngTemplateOutlet="radio; context: { checked: strategy === 'MOVE_TO_UNORGANISED' }"></ng-container>
|
|
88
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0">
|
|
89
|
+
<span class="cqa-text-sm cqa-font-semibold cqa-text-neutral-900">
|
|
90
|
+
{{ labels.deleteFolderDialogMoveToUnorganisedTitle }}
|
|
91
|
+
</span>
|
|
92
|
+
<span class="cqa-text-sm cqa-text-neutral-500">
|
|
93
|
+
{{ labels.deleteFolderDialogMoveToUnorganisedDescription }}
|
|
94
|
+
</span>
|
|
95
|
+
</div>
|
|
96
|
+
</button>
|
|
97
|
+
</div>
|
|
98
|
+
</div>
|
|
99
|
+
|
|
100
|
+
<!-- Reusable radio bullet -->
|
|
101
|
+
<ng-template #radio let-checked="checked">
|
|
102
|
+
<span
|
|
103
|
+
class="cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-5 cqa-h-5 cqa-rounded-full cqa-border-solid cqa-shrink-0 cqa-mt-0.5"
|
|
104
|
+
[ngClass]="checked ? 'cqa-border-indigo-600' : 'cqa-border-neutral-400'"
|
|
105
|
+
[style.borderWidth]="'2px'"
|
|
106
|
+
>
|
|
107
|
+
<span *ngIf="checked" class="cqa-w-2.5 cqa-h-2.5 cqa-rounded-full cqa-bg-indigo-600"></span>
|
|
108
|
+
</span>
|
|
109
|
+
</ng-template>
|
|
110
|
+
`, isInline: true, directives: [{ type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
111
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DeleteFolderDialogComponent, decorators: [{
|
|
112
|
+
type: Component,
|
|
113
|
+
args: [{ selector: 'cqa-delete-folder-dialog', template: `
|
|
114
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full">
|
|
115
|
+
<p class="cqa-text-sm cqa-text-neutral-800 cqa-m-0">{{ bodyText }}</p>
|
|
116
|
+
|
|
117
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-3">
|
|
118
|
+
<!-- Option 1: Move to parent folder (hidden when folder has no parent) -->
|
|
119
|
+
<button
|
|
120
|
+
*ngIf="hasParent"
|
|
121
|
+
type="button"
|
|
122
|
+
class="cqa-flex cqa-items-start cqa-gap-3 cqa-text-left cqa-p-4 cqa-rounded-[10px] cqa-border-solid cqa-border cqa-transition-colors cqa-w-full"
|
|
123
|
+
[ngClass]="strategy === 'MOVE_TO_PARENT' ? 'cqa-border-indigo-500 cqa-bg-indigo-50' : 'cqa-border-[#E2E2E3] hover:cqa-border-neutral-300 cqa-bg-white'"
|
|
124
|
+
(click)="setStrategy('MOVE_TO_PARENT')"
|
|
125
|
+
>
|
|
126
|
+
<ng-container *ngTemplateOutlet="radio; context: { checked: strategy === 'MOVE_TO_PARENT' }"></ng-container>
|
|
127
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0">
|
|
128
|
+
<span class="cqa-text-sm cqa-font-semibold cqa-text-neutral-900">
|
|
129
|
+
{{ labels.deleteFolderDialogMoveToParentTitle }}
|
|
130
|
+
</span>
|
|
131
|
+
<span class="cqa-text-sm cqa-text-neutral-500">
|
|
132
|
+
{{ labels.deleteFolderDialogMoveToParentDescription }}
|
|
133
|
+
</span>
|
|
134
|
+
</div>
|
|
135
|
+
</button>
|
|
136
|
+
|
|
137
|
+
<!-- Option 2: Move to Unorganised -->
|
|
138
|
+
<button
|
|
139
|
+
type="button"
|
|
140
|
+
class="cqa-flex cqa-items-start cqa-gap-3 cqa-text-left cqa-p-4 cqa-rounded-[10px] cqa-border-solid cqa-border cqa-transition-colors cqa-w-full"
|
|
141
|
+
[ngClass]="strategy === 'MOVE_TO_UNORGANISED' ? 'cqa-border-indigo-500 cqa-bg-indigo-50' : 'cqa-border-[#E2E2E3] hover:cqa-border-neutral-300 cqa-bg-white'"
|
|
142
|
+
(click)="setStrategy('MOVE_TO_UNORGANISED')"
|
|
143
|
+
>
|
|
144
|
+
<ng-container *ngTemplateOutlet="radio; context: { checked: strategy === 'MOVE_TO_UNORGANISED' }"></ng-container>
|
|
145
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0">
|
|
146
|
+
<span class="cqa-text-sm cqa-font-semibold cqa-text-neutral-900">
|
|
147
|
+
{{ labels.deleteFolderDialogMoveToUnorganisedTitle }}
|
|
148
|
+
</span>
|
|
149
|
+
<span class="cqa-text-sm cqa-text-neutral-500">
|
|
150
|
+
{{ labels.deleteFolderDialogMoveToUnorganisedDescription }}
|
|
151
|
+
</span>
|
|
152
|
+
</div>
|
|
153
|
+
</button>
|
|
154
|
+
</div>
|
|
155
|
+
</div>
|
|
156
|
+
|
|
157
|
+
<!-- Reusable radio bullet -->
|
|
158
|
+
<ng-template #radio let-checked="checked">
|
|
159
|
+
<span
|
|
160
|
+
class="cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-5 cqa-h-5 cqa-rounded-full cqa-border-solid cqa-shrink-0 cqa-mt-0.5"
|
|
161
|
+
[ngClass]="checked ? 'cqa-border-indigo-600' : 'cqa-border-neutral-400'"
|
|
162
|
+
[style.borderWidth]="'2px'"
|
|
163
|
+
>
|
|
164
|
+
<span *ngIf="checked" class="cqa-w-2.5 cqa-h-2.5 cqa-rounded-full cqa-bg-indigo-600"></span>
|
|
165
|
+
</span>
|
|
166
|
+
</ng-template>
|
|
167
|
+
`, host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.OnPush, styles: [] }]
|
|
168
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { folderName: [{
|
|
169
|
+
type: Input
|
|
170
|
+
}], testCount: [{
|
|
171
|
+
type: Input
|
|
172
|
+
}], hasParent: [{
|
|
173
|
+
type: Input
|
|
174
|
+
}], labels: [{
|
|
175
|
+
type: Input
|
|
176
|
+
}], strategy: [{
|
|
177
|
+
type: Input
|
|
178
|
+
}], strategyChange: [{
|
|
179
|
+
type: Output
|
|
180
|
+
}] } });
|
|
181
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"delete-folder-dialog.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/templates/modular-table-template/dialogs/delete-folder-dialog.component.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EAEvB,SAAS,EACT,YAAY,EACZ,KAAK,EAGL,MAAM,GAEP,MAAM,eAAe,CAAC;AACvB,OAAO,EACL,sBAAsB,GAGvB,MAAM,kCAAkC,CAAC;;;AAE1C;;;;;;;;;;;;GAYG;AA8DH,MAAM,OAAO,2BAA2B;IAStC,YAAoB,GAAsB;QAAtB,QAAG,GAAH,GAAG,CAAmB;QARjC,eAAU,GAAW,EAAE,CAAC;QACxB,cAAS,GAAW,CAAC,CAAC;QACtB,cAAS,GAAY,KAAK,CAAC;QAC3B,WAAM,GAAkB,EAAE,GAAG,sBAAsB,EAAE,CAAC;QACtD,aAAQ,GAAyB,gBAAgB,CAAC;QAEjD,mBAAc,GAAG,IAAI,YAAY,EAAwB,CAAC;IAEvB,CAAC;IAE9C,QAAQ;QACN,IAAI,CAAC,iBAAiB,EAAE,CAAC;IAC3B,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,WAAW,CAAC;YAAE,IAAI,CAAC,iBAAiB,EAAE,CAAC;IACrD,CAAC;IAED,4FAA4F;IACpF,iBAAiB;QACvB,IAAI,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,QAAQ,KAAK,gBAAgB,EAAE;YACzD,IAAI,CAAC,QAAQ,GAAG,qBAAqB,CAAC;SACvC;IACH,CAAC;IAED,IAAI,QAAQ;QACV,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,KAAK,CAAC;YACnC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,8BAA8B;YAC5C,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,4BAA4B,CAAC;QAC7C,OAAO,QAAQ,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC;IACtD,CAAC;IAED,WAAW,CAAC,KAA2B;QACrC,IAAI,IAAI,CAAC,QAAQ,KAAK,KAAK;YAAE,OAAO;QACpC,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;QACtB,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAChC,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;;wHAtCU,2BAA2B;4GAA3B,2BAA2B,2SA3D5B;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsDT;2FAKU,2BAA2B;kBA7DvC,SAAS;+BACE,0BAA0B,YAC1B;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsDT,QAEK,EAAE,KAAK,EAAE,aAAa,EAAE,mBACb,uBAAuB,CAAC,MAAM;wGAGtC,UAAU;sBAAlB,KAAK;gBACG,SAAS;sBAAjB,KAAK;gBACG,SAAS;sBAAjB,KAAK;gBACG,MAAM;sBAAd,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBAEI,cAAc;sBAAvB,MAAM","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n} from '@angular/core';\nimport {\n  DEFAULT_MODULAR_LABELS,\n  FolderDeleteStrategy,\n  ModularLabels,\n} from '../modular-table-template.models';\n\n/**\n * Body of the \"Delete folder\" dialog — a radio-card picker for what happens to\n * the folder's contained test cases.\n *\n * Designed to be usable two ways:\n *\n *  1. **From within this library** via `DialogService.open(...)`. The outer\n *     `cqa-dialog` supplies the title/Cancel/Delete-folder buttons; the Delete\n *     button reads `strategy` via `getComponentInstance()`.\n *\n *  2. **From the host/UI layer directly** — hosts render this inside their own\n *     modal and observe `(strategyChange)`.\n */\n@Component({\n  selector: 'cqa-delete-folder-dialog',\n  template: `\n    <div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n      <p class=\"cqa-text-sm cqa-text-neutral-800 cqa-m-0\">{{ bodyText }}</p>\n\n      <div class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n        <!-- Option 1: Move to parent folder (hidden when folder has no parent) -->\n        <button\n          *ngIf=\"hasParent\"\n          type=\"button\"\n          class=\"cqa-flex cqa-items-start cqa-gap-3 cqa-text-left cqa-p-4 cqa-rounded-[10px] cqa-border-solid cqa-border cqa-transition-colors cqa-w-full\"\n          [ngClass]=\"strategy === 'MOVE_TO_PARENT' ? 'cqa-border-indigo-500 cqa-bg-indigo-50' : 'cqa-border-[#E2E2E3] hover:cqa-border-neutral-300 cqa-bg-white'\"\n          (click)=\"setStrategy('MOVE_TO_PARENT')\"\n        >\n          <ng-container *ngTemplateOutlet=\"radio; context: { checked: strategy === 'MOVE_TO_PARENT' }\"></ng-container>\n          <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n            <span class=\"cqa-text-sm cqa-font-semibold cqa-text-neutral-900\">\n              {{ labels.deleteFolderDialogMoveToParentTitle }}\n            </span>\n            <span class=\"cqa-text-sm cqa-text-neutral-500\">\n              {{ labels.deleteFolderDialogMoveToParentDescription }}\n            </span>\n          </div>\n        </button>\n\n        <!-- Option 2: Move to Unorganised -->\n        <button\n          type=\"button\"\n          class=\"cqa-flex cqa-items-start cqa-gap-3 cqa-text-left cqa-p-4 cqa-rounded-[10px] cqa-border-solid cqa-border cqa-transition-colors cqa-w-full\"\n          [ngClass]=\"strategy === 'MOVE_TO_UNORGANISED' ? 'cqa-border-indigo-500 cqa-bg-indigo-50' : 'cqa-border-[#E2E2E3] hover:cqa-border-neutral-300 cqa-bg-white'\"\n          (click)=\"setStrategy('MOVE_TO_UNORGANISED')\"\n        >\n          <ng-container *ngTemplateOutlet=\"radio; context: { checked: strategy === 'MOVE_TO_UNORGANISED' }\"></ng-container>\n          <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n            <span class=\"cqa-text-sm cqa-font-semibold cqa-text-neutral-900\">\n              {{ labels.deleteFolderDialogMoveToUnorganisedTitle }}\n            </span>\n            <span class=\"cqa-text-sm cqa-text-neutral-500\">\n              {{ labels.deleteFolderDialogMoveToUnorganisedDescription }}\n            </span>\n          </div>\n        </button>\n      </div>\n    </div>\n\n    <!-- Reusable radio bullet -->\n    <ng-template #radio let-checked=\"checked\">\n      <span\n        class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-5 cqa-h-5 cqa-rounded-full cqa-border-solid cqa-shrink-0 cqa-mt-0.5\"\n        [ngClass]=\"checked ? 'cqa-border-indigo-600' : 'cqa-border-neutral-400'\"\n        [style.borderWidth]=\"'2px'\"\n      >\n        <span *ngIf=\"checked\" class=\"cqa-w-2.5 cqa-h-2.5 cqa-rounded-full cqa-bg-indigo-600\"></span>\n      </span>\n    </ng-template>\n  `,\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class DeleteFolderDialogComponent implements OnInit, OnChanges {\n  @Input() folderName: string = '';\n  @Input() testCount: number = 0;\n  @Input() hasParent: boolean = false;\n  @Input() labels: ModularLabels = { ...DEFAULT_MODULAR_LABELS };\n  @Input() strategy: FolderDeleteStrategy = 'MOVE_TO_PARENT';\n\n  @Output() strategyChange = new EventEmitter<FolderDeleteStrategy>();\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    this.normaliseStrategy();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['hasParent']) this.normaliseStrategy();\n  }\n\n  /** If the folder has no parent, the \"move to parent\" option isn't available — fall back. */\n  private normaliseStrategy(): void {\n    if (!this.hasParent && this.strategy === 'MOVE_TO_PARENT') {\n      this.strategy = 'MOVE_TO_UNORGANISED';\n    }\n  }\n\n  get bodyText(): string {\n    const template = this.testCount === 1\n      ? this.labels.deleteFolderDialogBodySingular\n      : this.labels.deleteFolderDialogBodyPlural;\n    return template.replace('{n}', '' + this.testCount);\n  }\n\n  setStrategy(value: FolderDeleteStrategy): void {\n    if (this.strategy === value) return;\n    this.strategy = value;\n    this.strategyChange.emit(value);\n    this.cdr.markForCheck();\n  }\n}\n"]}
|
package/esm2020/lib/templates/modular-table-template/dialogs/move-to-folder-dialog.component.mjs
ADDED
|
@@ -0,0 +1,264 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, 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/common";
|
|
5
|
+
/**
|
|
6
|
+
* Folder-picker body used inside the Move-to-Folder dialog.
|
|
7
|
+
*
|
|
8
|
+
* This component is designed to be usable two ways:
|
|
9
|
+
*
|
|
10
|
+
* 1. **From inside the library** via `DialogService.open({ content: { type: 'component', component: MoveToFolderDialogComponent, inputs: {...} } })`.
|
|
11
|
+
* The outer `cqa-dialog` provides the title, description, Cancel and "Move here"
|
|
12
|
+
* buttons; the Move button reads `pickedFolderId` via `dialogRef.getComponentInstance()`.
|
|
13
|
+
*
|
|
14
|
+
* 2. **From the host/UI layer directly** once API ownership moves up. The host
|
|
15
|
+
* renders `<cqa-move-to-folder-dialog>` inside its own modal shell and
|
|
16
|
+
* listens to `(folderPicked)` (fires on every pick) to drive its own "Move here"
|
|
17
|
+
* button. All visual state lives on `pickedFolderId` so hosts can also two-way
|
|
18
|
+
* bind if they prefer.
|
|
19
|
+
*/
|
|
20
|
+
export class MoveToFolderDialogComponent {
|
|
21
|
+
constructor(cdr) {
|
|
22
|
+
this.cdr = cdr;
|
|
23
|
+
/** Folder tree to pick from. */
|
|
24
|
+
this.folders = [];
|
|
25
|
+
/** Label overrides. */
|
|
26
|
+
this.labels = { ...DEFAULT_MODULAR_LABELS };
|
|
27
|
+
/** The folder the tests currently live in — rendered as disabled (can't move into itself). */
|
|
28
|
+
this.currentFolderId = null;
|
|
29
|
+
/** Controls which folder is visually selected. Two-way-bind friendly via `pickedFolderIdChange`. */
|
|
30
|
+
this.pickedFolderId = null;
|
|
31
|
+
/** Fires whenever the user picks a destination. */
|
|
32
|
+
this.folderPicked = new EventEmitter();
|
|
33
|
+
/** Companion output for banana-in-the-box `[(pickedFolderId)]` binding. */
|
|
34
|
+
this.pickedFolderIdChange = new EventEmitter();
|
|
35
|
+
this.expandedIds = new Set();
|
|
36
|
+
this.trackById = (_, row) => row.id ?? -1;
|
|
37
|
+
}
|
|
38
|
+
ngOnInit() {
|
|
39
|
+
this.seedExpandedIds();
|
|
40
|
+
}
|
|
41
|
+
ngOnChanges(changes) {
|
|
42
|
+
if (changes['initialExpandedIds'] || changes['currentFolderId'] || changes['folders']) {
|
|
43
|
+
this.seedExpandedIds();
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
seedExpandedIds() {
|
|
47
|
+
if (this.initialExpandedIds?.length) {
|
|
48
|
+
this.expandedIds = new Set(this.initialExpandedIds);
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
51
|
+
this.expandedIds = new Set();
|
|
52
|
+
if (this.currentFolderId != null) {
|
|
53
|
+
const trail = [];
|
|
54
|
+
this.collectAncestors(this.folders, this.currentFolderId, trail);
|
|
55
|
+
trail.forEach(id => this.expandedIds.add(id));
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
collectAncestors(nodes, targetId, acc) {
|
|
59
|
+
for (const n of nodes || []) {
|
|
60
|
+
acc.push(n.id);
|
|
61
|
+
if (n.id === targetId)
|
|
62
|
+
return true;
|
|
63
|
+
if (n.children && this.collectAncestors(n.children, targetId, acc))
|
|
64
|
+
return true;
|
|
65
|
+
acc.pop();
|
|
66
|
+
}
|
|
67
|
+
return false;
|
|
68
|
+
}
|
|
69
|
+
get flatRows() {
|
|
70
|
+
const out = [];
|
|
71
|
+
const walk = (nodes, depth) => {
|
|
72
|
+
for (const n of nodes || []) {
|
|
73
|
+
const hasChildren = !!(n.children && n.children.length);
|
|
74
|
+
const expanded = this.expandedIds.has(n.id);
|
|
75
|
+
out.push({
|
|
76
|
+
id: n.id,
|
|
77
|
+
name: n.name,
|
|
78
|
+
depth,
|
|
79
|
+
disabled: n.id === this.currentFolderId,
|
|
80
|
+
hasChildren,
|
|
81
|
+
expanded,
|
|
82
|
+
});
|
|
83
|
+
if (hasChildren && expanded)
|
|
84
|
+
walk(n.children, depth + 1);
|
|
85
|
+
}
|
|
86
|
+
};
|
|
87
|
+
walk(this.folders, 0);
|
|
88
|
+
return out;
|
|
89
|
+
}
|
|
90
|
+
isPicked(id) {
|
|
91
|
+
return this.pickedFolderId === id;
|
|
92
|
+
}
|
|
93
|
+
pick(id) {
|
|
94
|
+
this.pickedFolderId = id;
|
|
95
|
+
this.folderPicked.emit(id);
|
|
96
|
+
this.pickedFolderIdChange.emit(id);
|
|
97
|
+
this.cdr.markForCheck();
|
|
98
|
+
}
|
|
99
|
+
toggleExpanded(id, event) {
|
|
100
|
+
event.stopPropagation();
|
|
101
|
+
if (this.expandedIds.has(id)) {
|
|
102
|
+
this.expandedIds.delete(id);
|
|
103
|
+
}
|
|
104
|
+
else {
|
|
105
|
+
this.expandedIds.add(id);
|
|
106
|
+
}
|
|
107
|
+
this.cdr.markForCheck();
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
MoveToFolderDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: MoveToFolderDialogComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
111
|
+
MoveToFolderDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: MoveToFolderDialogComponent, selector: "cqa-move-to-folder-dialog", inputs: { folders: "folders", labels: "labels", currentFolderId: "currentFolderId", pickedFolderId: "pickedFolderId", initialExpandedIds: "initialExpandedIds" }, outputs: { folderPicked: "folderPicked", pickedFolderIdChange: "pickedFolderIdChange" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: `
|
|
112
|
+
<!-- Reusable outlined folder icon. Stroke uses currentColor so the row's text color drives the icon. -->
|
|
113
|
+
<ng-template #folderIconTpl>
|
|
114
|
+
<svg width="17" height="15" viewBox="0 0 17 15" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true" class="cqa-shrink-0">
|
|
115
|
+
<path d="M15.6375 12.6377C15.6375 13.0355 15.4794 13.4171 15.1981 13.6984C14.9168 13.9797 14.5353 14.1377 14.1375 14.1377H2.13745C1.73963 14.1377 1.3581 13.9797 1.07679 13.6984C0.795486 13.4171 0.637451 13.0355 0.637451 12.6377V2.1377C0.637451 1.73987 0.795486 1.35834 1.07679 1.07704C1.3581 0.795731 1.73963 0.637695 2.13745 0.637695H5.88745L7.38745 2.8877H14.1375C14.5353 2.8877 14.9168 3.04573 15.1981 3.32704C15.4794 3.60834 15.6375 3.98987 15.6375 4.3877V12.6377Z" stroke="currentColor" stroke-width="1.275" stroke-linecap="round" stroke-linejoin="round"/>
|
|
116
|
+
</svg>
|
|
117
|
+
</ng-template>
|
|
118
|
+
|
|
119
|
+
<!-- Reusable outlined archive/inbox icon for the "Unorganised" entry. -->
|
|
120
|
+
<ng-template #unorganisedIconTpl>
|
|
121
|
+
<svg width="17" height="15" viewBox="0 0 17 15" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true" class="cqa-shrink-0">
|
|
122
|
+
<path d="M14.6875 5.5V12.5C14.6875 13.3 14.0375 14 13.1875 14H3.8125C2.9625 14 2.3125 13.3 2.3125 12.5V5.5M14.6875 5.5L12.7875 1.7C12.5375 1.2 12.0875 1 11.5875 1H5.4125C4.9125 1 4.4625 1.2 4.2125 1.7L2.3125 5.5M14.6875 5.5H2.3125M6.1875 8.5H10.8125" stroke="currentColor" stroke-width="1.275" stroke-linecap="round" stroke-linejoin="round"/>
|
|
123
|
+
</svg>
|
|
124
|
+
</ng-template>
|
|
125
|
+
|
|
126
|
+
<div class="cqa-flex cqa-flex-col cqa-border-solid cqa-border cqa-border-[#E2E2E3] cqa-rounded-[10px] cqa-py-2 cqa-w-full cqa-h-[360px] cqa-overflow-y-auto cqa-bg-white cqa-scrollbar-thin">
|
|
127
|
+
<!-- Unorganised (root / no folder) -->
|
|
128
|
+
<div
|
|
129
|
+
role="button"
|
|
130
|
+
tabindex="0"
|
|
131
|
+
class="cqa-flex cqa-items-center cqa-gap-2.5 cqa-px-4 cqa-py-2 cqa-cursor-pointer cqa-transition-colors cqa-text-sm"
|
|
132
|
+
[ngClass]="isPicked(null) ? 'cqa-bg-indigo-50 cqa-text-indigo-700' : 'cqa-text-neutral-800 hover:cqa-bg-neutral-50'"
|
|
133
|
+
(click)="pick(null)"
|
|
134
|
+
(keydown.enter)="pick(null)"
|
|
135
|
+
(keydown.space)="$event.preventDefault(); pick(null)"
|
|
136
|
+
>
|
|
137
|
+
<!-- chevron-slot placeholder so all rows line up -->
|
|
138
|
+
<span class="cqa-inline-block cqa-w-5 cqa-h-5 cqa-shrink-0"></span>
|
|
139
|
+
<ng-container *ngTemplateOutlet="unorganisedIconTpl"></ng-container>
|
|
140
|
+
<span>{{ labels.moveDialogRoot }}</span>
|
|
141
|
+
</div>
|
|
142
|
+
|
|
143
|
+
<!-- Folder rows (flattened, depth-indented, collapsible) -->
|
|
144
|
+
<div
|
|
145
|
+
*ngFor="let row of flatRows; trackBy: trackById"
|
|
146
|
+
role="button"
|
|
147
|
+
tabindex="0"
|
|
148
|
+
class="cqa-flex cqa-items-center cqa-gap-2.5 cqa-py-2 cqa-pr-4 cqa-cursor-pointer cqa-transition-colors cqa-text-sm"
|
|
149
|
+
[ngClass]="isPicked(row.id) ? 'cqa-bg-indigo-50 cqa-text-indigo-700' : 'cqa-text-neutral-800 hover:cqa-bg-neutral-50'"
|
|
150
|
+
[class.cqa-opacity-40]="row.disabled"
|
|
151
|
+
[class.cqa-cursor-not-allowed]="row.disabled"
|
|
152
|
+
[style.paddingLeft.px]="16 + row.depth * 24"
|
|
153
|
+
(click)="!row.disabled && pick(row.id)"
|
|
154
|
+
(keydown.enter)="!row.disabled && pick(row.id)"
|
|
155
|
+
(keydown.space)="!row.disabled && $event.preventDefault(); !row.disabled && pick(row.id)"
|
|
156
|
+
>
|
|
157
|
+
<!-- Chevron toggle for expandable folders (placeholder otherwise so alignment stays) -->
|
|
158
|
+
<button
|
|
159
|
+
*ngIf="row.hasChildren; else chevronPlaceholder"
|
|
160
|
+
type="button"
|
|
161
|
+
class="cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-5 cqa-h-5 cqa-shrink-0 cqa-rounded cqa-text-neutral-600 hover:cqa-bg-neutral-100 hover:cqa-text-neutral-900"
|
|
162
|
+
[attr.aria-label]="row.expanded ? 'Collapse ' + row.name : 'Expand ' + row.name"
|
|
163
|
+
(click)="toggleExpanded(row.id!, $event)"
|
|
164
|
+
>
|
|
165
|
+
<svg width="14" height="14" viewBox="0 0 14 14" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true"
|
|
166
|
+
[class.cqa-rotate-90]="row.expanded" style="transition: transform 120ms ease;">
|
|
167
|
+
<path d="M5 3L9 7L5 11" stroke="currentColor" stroke-width="1.8" stroke-linecap="round" stroke-linejoin="round"/>
|
|
168
|
+
</svg>
|
|
169
|
+
</button>
|
|
170
|
+
<ng-template #chevronPlaceholder>
|
|
171
|
+
<span class="cqa-inline-block cqa-w-5 cqa-h-5 cqa-shrink-0"></span>
|
|
172
|
+
</ng-template>
|
|
173
|
+
|
|
174
|
+
<ng-container *ngTemplateOutlet="folderIconTpl"></ng-container>
|
|
175
|
+
<span class="cqa-truncate">{{ row.name }}</span>
|
|
176
|
+
</div>
|
|
177
|
+
</div>
|
|
178
|
+
`, isInline: true, directives: [{ type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
179
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: MoveToFolderDialogComponent, decorators: [{
|
|
180
|
+
type: Component,
|
|
181
|
+
args: [{ selector: 'cqa-move-to-folder-dialog', template: `
|
|
182
|
+
<!-- Reusable outlined folder icon. Stroke uses currentColor so the row's text color drives the icon. -->
|
|
183
|
+
<ng-template #folderIconTpl>
|
|
184
|
+
<svg width="17" height="15" viewBox="0 0 17 15" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true" class="cqa-shrink-0">
|
|
185
|
+
<path d="M15.6375 12.6377C15.6375 13.0355 15.4794 13.4171 15.1981 13.6984C14.9168 13.9797 14.5353 14.1377 14.1375 14.1377H2.13745C1.73963 14.1377 1.3581 13.9797 1.07679 13.6984C0.795486 13.4171 0.637451 13.0355 0.637451 12.6377V2.1377C0.637451 1.73987 0.795486 1.35834 1.07679 1.07704C1.3581 0.795731 1.73963 0.637695 2.13745 0.637695H5.88745L7.38745 2.8877H14.1375C14.5353 2.8877 14.9168 3.04573 15.1981 3.32704C15.4794 3.60834 15.6375 3.98987 15.6375 4.3877V12.6377Z" stroke="currentColor" stroke-width="1.275" stroke-linecap="round" stroke-linejoin="round"/>
|
|
186
|
+
</svg>
|
|
187
|
+
</ng-template>
|
|
188
|
+
|
|
189
|
+
<!-- Reusable outlined archive/inbox icon for the "Unorganised" entry. -->
|
|
190
|
+
<ng-template #unorganisedIconTpl>
|
|
191
|
+
<svg width="17" height="15" viewBox="0 0 17 15" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true" class="cqa-shrink-0">
|
|
192
|
+
<path d="M14.6875 5.5V12.5C14.6875 13.3 14.0375 14 13.1875 14H3.8125C2.9625 14 2.3125 13.3 2.3125 12.5V5.5M14.6875 5.5L12.7875 1.7C12.5375 1.2 12.0875 1 11.5875 1H5.4125C4.9125 1 4.4625 1.2 4.2125 1.7L2.3125 5.5M14.6875 5.5H2.3125M6.1875 8.5H10.8125" stroke="currentColor" stroke-width="1.275" stroke-linecap="round" stroke-linejoin="round"/>
|
|
193
|
+
</svg>
|
|
194
|
+
</ng-template>
|
|
195
|
+
|
|
196
|
+
<div class="cqa-flex cqa-flex-col cqa-border-solid cqa-border cqa-border-[#E2E2E3] cqa-rounded-[10px] cqa-py-2 cqa-w-full cqa-h-[360px] cqa-overflow-y-auto cqa-bg-white cqa-scrollbar-thin">
|
|
197
|
+
<!-- Unorganised (root / no folder) -->
|
|
198
|
+
<div
|
|
199
|
+
role="button"
|
|
200
|
+
tabindex="0"
|
|
201
|
+
class="cqa-flex cqa-items-center cqa-gap-2.5 cqa-px-4 cqa-py-2 cqa-cursor-pointer cqa-transition-colors cqa-text-sm"
|
|
202
|
+
[ngClass]="isPicked(null) ? 'cqa-bg-indigo-50 cqa-text-indigo-700' : 'cqa-text-neutral-800 hover:cqa-bg-neutral-50'"
|
|
203
|
+
(click)="pick(null)"
|
|
204
|
+
(keydown.enter)="pick(null)"
|
|
205
|
+
(keydown.space)="$event.preventDefault(); pick(null)"
|
|
206
|
+
>
|
|
207
|
+
<!-- chevron-slot placeholder so all rows line up -->
|
|
208
|
+
<span class="cqa-inline-block cqa-w-5 cqa-h-5 cqa-shrink-0"></span>
|
|
209
|
+
<ng-container *ngTemplateOutlet="unorganisedIconTpl"></ng-container>
|
|
210
|
+
<span>{{ labels.moveDialogRoot }}</span>
|
|
211
|
+
</div>
|
|
212
|
+
|
|
213
|
+
<!-- Folder rows (flattened, depth-indented, collapsible) -->
|
|
214
|
+
<div
|
|
215
|
+
*ngFor="let row of flatRows; trackBy: trackById"
|
|
216
|
+
role="button"
|
|
217
|
+
tabindex="0"
|
|
218
|
+
class="cqa-flex cqa-items-center cqa-gap-2.5 cqa-py-2 cqa-pr-4 cqa-cursor-pointer cqa-transition-colors cqa-text-sm"
|
|
219
|
+
[ngClass]="isPicked(row.id) ? 'cqa-bg-indigo-50 cqa-text-indigo-700' : 'cqa-text-neutral-800 hover:cqa-bg-neutral-50'"
|
|
220
|
+
[class.cqa-opacity-40]="row.disabled"
|
|
221
|
+
[class.cqa-cursor-not-allowed]="row.disabled"
|
|
222
|
+
[style.paddingLeft.px]="16 + row.depth * 24"
|
|
223
|
+
(click)="!row.disabled && pick(row.id)"
|
|
224
|
+
(keydown.enter)="!row.disabled && pick(row.id)"
|
|
225
|
+
(keydown.space)="!row.disabled && $event.preventDefault(); !row.disabled && pick(row.id)"
|
|
226
|
+
>
|
|
227
|
+
<!-- Chevron toggle for expandable folders (placeholder otherwise so alignment stays) -->
|
|
228
|
+
<button
|
|
229
|
+
*ngIf="row.hasChildren; else chevronPlaceholder"
|
|
230
|
+
type="button"
|
|
231
|
+
class="cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-5 cqa-h-5 cqa-shrink-0 cqa-rounded cqa-text-neutral-600 hover:cqa-bg-neutral-100 hover:cqa-text-neutral-900"
|
|
232
|
+
[attr.aria-label]="row.expanded ? 'Collapse ' + row.name : 'Expand ' + row.name"
|
|
233
|
+
(click)="toggleExpanded(row.id!, $event)"
|
|
234
|
+
>
|
|
235
|
+
<svg width="14" height="14" viewBox="0 0 14 14" fill="none" xmlns="http://www.w3.org/2000/svg" aria-hidden="true"
|
|
236
|
+
[class.cqa-rotate-90]="row.expanded" style="transition: transform 120ms ease;">
|
|
237
|
+
<path d="M5 3L9 7L5 11" stroke="currentColor" stroke-width="1.8" stroke-linecap="round" stroke-linejoin="round"/>
|
|
238
|
+
</svg>
|
|
239
|
+
</button>
|
|
240
|
+
<ng-template #chevronPlaceholder>
|
|
241
|
+
<span class="cqa-inline-block cqa-w-5 cqa-h-5 cqa-shrink-0"></span>
|
|
242
|
+
</ng-template>
|
|
243
|
+
|
|
244
|
+
<ng-container *ngTemplateOutlet="folderIconTpl"></ng-container>
|
|
245
|
+
<span class="cqa-truncate">{{ row.name }}</span>
|
|
246
|
+
</div>
|
|
247
|
+
</div>
|
|
248
|
+
`, host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.OnPush, styles: [] }]
|
|
249
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { folders: [{
|
|
250
|
+
type: Input
|
|
251
|
+
}], labels: [{
|
|
252
|
+
type: Input
|
|
253
|
+
}], currentFolderId: [{
|
|
254
|
+
type: Input
|
|
255
|
+
}], pickedFolderId: [{
|
|
256
|
+
type: Input
|
|
257
|
+
}], initialExpandedIds: [{
|
|
258
|
+
type: Input
|
|
259
|
+
}], folderPicked: [{
|
|
260
|
+
type: Output
|
|
261
|
+
}], pickedFolderIdChange: [{
|
|
262
|
+
type: Output
|
|
263
|
+
}] } });
|
|
264
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"move-to-folder-dialog.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/templates/modular-table-template/dialogs/move-to-folder-dialog.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n} from '@angular/core';\nimport { DEFAULT_MODULAR_LABELS, FolderNode, ModularLabels } from '../modular-table-template.models';\n\ninterface FlatFolderRow {\n  id: number | null;\n  name: string;\n  depth: number;\n  disabled: boolean;\n  hasChildren: boolean;\n  expanded: boolean;\n}\n\n/**\n * Folder-picker body used inside the Move-to-Folder dialog.\n *\n * This component is designed to be usable two ways:\n *\n *  1. **From inside the library** via `DialogService.open({ content: { type: 'component', component: MoveToFolderDialogComponent, inputs: {...} } })`.\n *     The outer `cqa-dialog` provides the title, description, Cancel and \"Move here\"\n *     buttons; the Move button reads `pickedFolderId` via `dialogRef.getComponentInstance()`.\n *\n *  2. **From the host/UI layer directly** once API ownership moves up. The host\n *     renders `<cqa-move-to-folder-dialog>` inside its own modal shell and\n *     listens to `(folderPicked)` (fires on every pick) to drive its own \"Move here\"\n *     button. All visual state lives on `pickedFolderId` so hosts can also two-way\n *     bind if they prefer.\n */\n@Component({\n  selector: 'cqa-move-to-folder-dialog',\n  template: `\n    <!-- Reusable outlined folder icon. Stroke uses currentColor so the row's text color drives the icon. -->\n    <ng-template #folderIconTpl>\n      <svg width=\"17\" height=\"15\" viewBox=\"0 0 17 15\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\" class=\"cqa-shrink-0\">\n        <path d=\"M15.6375 12.6377C15.6375 13.0355 15.4794 13.4171 15.1981 13.6984C14.9168 13.9797 14.5353 14.1377 14.1375 14.1377H2.13745C1.73963 14.1377 1.3581 13.9797 1.07679 13.6984C0.795486 13.4171 0.637451 13.0355 0.637451 12.6377V2.1377C0.637451 1.73987 0.795486 1.35834 1.07679 1.07704C1.3581 0.795731 1.73963 0.637695 2.13745 0.637695H5.88745L7.38745 2.8877H14.1375C14.5353 2.8877 14.9168 3.04573 15.1981 3.32704C15.4794 3.60834 15.6375 3.98987 15.6375 4.3877V12.6377Z\" stroke=\"currentColor\" stroke-width=\"1.275\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n      </svg>\n    </ng-template>\n\n    <!-- Reusable outlined archive/inbox icon for the \"Unorganised\" entry. -->\n    <ng-template #unorganisedIconTpl>\n      <svg width=\"17\" height=\"15\" viewBox=\"0 0 17 15\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\" class=\"cqa-shrink-0\">\n        <path d=\"M14.6875 5.5V12.5C14.6875 13.3 14.0375 14 13.1875 14H3.8125C2.9625 14 2.3125 13.3 2.3125 12.5V5.5M14.6875 5.5L12.7875 1.7C12.5375 1.2 12.0875 1 11.5875 1H5.4125C4.9125 1 4.4625 1.2 4.2125 1.7L2.3125 5.5M14.6875 5.5H2.3125M6.1875 8.5H10.8125\" stroke=\"currentColor\" stroke-width=\"1.275\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n      </svg>\n    </ng-template>\n\n    <div class=\"cqa-flex cqa-flex-col cqa-border-solid cqa-border cqa-border-[#E2E2E3] cqa-rounded-[10px] cqa-py-2 cqa-w-full cqa-h-[360px] cqa-overflow-y-auto cqa-bg-white cqa-scrollbar-thin\">\n      <!-- Unorganised (root / no folder) -->\n      <div\n        role=\"button\"\n        tabindex=\"0\"\n        class=\"cqa-flex cqa-items-center cqa-gap-2.5 cqa-px-4 cqa-py-2 cqa-cursor-pointer cqa-transition-colors cqa-text-sm\"\n        [ngClass]=\"isPicked(null) ? 'cqa-bg-indigo-50 cqa-text-indigo-700' : 'cqa-text-neutral-800 hover:cqa-bg-neutral-50'\"\n        (click)=\"pick(null)\"\n        (keydown.enter)=\"pick(null)\"\n        (keydown.space)=\"$event.preventDefault(); pick(null)\"\n      >\n        <!-- chevron-slot placeholder so all rows line up -->\n        <span class=\"cqa-inline-block cqa-w-5 cqa-h-5 cqa-shrink-0\"></span>\n        <ng-container *ngTemplateOutlet=\"unorganisedIconTpl\"></ng-container>\n        <span>{{ labels.moveDialogRoot }}</span>\n      </div>\n\n      <!-- Folder rows (flattened, depth-indented, collapsible) -->\n      <div\n        *ngFor=\"let row of flatRows; trackBy: trackById\"\n        role=\"button\"\n        tabindex=\"0\"\n        class=\"cqa-flex cqa-items-center cqa-gap-2.5 cqa-py-2 cqa-pr-4 cqa-cursor-pointer cqa-transition-colors cqa-text-sm\"\n        [ngClass]=\"isPicked(row.id) ? 'cqa-bg-indigo-50 cqa-text-indigo-700' : 'cqa-text-neutral-800 hover:cqa-bg-neutral-50'\"\n        [class.cqa-opacity-40]=\"row.disabled\"\n        [class.cqa-cursor-not-allowed]=\"row.disabled\"\n        [style.paddingLeft.px]=\"16 + row.depth * 24\"\n        (click)=\"!row.disabled && pick(row.id)\"\n        (keydown.enter)=\"!row.disabled && pick(row.id)\"\n        (keydown.space)=\"!row.disabled && $event.preventDefault(); !row.disabled && pick(row.id)\"\n      >\n        <!-- Chevron toggle for expandable folders (placeholder otherwise so alignment stays) -->\n        <button\n          *ngIf=\"row.hasChildren; else chevronPlaceholder\"\n          type=\"button\"\n          class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-5 cqa-h-5 cqa-shrink-0 cqa-rounded cqa-text-neutral-600 hover:cqa-bg-neutral-100 hover:cqa-text-neutral-900\"\n          [attr.aria-label]=\"row.expanded ? 'Collapse ' + row.name : 'Expand ' + row.name\"\n          (click)=\"toggleExpanded(row.id!, $event)\"\n        >\n          <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\"\n               [class.cqa-rotate-90]=\"row.expanded\" style=\"transition: transform 120ms ease;\">\n            <path d=\"M5 3L9 7L5 11\" stroke=\"currentColor\" stroke-width=\"1.8\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n          </svg>\n        </button>\n        <ng-template #chevronPlaceholder>\n          <span class=\"cqa-inline-block cqa-w-5 cqa-h-5 cqa-shrink-0\"></span>\n        </ng-template>\n\n        <ng-container *ngTemplateOutlet=\"folderIconTpl\"></ng-container>\n        <span class=\"cqa-truncate\">{{ row.name }}</span>\n      </div>\n    </div>\n  `,\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class MoveToFolderDialogComponent implements OnInit, OnChanges {\n  /** Folder tree to pick from. */\n  @Input() folders: FolderNode[] = [];\n  /** Label overrides. */\n  @Input() labels: ModularLabels = { ...DEFAULT_MODULAR_LABELS };\n  /** The folder the tests currently live in — rendered as disabled (can't move into itself). */\n  @Input() currentFolderId: number | null = null;\n  /** Controls which folder is visually selected. Two-way-bind friendly via `pickedFolderIdChange`. */\n  @Input() pickedFolderId: number | null = null;\n  /** Folder ids to expand by default. If not provided, ancestors of `currentFolderId` are expanded. */\n  @Input() initialExpandedIds?: number[];\n\n  /** Fires whenever the user picks a destination. */\n  @Output() folderPicked = new EventEmitter<number | null>();\n  /** Companion output for banana-in-the-box `[(pickedFolderId)]` binding. */\n  @Output() pickedFolderIdChange = new EventEmitter<number | null>();\n\n  private expandedIds = new Set<number>();\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    this.seedExpandedIds();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['initialExpandedIds'] || changes['currentFolderId'] || changes['folders']) {\n      this.seedExpandedIds();\n    }\n  }\n\n  private seedExpandedIds(): void {\n    if (this.initialExpandedIds?.length) {\n      this.expandedIds = new Set(this.initialExpandedIds);\n      return;\n    }\n    this.expandedIds = new Set();\n    if (this.currentFolderId != null) {\n      const trail: number[] = [];\n      this.collectAncestors(this.folders, this.currentFolderId, trail);\n      trail.forEach(id => this.expandedIds.add(id));\n    }\n  }\n\n  private collectAncestors(nodes: FolderNode[], targetId: number, acc: number[]): boolean {\n    for (const n of nodes || []) {\n      acc.push(n.id);\n      if (n.id === targetId) return true;\n      if (n.children && this.collectAncestors(n.children, targetId, acc)) return true;\n      acc.pop();\n    }\n    return false;\n  }\n\n  get flatRows(): FlatFolderRow[] {\n    const out: FlatFolderRow[] = [];\n    const walk = (nodes: FolderNode[], depth: number) => {\n      for (const n of nodes || []) {\n        const hasChildren = !!(n.children && n.children.length);\n        const expanded = this.expandedIds.has(n.id);\n        out.push({\n          id: n.id,\n          name: n.name,\n          depth,\n          disabled: n.id === this.currentFolderId,\n          hasChildren,\n          expanded,\n        });\n        if (hasChildren && expanded) walk(n.children!, depth + 1);\n      }\n    };\n    walk(this.folders, 0);\n    return out;\n  }\n\n  isPicked(id: number | null): boolean {\n    return this.pickedFolderId === id;\n  }\n\n  pick(id: number | null): void {\n    this.pickedFolderId = id;\n    this.folderPicked.emit(id);\n    this.pickedFolderIdChange.emit(id);\n    this.cdr.markForCheck();\n  }\n\n  toggleExpanded(id: number, event: Event): void {\n    event.stopPropagation();\n    if (this.expandedIds.has(id)) {\n      this.expandedIds.delete(id);\n    } else {\n      this.expandedIds.add(id);\n    }\n    this.cdr.markForCheck();\n  }\n\n  trackById = (_: number, row: FlatFolderRow) => row.id ?? -1;\n}\n"]}
|