@cqa-lib/cqa-ui 1.1.554 → 1.1.555
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/test-suite/move-test-suite-dialog/move-test-suite-dialog.component.mjs +307 -0
- package/esm2020/lib/test-suite/test-suite.models.mjs +11 -0
- package/esm2020/lib/ui-kit.module.mjs +6 -1
- package/esm2020/public-api.mjs +3 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +319 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +317 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/test-suite/move-test-suite-dialog/move-test-suite-dialog.component.d.ts +106 -0
- package/lib/test-suite/test-suite.models.d.ts +19 -0
- package/lib/ui-kit.module.d.ts +156 -155
- package/package.json +1 -1
- package/public-api.d.ts +2 -0
|
@@ -0,0 +1,307 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, } from '@angular/core';
|
|
2
|
+
import { DEFAULT_MODULAR_LABELS, } from '../../templates/modular-table-template/modular-table-template.models';
|
|
3
|
+
import { DEFAULT_TEST_SUITE_LABELS } from '../test-suite.models';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "../../custom-input/custom-input.component";
|
|
6
|
+
import * as i2 from "../../templates/modular-table-template/dialogs/move-to-folder-dialog.component";
|
|
7
|
+
/**
|
|
8
|
+
* Body of the "Move Test Suite" dialog.
|
|
9
|
+
*
|
|
10
|
+
* Reuses `<cqa-move-to-folder-dialog>` as the tree picker (same embedding pattern
|
|
11
|
+
* as `NewFolderDialogComponent`) so consumers get the same folder-tree UX, while
|
|
12
|
+
* adding two read-only rows above it: the suite's **current folder** (source)
|
|
13
|
+
* and the **destination folder** the user has picked (live-updates).
|
|
14
|
+
*
|
|
15
|
+
* Two integration paths:
|
|
16
|
+
*
|
|
17
|
+
* 1. **From within this library** via `DialogService.open(...)`. The outer
|
|
18
|
+
* `cqa-dialog` supplies title/description/Cancel/Move buttons; the Move
|
|
19
|
+
* button reads `pickedFolderId` via `dialogRef.getComponentInstance()` and
|
|
20
|
+
* checks `isValid` before closing.
|
|
21
|
+
*
|
|
22
|
+
* 2. **Host-driven** — host renders `<cqa-move-test-suite-dialog>` inside its
|
|
23
|
+
* own modal and listens to `(submitted)` / `(cancelled)` / `(pickedFolderIdChange)`.
|
|
24
|
+
*/
|
|
25
|
+
export class MoveTestSuiteDialogComponent {
|
|
26
|
+
constructor(cdr) {
|
|
27
|
+
this.cdr = cdr;
|
|
28
|
+
// NOTE: `folders` and `labels` are declared as setters (instead of plain @Input fields)
|
|
29
|
+
// because `DialogService.open(...)` assigns the `inputs` map via direct property
|
|
30
|
+
// assignment *after* `attachComponent` has already run the component's lifecycle
|
|
31
|
+
// hooks — direct assignment bypasses `ngOnChanges`, so setters keep derived state
|
|
32
|
+
// in sync no matter the assignment order. Same pattern as `NewFolderDialogComponent`.
|
|
33
|
+
this._folders = [];
|
|
34
|
+
this._labels = { ...DEFAULT_MODULAR_LABELS };
|
|
35
|
+
/** Test-suite-section labels (title, description, button text, field labels,
|
|
36
|
+
* placeholder). Kept separate from `labels` (ModularLabels) so the embedded
|
|
37
|
+
* picker can keep its modular-table-template label model. Same setter
|
|
38
|
+
* pattern as `labels` for DialogService property-assignment compatibility. */
|
|
39
|
+
this._testSuiteLabels = { ...DEFAULT_TEST_SUITE_LABELS };
|
|
40
|
+
/** Folder the suite currently lives in. Used both to gray out the row in the
|
|
41
|
+
* embedded picker and to render the "Current folder" readout. */
|
|
42
|
+
this.currentFolderId = null;
|
|
43
|
+
/** Optional override for the current-folder display name. When not provided,
|
|
44
|
+
* the component resolves the name from `folders` by walking to `currentFolderId`,
|
|
45
|
+
* falling back to `labels.moveDialogRoot` ("Unorganized") when null. */
|
|
46
|
+
this.currentFolderName = null;
|
|
47
|
+
/** Name of the test suite being moved. Rendered as a disabled input at the
|
|
48
|
+
* top of the dialog body so the user can confirm which suite the move
|
|
49
|
+
* applies to. Defaults to empty string. */
|
|
50
|
+
this.suiteName = '';
|
|
51
|
+
/** Destination folder id picked by the user. Two-way-bind friendly. */
|
|
52
|
+
this.pickedFolderId = null;
|
|
53
|
+
/** Height of the embedded picker. Defaults to 240px (matches the parent-folder
|
|
54
|
+
* picker height in `NewFolderDialogComponent`) so the modal stays compact. */
|
|
55
|
+
this.pickerHeight = '240px';
|
|
56
|
+
/** When true (default), both the Current folder and Destination folder inputs
|
|
57
|
+
* render as disabled (read-only via the cqa-custom-input `[disabled]` flag).
|
|
58
|
+
* Set to false if a host needs to re-enable interaction on those fields. */
|
|
59
|
+
this.fieldsDisabled = true;
|
|
60
|
+
this.pickedFolderIdChange = new EventEmitter();
|
|
61
|
+
this.submitted = new EventEmitter();
|
|
62
|
+
this.cancelled = new EventEmitter();
|
|
63
|
+
/** Bubble cqa-custom-input focus/blur events up to hosts. Useful when
|
|
64
|
+
* `fieldsDisabled = false` and the host wants to react to interaction. */
|
|
65
|
+
this.currentFieldFocus = new EventEmitter();
|
|
66
|
+
this.currentFieldBlur = new EventEmitter();
|
|
67
|
+
this.destinationFieldFocus = new EventEmitter();
|
|
68
|
+
this.destinationFieldBlur = new EventEmitter();
|
|
69
|
+
/** Set to true the first time the user picks anything. Lets `isValid` treat
|
|
70
|
+
* the initial pre-touch state as "nothing selected" — important when the
|
|
71
|
+
* suite starts in Unorganized (`currentFolderId = null`) so a fresh dialog
|
|
72
|
+
* doesn't appear valid just because `pickedFolderId` also defaults to null. */
|
|
73
|
+
this.pickedTouched = false;
|
|
74
|
+
}
|
|
75
|
+
set folders(value) {
|
|
76
|
+
this._folders = value || [];
|
|
77
|
+
this.cdr.markForCheck();
|
|
78
|
+
}
|
|
79
|
+
get folders() { return this._folders; }
|
|
80
|
+
set labels(value) {
|
|
81
|
+
this._labels = value || { ...DEFAULT_MODULAR_LABELS };
|
|
82
|
+
this.cdr.markForCheck();
|
|
83
|
+
}
|
|
84
|
+
get labels() { return this._labels; }
|
|
85
|
+
set testSuiteLabels(value) {
|
|
86
|
+
this._testSuiteLabels = value || { ...DEFAULT_TEST_SUITE_LABELS };
|
|
87
|
+
this.cdr.markForCheck();
|
|
88
|
+
}
|
|
89
|
+
get testSuiteLabels() { return this._testSuiteLabels; }
|
|
90
|
+
onParentPicked(id) {
|
|
91
|
+
let next = null;
|
|
92
|
+
if (id != null) {
|
|
93
|
+
const coerced = typeof id === 'number' ? id : Number(id);
|
|
94
|
+
if (Number.isFinite(coerced))
|
|
95
|
+
next = coerced;
|
|
96
|
+
}
|
|
97
|
+
this.pickedFolderId = next;
|
|
98
|
+
this.pickedTouched = true;
|
|
99
|
+
this.pickedFolderIdChange.emit(next);
|
|
100
|
+
this.cdr.markForCheck();
|
|
101
|
+
}
|
|
102
|
+
/** True once the user has picked a destination via the embedded tree. Drives
|
|
103
|
+
* the destination-readout styling (indigo when picked, muted when not). */
|
|
104
|
+
get hasPick() {
|
|
105
|
+
return this.pickedTouched;
|
|
106
|
+
}
|
|
107
|
+
get resolvedCurrentName() {
|
|
108
|
+
if (this.currentFolderName && this.currentFolderName.trim().length) {
|
|
109
|
+
return this.currentFolderName;
|
|
110
|
+
}
|
|
111
|
+
if (this.currentFolderId == null) {
|
|
112
|
+
return this.labels.moveDialogRoot;
|
|
113
|
+
}
|
|
114
|
+
const name = this.findFolderName(this._folders, this.currentFolderId);
|
|
115
|
+
return name ?? this.labels.moveDialogRoot;
|
|
116
|
+
}
|
|
117
|
+
/** Value shown inside the Destination input. Empty until the user picks; once
|
|
118
|
+
* picked, the resolved folder name (or `labels.moveDialogRoot` for Unorganized).
|
|
119
|
+
* Empty state lets the input's `[placeholder]` show through. */
|
|
120
|
+
get destinationFieldValue() {
|
|
121
|
+
if (!this.pickedTouched)
|
|
122
|
+
return '';
|
|
123
|
+
if (this.pickedFolderId == null)
|
|
124
|
+
return this.labels.moveDialogRoot;
|
|
125
|
+
return this.findFolderName(this._folders, this.pickedFolderId) ?? this.labels.moveDialogRoot;
|
|
126
|
+
}
|
|
127
|
+
/** Placeholder shown in the Destination input before the user picks. */
|
|
128
|
+
get destinationFieldPlaceholder() {
|
|
129
|
+
return this._testSuiteLabels.moveTestSuiteDialogDestinationPlaceholder;
|
|
130
|
+
}
|
|
131
|
+
/** Move is valid when the user has picked a destination different from the
|
|
132
|
+
* current folder. Picking the same folder as the source is a no-op, and we
|
|
133
|
+
* also require an explicit interaction (`pickedTouched`). */
|
|
134
|
+
get isValid() {
|
|
135
|
+
if (!this.pickedTouched)
|
|
136
|
+
return false;
|
|
137
|
+
return this.pickedFolderId !== this.currentFolderId;
|
|
138
|
+
}
|
|
139
|
+
submit() {
|
|
140
|
+
if (!this.isValid)
|
|
141
|
+
return;
|
|
142
|
+
this.submitted.emit({ targetFolderId: this.pickedFolderId });
|
|
143
|
+
}
|
|
144
|
+
cancel() {
|
|
145
|
+
this.cancelled.emit();
|
|
146
|
+
}
|
|
147
|
+
/** Hooks for the Current folder cqa-custom-input. Field is disabled by
|
|
148
|
+
* default, so these mostly no-op; they're wired so the component still
|
|
149
|
+
* behaves correctly when `fieldsDisabled = false`. */
|
|
150
|
+
onCurrentFieldValueChange(_value) {
|
|
151
|
+
// Read-only field — ignore programmatic changes.
|
|
152
|
+
}
|
|
153
|
+
onCurrentFieldFocused(event) {
|
|
154
|
+
this.currentFieldFocus.emit(event);
|
|
155
|
+
}
|
|
156
|
+
onCurrentFieldBlurred(event) {
|
|
157
|
+
this.currentFieldBlur.emit(event);
|
|
158
|
+
}
|
|
159
|
+
/** Hooks for the Destination folder cqa-custom-input. Value is driven by the
|
|
160
|
+
* tree picker below, so the input itself is informational. */
|
|
161
|
+
onDestinationFieldValueChange(_value) {
|
|
162
|
+
// Destination is driven by the tree picker, not the input itself.
|
|
163
|
+
}
|
|
164
|
+
onDestinationFieldFocused(event) {
|
|
165
|
+
this.destinationFieldFocus.emit(event);
|
|
166
|
+
}
|
|
167
|
+
onDestinationFieldBlurred(event) {
|
|
168
|
+
this.destinationFieldBlur.emit(event);
|
|
169
|
+
}
|
|
170
|
+
findFolderName(nodes, id) {
|
|
171
|
+
for (const n of nodes || []) {
|
|
172
|
+
if (n.id === id)
|
|
173
|
+
return n.name;
|
|
174
|
+
if (n.children?.length) {
|
|
175
|
+
const hit = this.findFolderName(n.children, id);
|
|
176
|
+
if (hit != null)
|
|
177
|
+
return hit;
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
return null;
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
MoveTestSuiteDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: MoveTestSuiteDialogComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
184
|
+
MoveTestSuiteDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: MoveTestSuiteDialogComponent, selector: "cqa-move-test-suite-dialog", inputs: { folders: "folders", labels: "labels", testSuiteLabels: "testSuiteLabels", currentFolderId: "currentFolderId", currentFolderName: "currentFolderName", suiteName: "suiteName", pickedFolderId: "pickedFolderId", pickerHeight: "pickerHeight", fieldsDisabled: "fieldsDisabled" }, outputs: { pickedFolderIdChange: "pickedFolderIdChange", submitted: "submitted", cancelled: "cancelled", currentFieldFocus: "currentFieldFocus", currentFieldBlur: "currentFieldBlur", destinationFieldFocus: "destinationFieldFocus", destinationFieldBlur: "destinationFieldBlur" }, host: { classAttribute: "cqa-ui-root" }, ngImport: i0, template: `
|
|
185
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-2 cqa-w-full">
|
|
186
|
+
<!-- Test suite name (read-only display; surfaces which suite is being moved) -->
|
|
187
|
+
<cqa-custom-input
|
|
188
|
+
[label]="testSuiteLabels.moveTestSuiteDialogSuiteNameLabel"
|
|
189
|
+
[value]="suiteName"
|
|
190
|
+
[disabled]="fieldsDisabled"
|
|
191
|
+
[fullWidth]="true"
|
|
192
|
+
></cqa-custom-input>
|
|
193
|
+
|
|
194
|
+
<!-- Current folder (rendered as a disabled input) -->
|
|
195
|
+
<cqa-custom-input
|
|
196
|
+
[label]="testSuiteLabels.moveTestSuiteDialogCurrentLabel"
|
|
197
|
+
[value]="resolvedCurrentName"
|
|
198
|
+
[disabled]="fieldsDisabled"
|
|
199
|
+
[fullWidth]="true"
|
|
200
|
+
(valueChange)="onCurrentFieldValueChange($event)"
|
|
201
|
+
(focused)="onCurrentFieldFocused($event)"
|
|
202
|
+
(blurred)="onCurrentFieldBlurred($event)"
|
|
203
|
+
></cqa-custom-input>
|
|
204
|
+
|
|
205
|
+
<!-- Destination folder (rendered as a disabled input; value driven by the tree picker below) -->
|
|
206
|
+
<cqa-custom-input
|
|
207
|
+
[label]="testSuiteLabels.moveTestSuiteDialogDestinationLabel"
|
|
208
|
+
[value]="destinationFieldValue"
|
|
209
|
+
[placeholder]="destinationFieldPlaceholder"
|
|
210
|
+
[disabled]="fieldsDisabled"
|
|
211
|
+
[fullWidth]="true"
|
|
212
|
+
(valueChange)="onDestinationFieldValueChange($event)"
|
|
213
|
+
(focused)="onDestinationFieldFocused($event)"
|
|
214
|
+
(blurred)="onDestinationFieldBlurred($event)"
|
|
215
|
+
></cqa-custom-input>
|
|
216
|
+
|
|
217
|
+
<!-- Folder tree picker -->
|
|
218
|
+
<cqa-move-to-folder-dialog
|
|
219
|
+
[folders]="folders"
|
|
220
|
+
[labels]="labels"
|
|
221
|
+
[currentFolderId]="currentFolderId"
|
|
222
|
+
[pickedFolderId]="pickedFolderId"
|
|
223
|
+
[pickerHeight]="pickerHeight"
|
|
224
|
+
(pickedFolderIdChange)="onParentPicked($event)"
|
|
225
|
+
></cqa-move-to-folder-dialog>
|
|
226
|
+
</div>
|
|
227
|
+
`, isInline: true, components: [{ type: i1.CustomInputComponent, selector: "cqa-custom-input", inputs: ["inputId", "label", "type", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "inputInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused", "enterPressed"] }, { type: i2.MoveToFolderDialogComponent, selector: "cqa-move-to-folder-dialog", inputs: ["folders", "labels", "currentFolderId", "pickedFolderId", "initialExpandedIds", "rootDisabled", "pickerHeight"], outputs: ["folderPicked", "pickedFolderIdChange"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
228
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: MoveTestSuiteDialogComponent, decorators: [{
|
|
229
|
+
type: Component,
|
|
230
|
+
args: [{ selector: 'cqa-move-test-suite-dialog', template: `
|
|
231
|
+
<div class="cqa-flex cqa-flex-col cqa-gap-2 cqa-w-full">
|
|
232
|
+
<!-- Test suite name (read-only display; surfaces which suite is being moved) -->
|
|
233
|
+
<cqa-custom-input
|
|
234
|
+
[label]="testSuiteLabels.moveTestSuiteDialogSuiteNameLabel"
|
|
235
|
+
[value]="suiteName"
|
|
236
|
+
[disabled]="fieldsDisabled"
|
|
237
|
+
[fullWidth]="true"
|
|
238
|
+
></cqa-custom-input>
|
|
239
|
+
|
|
240
|
+
<!-- Current folder (rendered as a disabled input) -->
|
|
241
|
+
<cqa-custom-input
|
|
242
|
+
[label]="testSuiteLabels.moveTestSuiteDialogCurrentLabel"
|
|
243
|
+
[value]="resolvedCurrentName"
|
|
244
|
+
[disabled]="fieldsDisabled"
|
|
245
|
+
[fullWidth]="true"
|
|
246
|
+
(valueChange)="onCurrentFieldValueChange($event)"
|
|
247
|
+
(focused)="onCurrentFieldFocused($event)"
|
|
248
|
+
(blurred)="onCurrentFieldBlurred($event)"
|
|
249
|
+
></cqa-custom-input>
|
|
250
|
+
|
|
251
|
+
<!-- Destination folder (rendered as a disabled input; value driven by the tree picker below) -->
|
|
252
|
+
<cqa-custom-input
|
|
253
|
+
[label]="testSuiteLabels.moveTestSuiteDialogDestinationLabel"
|
|
254
|
+
[value]="destinationFieldValue"
|
|
255
|
+
[placeholder]="destinationFieldPlaceholder"
|
|
256
|
+
[disabled]="fieldsDisabled"
|
|
257
|
+
[fullWidth]="true"
|
|
258
|
+
(valueChange)="onDestinationFieldValueChange($event)"
|
|
259
|
+
(focused)="onDestinationFieldFocused($event)"
|
|
260
|
+
(blurred)="onDestinationFieldBlurred($event)"
|
|
261
|
+
></cqa-custom-input>
|
|
262
|
+
|
|
263
|
+
<!-- Folder tree picker -->
|
|
264
|
+
<cqa-move-to-folder-dialog
|
|
265
|
+
[folders]="folders"
|
|
266
|
+
[labels]="labels"
|
|
267
|
+
[currentFolderId]="currentFolderId"
|
|
268
|
+
[pickedFolderId]="pickedFolderId"
|
|
269
|
+
[pickerHeight]="pickerHeight"
|
|
270
|
+
(pickedFolderIdChange)="onParentPicked($event)"
|
|
271
|
+
></cqa-move-to-folder-dialog>
|
|
272
|
+
</div>
|
|
273
|
+
`, host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.OnPush, styles: [] }]
|
|
274
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { folders: [{
|
|
275
|
+
type: Input
|
|
276
|
+
}], labels: [{
|
|
277
|
+
type: Input
|
|
278
|
+
}], testSuiteLabels: [{
|
|
279
|
+
type: Input
|
|
280
|
+
}], currentFolderId: [{
|
|
281
|
+
type: Input
|
|
282
|
+
}], currentFolderName: [{
|
|
283
|
+
type: Input
|
|
284
|
+
}], suiteName: [{
|
|
285
|
+
type: Input
|
|
286
|
+
}], pickedFolderId: [{
|
|
287
|
+
type: Input
|
|
288
|
+
}], pickerHeight: [{
|
|
289
|
+
type: Input
|
|
290
|
+
}], fieldsDisabled: [{
|
|
291
|
+
type: Input
|
|
292
|
+
}], pickedFolderIdChange: [{
|
|
293
|
+
type: Output
|
|
294
|
+
}], submitted: [{
|
|
295
|
+
type: Output
|
|
296
|
+
}], cancelled: [{
|
|
297
|
+
type: Output
|
|
298
|
+
}], currentFieldFocus: [{
|
|
299
|
+
type: Output
|
|
300
|
+
}], currentFieldBlur: [{
|
|
301
|
+
type: Output
|
|
302
|
+
}], destinationFieldFocus: [{
|
|
303
|
+
type: Output
|
|
304
|
+
}], destinationFieldBlur: [{
|
|
305
|
+
type: Output
|
|
306
|
+
}] } });
|
|
307
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"move-test-suite-dialog.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-suite/move-test-suite-dialog/move-test-suite-dialog.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  Output,\n} from '@angular/core';\nimport {\n  DEFAULT_MODULAR_LABELS,\n  FolderNode,\n  ModularLabels,\n} from '../../templates/modular-table-template/modular-table-template.models';\nimport { DEFAULT_TEST_SUITE_LABELS, TestSuiteLabels } from '../test-suite.models';\n\n/**\n * Body of the \"Move Test Suite\" dialog.\n *\n * Reuses `<cqa-move-to-folder-dialog>` as the tree picker (same embedding pattern\n * as `NewFolderDialogComponent`) so consumers get the same folder-tree UX, while\n * adding two read-only rows above it: the suite's **current folder** (source)\n * and the **destination folder** the user has picked (live-updates).\n *\n * Two integration paths:\n *\n *  1. **From within this library** via `DialogService.open(...)`. The outer\n *     `cqa-dialog` supplies title/description/Cancel/Move buttons; the Move\n *     button reads `pickedFolderId` via `dialogRef.getComponentInstance()` and\n *     checks `isValid` before closing.\n *\n *  2. **Host-driven** — host renders `<cqa-move-test-suite-dialog>` inside its\n *     own modal and listens to `(submitted)` / `(cancelled)` / `(pickedFolderIdChange)`.\n */\n@Component({\n  selector: 'cqa-move-test-suite-dialog',\n  template: `\n    <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-w-full\">\n      <!-- Test suite name (read-only display; surfaces which suite is being moved) -->\n      <cqa-custom-input\n        [label]=\"testSuiteLabels.moveTestSuiteDialogSuiteNameLabel\"\n        [value]=\"suiteName\"\n        [disabled]=\"fieldsDisabled\"\n        [fullWidth]=\"true\"\n      ></cqa-custom-input>\n\n      <!-- Current folder (rendered as a disabled input) -->\n      <cqa-custom-input\n        [label]=\"testSuiteLabels.moveTestSuiteDialogCurrentLabel\"\n        [value]=\"resolvedCurrentName\"\n        [disabled]=\"fieldsDisabled\"\n        [fullWidth]=\"true\"\n        (valueChange)=\"onCurrentFieldValueChange($event)\"\n        (focused)=\"onCurrentFieldFocused($event)\"\n        (blurred)=\"onCurrentFieldBlurred($event)\"\n      ></cqa-custom-input>\n\n      <!-- Destination folder (rendered as a disabled input; value driven by the tree picker below) -->\n      <cqa-custom-input\n        [label]=\"testSuiteLabels.moveTestSuiteDialogDestinationLabel\"\n        [value]=\"destinationFieldValue\"\n        [placeholder]=\"destinationFieldPlaceholder\"\n        [disabled]=\"fieldsDisabled\"\n        [fullWidth]=\"true\"\n        (valueChange)=\"onDestinationFieldValueChange($event)\"\n        (focused)=\"onDestinationFieldFocused($event)\"\n        (blurred)=\"onDestinationFieldBlurred($event)\"\n      ></cqa-custom-input>\n\n      <!-- Folder tree picker -->\n      <cqa-move-to-folder-dialog\n        [folders]=\"folders\"\n        [labels]=\"labels\"\n        [currentFolderId]=\"currentFolderId\"\n        [pickedFolderId]=\"pickedFolderId\"\n        [pickerHeight]=\"pickerHeight\"\n        (pickedFolderIdChange)=\"onParentPicked($event)\"\n      ></cqa-move-to-folder-dialog>\n    </div>\n  `,\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class MoveTestSuiteDialogComponent {\n  // NOTE: `folders` and `labels` are declared as setters (instead of plain @Input fields)\n  // because `DialogService.open(...)` assigns the `inputs` map via direct property\n  // assignment *after* `attachComponent` has already run the component's lifecycle\n  // hooks — direct assignment bypasses `ngOnChanges`, so setters keep derived state\n  // in sync no matter the assignment order. Same pattern as `NewFolderDialogComponent`.\n\n  private _folders: FolderNode[] = [];\n  @Input() set folders(value: FolderNode[]) {\n    this._folders = value || [];\n    this.cdr.markForCheck();\n  }\n  get folders(): FolderNode[] { return this._folders; }\n\n  private _labels: ModularLabels = { ...DEFAULT_MODULAR_LABELS };\n  @Input() set labels(value: ModularLabels) {\n    this._labels = value || { ...DEFAULT_MODULAR_LABELS };\n    this.cdr.markForCheck();\n  }\n  get labels(): ModularLabels { return this._labels; }\n\n  /** Test-suite-section labels (title, description, button text, field labels,\n   *  placeholder). Kept separate from `labels` (ModularLabels) so the embedded\n   *  picker can keep its modular-table-template label model. Same setter\n   *  pattern as `labels` for DialogService property-assignment compatibility. */\n  private _testSuiteLabels: TestSuiteLabels = { ...DEFAULT_TEST_SUITE_LABELS };\n  @Input() set testSuiteLabels(value: TestSuiteLabels) {\n    this._testSuiteLabels = value || { ...DEFAULT_TEST_SUITE_LABELS };\n    this.cdr.markForCheck();\n  }\n  get testSuiteLabels(): TestSuiteLabels { return this._testSuiteLabels; }\n\n  /** Folder the suite currently lives in. Used both to gray out the row in the\n   *  embedded picker and to render the \"Current folder\" readout. */\n  @Input() currentFolderId: number | null = null;\n\n  /** Optional override for the current-folder display name. When not provided,\n   *  the component resolves the name from `folders` by walking to `currentFolderId`,\n   *  falling back to `labels.moveDialogRoot` (\"Unorganized\") when null. */\n  @Input() currentFolderName: string | null = null;\n\n  /** Name of the test suite being moved. Rendered as a disabled input at the\n   *  top of the dialog body so the user can confirm which suite the move\n   *  applies to. Defaults to empty string. */\n  @Input() suiteName: string = '';\n\n  /** Destination folder id picked by the user. Two-way-bind friendly. */\n  @Input() pickedFolderId: number | null = null;\n\n  /** Height of the embedded picker. Defaults to 240px (matches the parent-folder\n   *  picker height in `NewFolderDialogComponent`) so the modal stays compact. */\n  @Input() pickerHeight: string = '240px';\n\n  /** When true (default), both the Current folder and Destination folder inputs\n   *  render as disabled (read-only via the cqa-custom-input `[disabled]` flag).\n   *  Set to false if a host needs to re-enable interaction on those fields. */\n  @Input() fieldsDisabled: boolean = true;\n\n  @Output() pickedFolderIdChange = new EventEmitter<number | null>();\n  @Output() submitted = new EventEmitter<{ targetFolderId: number | null }>();\n  @Output() cancelled = new EventEmitter<void>();\n\n  /** Bubble cqa-custom-input focus/blur events up to hosts. Useful when\n   *  `fieldsDisabled = false` and the host wants to react to interaction. */\n  @Output() currentFieldFocus = new EventEmitter<FocusEvent>();\n  @Output() currentFieldBlur = new EventEmitter<FocusEvent>();\n  @Output() destinationFieldFocus = new EventEmitter<FocusEvent>();\n  @Output() destinationFieldBlur = new EventEmitter<FocusEvent>();\n\n  /** Set to true the first time the user picks anything. Lets `isValid` treat\n   *  the initial pre-touch state as \"nothing selected\" — important when the\n   *  suite starts in Unorganized (`currentFolderId = null`) so a fresh dialog\n   *  doesn't appear valid just because `pickedFolderId` also defaults to null. */\n  private pickedTouched = false;\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  onParentPicked(id: number | null): void {\n    let next: number | null = null;\n    if (id != null) {\n      const coerced = typeof id === 'number' ? id : Number(id);\n      if (Number.isFinite(coerced)) next = coerced;\n    }\n    this.pickedFolderId = next;\n    this.pickedTouched = true;\n    this.pickedFolderIdChange.emit(next);\n    this.cdr.markForCheck();\n  }\n\n  /** True once the user has picked a destination via the embedded tree. Drives\n   *  the destination-readout styling (indigo when picked, muted when not). */\n  get hasPick(): boolean {\n    return this.pickedTouched;\n  }\n\n  get resolvedCurrentName(): string {\n    if (this.currentFolderName && this.currentFolderName.trim().length) {\n      return this.currentFolderName;\n    }\n    if (this.currentFolderId == null) {\n      return this.labels.moveDialogRoot;\n    }\n    const name = this.findFolderName(this._folders, this.currentFolderId);\n    return name ?? this.labels.moveDialogRoot;\n  }\n\n  /** Value shown inside the Destination input. Empty until the user picks; once\n   *  picked, the resolved folder name (or `labels.moveDialogRoot` for Unorganized).\n   *  Empty state lets the input's `[placeholder]` show through. */\n  get destinationFieldValue(): string {\n    if (!this.pickedTouched) return '';\n    if (this.pickedFolderId == null) return this.labels.moveDialogRoot;\n    return this.findFolderName(this._folders, this.pickedFolderId) ?? this.labels.moveDialogRoot;\n  }\n\n  /** Placeholder shown in the Destination input before the user picks. */\n  get destinationFieldPlaceholder(): string {\n    return this._testSuiteLabels.moveTestSuiteDialogDestinationPlaceholder;\n  }\n\n  /** Move is valid when the user has picked a destination different from the\n   *  current folder. Picking the same folder as the source is a no-op, and we\n   *  also require an explicit interaction (`pickedTouched`). */\n  get isValid(): boolean {\n    if (!this.pickedTouched) return false;\n    return this.pickedFolderId !== this.currentFolderId;\n  }\n\n  submit(): void {\n    if (!this.isValid) return;\n    this.submitted.emit({ targetFolderId: this.pickedFolderId });\n  }\n\n  cancel(): void {\n    this.cancelled.emit();\n  }\n\n  /** Hooks for the Current folder cqa-custom-input. Field is disabled by\n   *  default, so these mostly no-op; they're wired so the component still\n   *  behaves correctly when `fieldsDisabled = false`. */\n  onCurrentFieldValueChange(_value: string): void {\n    // Read-only field — ignore programmatic changes.\n  }\n  onCurrentFieldFocused(event: FocusEvent): void {\n    this.currentFieldFocus.emit(event);\n  }\n  onCurrentFieldBlurred(event: FocusEvent): void {\n    this.currentFieldBlur.emit(event);\n  }\n\n  /** Hooks for the Destination folder cqa-custom-input. Value is driven by the\n   *  tree picker below, so the input itself is informational. */\n  onDestinationFieldValueChange(_value: string): void {\n    // Destination is driven by the tree picker, not the input itself.\n  }\n  onDestinationFieldFocused(event: FocusEvent): void {\n    this.destinationFieldFocus.emit(event);\n  }\n  onDestinationFieldBlurred(event: FocusEvent): void {\n    this.destinationFieldBlur.emit(event);\n  }\n\n  private findFolderName(nodes: FolderNode[], id: number): string | null {\n    for (const n of nodes || []) {\n      if (n.id === id) return n.name;\n      if (n.children?.length) {\n        const hit = this.findFolderName(n.children, id);\n        if (hit != null) return hit;\n      }\n    }\n    return null;\n  }\n}\n"]}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
export const DEFAULT_TEST_SUITE_LABELS = {
|
|
2
|
+
moveTestSuiteDialogTitle: 'Move test suite',
|
|
3
|
+
moveTestSuiteDialogDescription: 'Pick a destination folder for this test suite.',
|
|
4
|
+
moveTestSuiteDialogSuiteNameLabel: 'Test suite',
|
|
5
|
+
moveTestSuiteDialogCurrentLabel: 'Current folder',
|
|
6
|
+
moveTestSuiteDialogDestinationLabel: 'Destination folder',
|
|
7
|
+
moveTestSuiteDialogDestinationPlaceholder: 'Select a folder below',
|
|
8
|
+
moveTestSuiteDialogCancel: 'Cancel',
|
|
9
|
+
moveTestSuiteDialogConfirm: 'Move',
|
|
10
|
+
};
|
|
11
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVzdC1zdWl0ZS5tb2RlbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL3Rlc3Qtc3VpdGUvdGVzdC1zdWl0ZS5tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBbUJBLE1BQU0sQ0FBQyxNQUFNLHlCQUF5QixHQUFvQjtJQUN4RCx3QkFBd0IsRUFBRSxpQkFBaUI7SUFDM0MsOEJBQThCLEVBQUUsZ0RBQWdEO0lBQ2hGLGlDQUFpQyxFQUFFLFlBQVk7SUFDL0MsK0JBQStCLEVBQUUsZ0JBQWdCO0lBQ2pELG1DQUFtQyxFQUFFLG9CQUFvQjtJQUN6RCx5Q0FBeUMsRUFBRSx1QkFBdUI7SUFDbEUseUJBQXlCLEVBQUUsUUFBUTtJQUNuQywwQkFBMEIsRUFBRSxNQUFNO0NBQ25DLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIExhYmVscyBmb3IgdGhlIFRlc3QgU3VpdGUgc2VjdGlvbidzIGRpYWxvZ3MuIEtlcHQgc2VwYXJhdGUgZnJvbVxuICogYE1vZHVsYXJMYWJlbHNgIGJlY2F1c2UgdGhlc2UgYXJlIHRlc3Qtc3VpdGUtc3BlY2lmaWMgY29weSDigJQgdGhlIG1vZHVsYXJcbiAqIHRhYmxlIHRlbXBsYXRlIGRvZXNuJ3QgKGFuZCBzaG91bGRuJ3QpIG5lZWQgdGhlbS5cbiAqXG4gKiBDdXJyZW50bHkgY29uc3VtZWQgYnkgYE1vdmVUZXN0U3VpdGVEaWFsb2dDb21wb25lbnRgIGZvciBpdHMgb3duIGZpZWxkXG4gKiBsYWJlbHMgYW5kIGJ5IHRoZSBvdXRlciBkaWFsb2cgc2hlbGwgKHRpdGxlIC8gZGVzY3JpcHRpb24gLyBDYW5jZWwgLyBNb3ZlKS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBUZXN0U3VpdGVMYWJlbHMge1xuICBtb3ZlVGVzdFN1aXRlRGlhbG9nVGl0bGU6IHN0cmluZztcbiAgbW92ZVRlc3RTdWl0ZURpYWxvZ0Rlc2NyaXB0aW9uOiBzdHJpbmc7XG4gIG1vdmVUZXN0U3VpdGVEaWFsb2dTdWl0ZU5hbWVMYWJlbDogc3RyaW5nO1xuICBtb3ZlVGVzdFN1aXRlRGlhbG9nQ3VycmVudExhYmVsOiBzdHJpbmc7XG4gIG1vdmVUZXN0U3VpdGVEaWFsb2dEZXN0aW5hdGlvbkxhYmVsOiBzdHJpbmc7XG4gIG1vdmVUZXN0U3VpdGVEaWFsb2dEZXN0aW5hdGlvblBsYWNlaG9sZGVyOiBzdHJpbmc7XG4gIG1vdmVUZXN0U3VpdGVEaWFsb2dDYW5jZWw6IHN0cmluZztcbiAgbW92ZVRlc3RTdWl0ZURpYWxvZ0NvbmZpcm06IHN0cmluZztcbn1cblxuZXhwb3J0IGNvbnN0IERFRkFVTFRfVEVTVF9TVUlURV9MQUJFTFM6IFRlc3RTdWl0ZUxhYmVscyA9IHtcbiAgbW92ZVRlc3RTdWl0ZURpYWxvZ1RpdGxlOiAnTW92ZSB0ZXN0IHN1aXRlJyxcbiAgbW92ZVRlc3RTdWl0ZURpYWxvZ0Rlc2NyaXB0aW9uOiAnUGljayBhIGRlc3RpbmF0aW9uIGZvbGRlciBmb3IgdGhpcyB0ZXN0IHN1aXRlLicsXG4gIG1vdmVUZXN0U3VpdGVEaWFsb2dTdWl0ZU5hbWVMYWJlbDogJ1Rlc3Qgc3VpdGUnLFxuICBtb3ZlVGVzdFN1aXRlRGlhbG9nQ3VycmVudExhYmVsOiAnQ3VycmVudCBmb2xkZXInLFxuICBtb3ZlVGVzdFN1aXRlRGlhbG9nRGVzdGluYXRpb25MYWJlbDogJ0Rlc3RpbmF0aW9uIGZvbGRlcicsXG4gIG1vdmVUZXN0U3VpdGVEaWFsb2dEZXN0aW5hdGlvblBsYWNlaG9sZGVyOiAnU2VsZWN0IGEgZm9sZGVyIGJlbG93JyxcbiAgbW92ZVRlc3RTdWl0ZURpYWxvZ0NhbmNlbDogJ0NhbmNlbCcsXG4gIG1vdmVUZXN0U3VpdGVEaWFsb2dDb25maXJtOiAnTW92ZScsXG59O1xuIl19
|