@cqa-lib/cqa-ui 1.1.548-gamma.20 → 1.1.548-gamma.21
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/custom-input/custom-input.component.mjs +20 -4
- package/esm2020/lib/dialogs/name-prompt-modal.component.mjs +1 -1
- package/esm2020/lib/export-code-modal/export-code-modal.component.mjs +1 -1
- package/esm2020/lib/manage-columns-dialog/manage-columns-dialog.component.mjs +1 -1
- package/esm2020/lib/new-db-config-dialog/new-db-config-dialog.component.mjs +1 -1
- package/esm2020/lib/new-environment-dialog/new-environment-dialog.component.mjs +1 -1
- package/esm2020/lib/new-environment-variable-dialog/new-environment-variable-dialog.component.mjs +5 -13
- package/esm2020/lib/new-global-variable-dialog/new-global-variable-dialog.component.mjs +1 -1
- package/esm2020/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.component.mjs +172 -13
- package/esm2020/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.models.mjs +1 -1
- package/esm2020/lib/questionnaire-list/questionnaire-list.component.mjs +1 -1
- package/esm2020/lib/step-builder/advanced-variables-form/advanced-variables-form.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-ai-agent/step-builder-ai-agent.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-custom-code/step-builder-custom-code.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-database/step-builder-database.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-document/step-builder-document.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/step-builder/step-builder-group/step-builder-group.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-loop/step-builder-loop.component.mjs +1 -1
- package/esm2020/lib/step-builder/template-variables-form/template-variables-form.component.mjs +1 -1
- package/esm2020/lib/templates/modular-table-template/dialogs/new-folder-dialog.component.mjs +1 -1
- package/esm2020/lib/test-case-details/api-edit-step/api-edit-step.component.mjs +1 -1
- package/esm2020/lib/test-case-details/condition-step/condition-step.component.mjs +1 -1
- package/esm2020/lib/test-case-details/create-step-group/create-step-group.component.mjs +1 -1
- package/esm2020/lib/test-case-details/data-library-panel/data-library-panel.component.mjs +3 -3
- package/esm2020/lib/test-case-details/element-popup/element-form/element-form.component.mjs +1 -1
- package/esm2020/lib/test-case-details/step-details-drawer/step-details-drawer.component.mjs +1 -1
- package/esm2020/lib/test-case-details/test-case-details-edit/test-case-details-edit.component.mjs +1 -1
- package/esm2020/lib/test-case-details/test-data-modal/test-data-modal.component.mjs +1 -1
- package/esm2020/lib/viewport-selector/viewport-selector.component.mjs +1 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +218 -53
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +217 -53
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/custom-input/custom-input.component.d.ts +6 -1
- package/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.component.d.ts +39 -5
- package/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.models.d.ts +18 -0
- package/package.json +1 -1
- package/styles.css +1 -1
|
@@ -1,5 +1,7 @@
|
|
|
1
1
|
import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
|
|
2
2
|
import { FormControl, FormGroup } from '@angular/forms';
|
|
3
|
+
import { Subject, of } from 'rxjs';
|
|
4
|
+
import { catchError, debounceTime, distinctUntilChanged, switchMap, takeUntil } from 'rxjs/operators';
|
|
3
5
|
import * as i0 from "@angular/core";
|
|
4
6
|
import * as i1 from "../custom-input/custom-input.component";
|
|
5
7
|
import * as i2 from "../permission-toggle/permission-toggle.component";
|
|
@@ -12,6 +14,7 @@ export class NewTestDataProfileDialogComponent {
|
|
|
12
14
|
this.mode = 'create';
|
|
13
15
|
this.existingNames = [];
|
|
14
16
|
this.environments = [];
|
|
17
|
+
this.pageSize = 50;
|
|
15
18
|
this.name = '';
|
|
16
19
|
this.description = '';
|
|
17
20
|
this.readWriteMode = 'RW';
|
|
@@ -20,10 +23,23 @@ export class NewTestDataProfileDialogComponent {
|
|
|
20
23
|
this.envForm = new FormGroup({
|
|
21
24
|
environmentIds: new FormControl([]),
|
|
22
25
|
});
|
|
23
|
-
this.envConfig = this.
|
|
26
|
+
this.envConfig = this.buildStaticEnvConfig([]);
|
|
24
27
|
this.nameError = null;
|
|
25
28
|
this.matrixError = null;
|
|
26
29
|
this.envsError = null;
|
|
30
|
+
// Server-search state — populated only when `searchFn` is provided.
|
|
31
|
+
this.currentEnvOptions = [];
|
|
32
|
+
this.totalEnvElements = 0;
|
|
33
|
+
this.currentSearchTerm = '';
|
|
34
|
+
this.currentPageIndex = 0;
|
|
35
|
+
this.isLoadingEnvs = false;
|
|
36
|
+
/** Cache of every env the user has selected at any point during this dialog
|
|
37
|
+
* session — keyed by id. Merged into the visible options on every page
|
|
38
|
+
* refresh so the multi-select trigger always shows the right names even
|
|
39
|
+
* after the user types a search that filters the selection out. */
|
|
40
|
+
this.selectedEnvCache = new Map();
|
|
41
|
+
this.searchInput$ = new Subject();
|
|
42
|
+
this.destroy$ = new Subject();
|
|
27
43
|
}
|
|
28
44
|
ngOnInit() {
|
|
29
45
|
if (this.initialValue) {
|
|
@@ -37,15 +53,38 @@ export class NewTestDataProfileDialogComponent {
|
|
|
37
53
|
const ids = this.initialValue.environmentIds ?? [];
|
|
38
54
|
this.envForm.get('environmentIds').setValue(ids);
|
|
39
55
|
}
|
|
40
|
-
|
|
56
|
+
if (this.searchFn) {
|
|
57
|
+
this.envConfig = this.buildServerSearchEnvConfig();
|
|
58
|
+
// Debounced typing pipeline. Each new term resets to page 0 and replaces
|
|
59
|
+
// the visible options. switchMap drops in-flight responses for stale terms.
|
|
60
|
+
this.searchInput$
|
|
61
|
+
.pipe(debounceTime(300), distinctUntilChanged(), switchMap(term => this.fetchPage(term, 0)), takeUntil(this.destroy$))
|
|
62
|
+
.subscribe(({ term, page, payload }) => {
|
|
63
|
+
this.applyPage(term, page, /* append */ false, payload);
|
|
64
|
+
});
|
|
65
|
+
// Pre-fetch page 0 so the dropdown is populated by the time the user opens it.
|
|
66
|
+
this.runFetch('', 0, /* append */ false);
|
|
67
|
+
}
|
|
68
|
+
else {
|
|
69
|
+
this.envConfig = this.buildStaticEnvConfig(this.environments ?? []);
|
|
70
|
+
}
|
|
41
71
|
// Clear envs error as soon as the user picks something.
|
|
42
|
-
this.envForm.get('environmentIds').valueChanges
|
|
72
|
+
this.envForm.get('environmentIds').valueChanges
|
|
73
|
+
.pipe(takeUntil(this.destroy$))
|
|
74
|
+
.subscribe(value => {
|
|
43
75
|
if (this.envsError) {
|
|
44
76
|
this.envsError = null;
|
|
45
77
|
this.cdr.markForCheck();
|
|
46
78
|
}
|
|
79
|
+
if (this.searchFn) {
|
|
80
|
+
this.rememberSelectedEnvs(value);
|
|
81
|
+
}
|
|
47
82
|
});
|
|
48
83
|
}
|
|
84
|
+
ngOnDestroy() {
|
|
85
|
+
this.destroy$.next();
|
|
86
|
+
this.destroy$.complete();
|
|
87
|
+
}
|
|
49
88
|
get title() {
|
|
50
89
|
return this.mode === 'edit' ? 'Edit test data profile' : 'New test data profile';
|
|
51
90
|
}
|
|
@@ -144,14 +183,9 @@ export class NewTestDataProfileDialogComponent {
|
|
|
144
183
|
this.matrixError = null;
|
|
145
184
|
return { rows, cols };
|
|
146
185
|
}
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
value: e.id,
|
|
151
|
-
name: e.name,
|
|
152
|
-
label: e.name,
|
|
153
|
-
statusColor: e.color,
|
|
154
|
-
}));
|
|
186
|
+
// -- Static-options mode (legacy) ---------------------------------------
|
|
187
|
+
buildStaticEnvConfig(envs) {
|
|
188
|
+
const options = (envs || []).map(e => this.toSelectOption(e));
|
|
155
189
|
return {
|
|
156
190
|
key: 'environmentIds',
|
|
157
191
|
label: '',
|
|
@@ -163,6 +197,127 @@ export class NewTestDataProfileDialogComponent {
|
|
|
163
197
|
options,
|
|
164
198
|
};
|
|
165
199
|
}
|
|
200
|
+
// -- Server-search mode ---------------------------------------------------
|
|
201
|
+
buildServerSearchEnvConfig() {
|
|
202
|
+
const merged = this.mergeSelectedIntoOptions(this.currentEnvOptions);
|
|
203
|
+
return {
|
|
204
|
+
key: 'environmentIds',
|
|
205
|
+
label: '',
|
|
206
|
+
placeholder: 'Select environments…',
|
|
207
|
+
multiple: true,
|
|
208
|
+
searchable: true,
|
|
209
|
+
optionStyle: 'checkbox',
|
|
210
|
+
showSelectAll: false,
|
|
211
|
+
serverSearch: true,
|
|
212
|
+
isLoading: this.isLoadingEnvs,
|
|
213
|
+
hasMore: this.currentEnvOptions.length < this.totalEnvElements,
|
|
214
|
+
options: merged,
|
|
215
|
+
onSearch: (q) => this.searchInput$.next(q ?? ''),
|
|
216
|
+
onLoadMore: () => this.onEnvLoadMore(),
|
|
217
|
+
};
|
|
218
|
+
}
|
|
219
|
+
onEnvLoadMore() {
|
|
220
|
+
if (this.isLoadingEnvs) {
|
|
221
|
+
return;
|
|
222
|
+
}
|
|
223
|
+
if (this.currentEnvOptions.length >= this.totalEnvElements) {
|
|
224
|
+
return;
|
|
225
|
+
}
|
|
226
|
+
this.runFetch(this.currentSearchTerm, this.currentPageIndex + 1, /* append */ true);
|
|
227
|
+
}
|
|
228
|
+
runFetch(term, page, append) {
|
|
229
|
+
this.isLoadingEnvs = true;
|
|
230
|
+
this.envConfig = this.buildServerSearchEnvConfig();
|
|
231
|
+
this.cdr.markForCheck();
|
|
232
|
+
this.fetchPage(term, page)
|
|
233
|
+
.pipe(takeUntil(this.destroy$))
|
|
234
|
+
.subscribe(({ term: t, page: p, payload }) => {
|
|
235
|
+
this.applyPage(t, p, append, payload);
|
|
236
|
+
});
|
|
237
|
+
}
|
|
238
|
+
fetchPage(term, page) {
|
|
239
|
+
if (!this.searchFn) {
|
|
240
|
+
return of({ term, page, payload: { items: [], total: 0 } });
|
|
241
|
+
}
|
|
242
|
+
return this.searchFn(term, page, this.pageSize).pipe(catchError(() => of({ items: [], total: 0 })), switchMap(payload => of({ term, page, payload })));
|
|
243
|
+
}
|
|
244
|
+
applyPage(term, page, append, payload) {
|
|
245
|
+
// Drop stale load-more responses — if the user typed a new search term
|
|
246
|
+
// while a load-more was in flight, this append is for an old term and
|
|
247
|
+
// would corrupt the freshly-replaced options list. The search$ pipeline
|
|
248
|
+
// already drops stale typing-debounce responses via switchMap.
|
|
249
|
+
if (append && term !== this.currentSearchTerm) {
|
|
250
|
+
return;
|
|
251
|
+
}
|
|
252
|
+
if (!append) {
|
|
253
|
+
this.currentSearchTerm = term;
|
|
254
|
+
}
|
|
255
|
+
const items = (payload?.items ?? []).map(e => this.toSelectOption(e));
|
|
256
|
+
const total = payload?.total ?? items.length;
|
|
257
|
+
if (append) {
|
|
258
|
+
// Dedupe by id so a re-fetch can't double up an already-rendered env.
|
|
259
|
+
const seen = new Set(this.currentEnvOptions.map(o => Number(o.id)));
|
|
260
|
+
const fresh = items.filter(o => !seen.has(Number(o.id)));
|
|
261
|
+
this.currentEnvOptions = [...this.currentEnvOptions, ...fresh];
|
|
262
|
+
}
|
|
263
|
+
else {
|
|
264
|
+
this.currentEnvOptions = items;
|
|
265
|
+
}
|
|
266
|
+
this.totalEnvElements = total;
|
|
267
|
+
this.currentPageIndex = page;
|
|
268
|
+
this.isLoadingEnvs = false;
|
|
269
|
+
this.envConfig = this.buildServerSearchEnvConfig();
|
|
270
|
+
this.cdr.markForCheck();
|
|
271
|
+
}
|
|
272
|
+
/** Merge cached selected options with the visible page so the multi-select
|
|
273
|
+
* trigger always shows the right names — even when the user types a query
|
|
274
|
+
* that filters previously selected envs out of view. Selected ones come
|
|
275
|
+
* first to keep the state visible. */
|
|
276
|
+
mergeSelectedIntoOptions(visible) {
|
|
277
|
+
if (this.selectedEnvCache.size === 0) {
|
|
278
|
+
return visible;
|
|
279
|
+
}
|
|
280
|
+
const visibleIds = new Set(visible.map(o => Number(o.id)));
|
|
281
|
+
const onlyCached = [];
|
|
282
|
+
for (const opt of this.selectedEnvCache.values()) {
|
|
283
|
+
if (!visibleIds.has(Number(opt.id))) {
|
|
284
|
+
onlyCached.push(opt);
|
|
285
|
+
}
|
|
286
|
+
}
|
|
287
|
+
return [...onlyCached, ...visible];
|
|
288
|
+
}
|
|
289
|
+
/** Update the selected-env cache from the FormControl value so the trigger
|
|
290
|
+
* display + future merges reflect the latest selection. We cache the full
|
|
291
|
+
* SelectOption (id + name) by looking up either the visible page or the
|
|
292
|
+
* prior cache. Selections are accumulated, never removed — even unselect →
|
|
293
|
+
* reselect works without a fresh fetch. */
|
|
294
|
+
rememberSelectedEnvs(value) {
|
|
295
|
+
const ids = Array.isArray(value)
|
|
296
|
+
? value.map(v => Number(v)).filter(n => !Number.isNaN(n))
|
|
297
|
+
: [];
|
|
298
|
+
if (ids.length === 0) {
|
|
299
|
+
return;
|
|
300
|
+
}
|
|
301
|
+
for (const id of ids) {
|
|
302
|
+
if (this.selectedEnvCache.has(id)) {
|
|
303
|
+
continue;
|
|
304
|
+
}
|
|
305
|
+
const fromVisible = this.currentEnvOptions.find(o => Number(o.id) === id);
|
|
306
|
+
if (fromVisible) {
|
|
307
|
+
this.selectedEnvCache.set(id, fromVisible);
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
}
|
|
311
|
+
// -- Helpers --------------------------------------------------------------
|
|
312
|
+
toSelectOption(e) {
|
|
313
|
+
return {
|
|
314
|
+
id: e.id,
|
|
315
|
+
value: e.id,
|
|
316
|
+
name: e.name,
|
|
317
|
+
label: e.name,
|
|
318
|
+
statusColor: e.color,
|
|
319
|
+
};
|
|
320
|
+
}
|
|
166
321
|
isDuplicateName(candidate) {
|
|
167
322
|
const existing = (this.existingNames ?? []).map(n => (n ?? '').trim().toLowerCase());
|
|
168
323
|
const lowered = candidate.toLowerCase();
|
|
@@ -176,7 +331,7 @@ export class NewTestDataProfileDialogComponent {
|
|
|
176
331
|
NewTestDataProfileDialogComponent.INTEGER_REGEX = /^\s*(\d+)\s*$/;
|
|
177
332
|
NewTestDataProfileDialogComponent.MAX_DIM = 100;
|
|
178
333
|
NewTestDataProfileDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NewTestDataProfileDialogComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
179
|
-
NewTestDataProfileDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: NewTestDataProfileDialogComponent, selector: "cqa-new-test-data-profile-dialog", inputs: { mode: "mode", initialValue: "initialValue", existingNames: "existingNames", environments: "environments" }, host: { styleAttribute: "display:block;width:100%;", classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n <!-- Name + Permission -->\n <div class=\"cqa-grid cqa-grid-cols-[1fr_auto] cqa-gap-4 cqa-items-start\">\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Profile name\"\n placeholder=\"e.g. Guest Personas\"\n type=\"text\"\n [value]=\"name\"\n [required]=\"true\"\n [fullWidth]=\"true\"\n [errors]=\"nameErrorsArray\"\n (valueChange)=\"onNameChange($event)\">\n </cqa-custom-input>\n </div>\n <div class=\"cqa-flex cqa-flex-col\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Permission</label>\n <cqa-permission-toggle\n [value]=\"readWriteMode\"\n (valueChange)=\"onPermissionChange($event)\">\n </cqa-permission-toggle>\n </div>\n </div>\n\n <!-- Description -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Description\"\n placeholder=\"What data is this profile for?\"\n type=\"text\"\n [value]=\"description\"\n [fullWidth]=\"true\"\n (valueChange)=\"onDescriptionChange($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- Rows + Columns matrix -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1\">\n <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n <cqa-custom-input\n label=\"Rows\"\n placeholder=\"e.g. 2\"\n type=\"number\"\n [value]=\"rowsInput\"\n [fullWidth]=\"true\"\n (valueChange)=\"onRowsChange($event)\">\n </cqa-custom-input>\n <cqa-custom-input\n label=\"Columns\"\n placeholder=\"e.g. 3\"\n type=\"number\"\n [value]=\"columnsInput\"\n [fullWidth]=\"true\"\n (valueChange)=\"onColumnsChange($event)\">\n </cqa-custom-input>\n </div>\n <span *ngIf=\"matrixError\" class=\"cqa-flex cqa-items-center cqa-gap-1 cqa-text-xs cqa-text-red-600\">\n <mat-icon class=\"cqa-flex-none\" style=\"font-size:14px;width:14px;height:14px;line-height:14px;\">error</mat-icon>\n <span>{{ matrixError }}</span>\n </span>\n </div>\n\n <!-- Assign to environments -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">\n Assign to environments <span class=\"cqa-text-red-600\">*</span>\n </label>\n <cqa-dynamic-select\n [form]=\"envForm\"\n [config]=\"envConfig\">\n </cqa-dynamic-select>\n <span *ngIf=\"envsError\" class=\"cqa-flex cqa-items-center cqa-gap-1 cqa-text-xs cqa-text-red-600\">\n <mat-icon class=\"cqa-flex-none\" style=\"font-size:14px;width:14px;height:14px;line-height:14px;\">error</mat-icon>\n <span>{{ envsError }}</span>\n </span>\n <span *ngIf=\"!envsError\" class=\"cqa-text-xs cqa-text-gray-500\">\n Each environment gets an independent copy of the data (same columns, separate rows).\n </span>\n </div>\n\n</div>\n", 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.PermissionToggleComponent, selector: "cqa-permission-toggle", inputs: ["value", "disabled", "displayOnly", "roTooltip", "rwTooltip"], outputs: ["valueChange"] }, { type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i4.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }], directives: [{ type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
334
|
+
NewTestDataProfileDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: NewTestDataProfileDialogComponent, selector: "cqa-new-test-data-profile-dialog", inputs: { mode: "mode", initialValue: "initialValue", existingNames: "existingNames", environments: "environments", searchFn: "searchFn", pageSize: "pageSize" }, host: { styleAttribute: "display:block;width:100%;", classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n <!-- Name + Permission -->\n <div class=\"cqa-grid cqa-grid-cols-[1fr_auto] cqa-gap-4 cqa-items-start\">\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Profile name\"\n placeholder=\"e.g. Guest Personas\"\n type=\"text\"\n [value]=\"name\"\n [required]=\"true\"\n [fullWidth]=\"true\"\n [errors]=\"nameErrorsArray\"\n (valueChange)=\"onNameChange($event)\">\n </cqa-custom-input>\n </div>\n <div class=\"cqa-flex cqa-flex-col\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Permission</label>\n <cqa-permission-toggle\n [value]=\"readWriteMode\"\n (valueChange)=\"onPermissionChange($event)\">\n </cqa-permission-toggle>\n </div>\n </div>\n\n <!-- Description -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Description\"\n placeholder=\"What data is this profile for?\"\n type=\"text\"\n [value]=\"description\"\n [fullWidth]=\"true\"\n (valueChange)=\"onDescriptionChange($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- Rows + Columns matrix -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1\">\n <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n <cqa-custom-input\n label=\"Rows\"\n placeholder=\"e.g. 2\"\n type=\"number\"\n [value]=\"rowsInput\"\n [fullWidth]=\"true\"\n (valueChange)=\"onRowsChange($event)\">\n </cqa-custom-input>\n <cqa-custom-input\n label=\"Columns\"\n placeholder=\"e.g. 3\"\n type=\"number\"\n [value]=\"columnsInput\"\n [fullWidth]=\"true\"\n (valueChange)=\"onColumnsChange($event)\">\n </cqa-custom-input>\n </div>\n <span *ngIf=\"matrixError\" class=\"cqa-flex cqa-items-center cqa-gap-1 cqa-text-xs cqa-text-red-600\">\n <mat-icon class=\"cqa-flex-none\" style=\"font-size:14px;width:14px;height:14px;line-height:14px;\">error</mat-icon>\n <span>{{ matrixError }}</span>\n </span>\n </div>\n\n <!-- Assign to environments -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">\n Assign to environments <span class=\"cqa-text-red-600\">*</span>\n </label>\n <cqa-dynamic-select\n [form]=\"envForm\"\n [config]=\"envConfig\">\n </cqa-dynamic-select>\n <span *ngIf=\"envsError\" class=\"cqa-flex cqa-items-center cqa-gap-1 cqa-text-xs cqa-text-red-600\">\n <mat-icon class=\"cqa-flex-none\" style=\"font-size:14px;width:14px;height:14px;line-height:14px;\">error</mat-icon>\n <span>{{ envsError }}</span>\n </span>\n <span *ngIf=\"!envsError\" class=\"cqa-text-xs cqa-text-gray-500\">\n Each environment gets an independent copy of the data (same columns, separate rows).\n </span>\n </div>\n\n</div>\n", components: [{ type: i1.CustomInputComponent, selector: "cqa-custom-input", inputs: ["inputId", "label", "type", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "inputInlineStyle", "labelInlineStyle", "showPasswordToggle"], outputs: ["valueChange", "blurred", "focused", "enterPressed"] }, { type: i2.PermissionToggleComponent, selector: "cqa-permission-toggle", inputs: ["value", "disabled", "displayOnly", "roTooltip", "rwTooltip"], outputs: ["valueChange"] }, { type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i4.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }], directives: [{ type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
180
335
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NewTestDataProfileDialogComponent, decorators: [{
|
|
181
336
|
type: Component,
|
|
182
337
|
args: [{ selector: 'cqa-new-test-data-profile-dialog', changeDetection: ChangeDetectionStrategy.OnPush, host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n <!-- Name + Permission -->\n <div class=\"cqa-grid cqa-grid-cols-[1fr_auto] cqa-gap-4 cqa-items-start\">\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Profile name\"\n placeholder=\"e.g. Guest Personas\"\n type=\"text\"\n [value]=\"name\"\n [required]=\"true\"\n [fullWidth]=\"true\"\n [errors]=\"nameErrorsArray\"\n (valueChange)=\"onNameChange($event)\">\n </cqa-custom-input>\n </div>\n <div class=\"cqa-flex cqa-flex-col\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Permission</label>\n <cqa-permission-toggle\n [value]=\"readWriteMode\"\n (valueChange)=\"onPermissionChange($event)\">\n </cqa-permission-toggle>\n </div>\n </div>\n\n <!-- Description -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Description\"\n placeholder=\"What data is this profile for?\"\n type=\"text\"\n [value]=\"description\"\n [fullWidth]=\"true\"\n (valueChange)=\"onDescriptionChange($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- Rows + Columns matrix -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1\">\n <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n <cqa-custom-input\n label=\"Rows\"\n placeholder=\"e.g. 2\"\n type=\"number\"\n [value]=\"rowsInput\"\n [fullWidth]=\"true\"\n (valueChange)=\"onRowsChange($event)\">\n </cqa-custom-input>\n <cqa-custom-input\n label=\"Columns\"\n placeholder=\"e.g. 3\"\n type=\"number\"\n [value]=\"columnsInput\"\n [fullWidth]=\"true\"\n (valueChange)=\"onColumnsChange($event)\">\n </cqa-custom-input>\n </div>\n <span *ngIf=\"matrixError\" class=\"cqa-flex cqa-items-center cqa-gap-1 cqa-text-xs cqa-text-red-600\">\n <mat-icon class=\"cqa-flex-none\" style=\"font-size:14px;width:14px;height:14px;line-height:14px;\">error</mat-icon>\n <span>{{ matrixError }}</span>\n </span>\n </div>\n\n <!-- Assign to environments -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">\n Assign to environments <span class=\"cqa-text-red-600\">*</span>\n </label>\n <cqa-dynamic-select\n [form]=\"envForm\"\n [config]=\"envConfig\">\n </cqa-dynamic-select>\n <span *ngIf=\"envsError\" class=\"cqa-flex cqa-items-center cqa-gap-1 cqa-text-xs cqa-text-red-600\">\n <mat-icon class=\"cqa-flex-none\" style=\"font-size:14px;width:14px;height:14px;line-height:14px;\">error</mat-icon>\n <span>{{ envsError }}</span>\n </span>\n <span *ngIf=\"!envsError\" class=\"cqa-text-xs cqa-text-gray-500\">\n Each environment gets an independent copy of the data (same columns, separate rows).\n </span>\n </div>\n\n</div>\n" }]
|
|
@@ -188,5 +343,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
|
|
|
188
343
|
type: Input
|
|
189
344
|
}], environments: [{
|
|
190
345
|
type: Input
|
|
346
|
+
}], searchFn: [{
|
|
347
|
+
type: Input
|
|
348
|
+
}], pageSize: [{
|
|
349
|
+
type: Input
|
|
191
350
|
}] } });
|
|
192
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"new-test-data-profile-dialog.component.js","sourceRoot":"","sources":["../../../../../src/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.component.ts","../../../../../src/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Input, OnInit } from '@angular/core';\nimport { FormControl, FormGroup } from '@angular/forms';\n\nimport {\n  DynamicSelectFieldConfig,\n  SelectOption,\n} from '../dynamic-select/dynamic-select-field.component';\nimport {\n  TdpDialogPermission,\n  TdpDialogValue,\n  TdpEnvironmentOption,\n} from './new-test-data-profile-dialog.models';\n\n@Component({\n  selector: 'cqa-new-test-data-profile-dialog',\n  templateUrl: './new-test-data-profile-dialog.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' },\n})\nexport class NewTestDataProfileDialogComponent implements OnInit {\n  private static readonly INTEGER_REGEX = /^\\s*(\\d+)\\s*$/;\n  private static readonly MAX_DIM = 100;\n\n  @Input() mode: 'create' | 'edit' = 'create';\n  @Input() initialValue?: Partial<TdpDialogValue>;\n  @Input() existingNames: string[] = [];\n  @Input() environments: TdpEnvironmentOption[] = [];\n\n  public name = '';\n  public description = '';\n  public readWriteMode: TdpDialogPermission = 'RW';\n  public rowsInput = '1';\n  public columnsInput = '1';\n\n  public readonly envForm = new FormGroup({\n    environmentIds: new FormControl([]),\n  });\n  public envConfig: DynamicSelectFieldConfig = this.buildEnvConfig([]);\n\n  public nameError: string | null = null;\n  public matrixError: string | null = null;\n  public envsError: string | null = null;\n\n  constructor(private readonly cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    if (this.initialValue) {\n      this.name = this.initialValue.name ?? '';\n      this.description = this.initialValue.description ?? '';\n      this.readWriteMode = this.initialValue.readWriteMode ?? 'RW';\n      const colCount = this.initialValue.columns?.length ?? 1;\n      const rowCount = this.initialValue.rows?.length ?? 1;\n      this.rowsInput = String(rowCount);\n      this.columnsInput = String(colCount);\n      const ids = this.initialValue.environmentIds ?? [];\n      this.envForm.get('environmentIds')!.setValue(ids);\n    }\n    this.envConfig = this.buildEnvConfig(this.environments ?? []);\n\n    // Clear envs error as soon as the user picks something.\n    this.envForm.get('environmentIds')!.valueChanges.subscribe(() => {\n      if (this.envsError) {\n        this.envsError = null;\n        this.cdr.markForCheck();\n      }\n    });\n  }\n\n  public get title(): string {\n    return this.mode === 'edit' ? 'Edit test data profile' : 'New test data profile';\n  }\n\n  public get subtitle(): string {\n    return 'Define the matrix; data values can be added per environment after creation.';\n  }\n\n  public get primaryButtonLabel(): string {\n    return this.mode === 'edit' ? 'Save changes' : 'Create profile';\n  }\n\n  public get nameErrorsArray(): string[] { return this.nameError ? [this.nameError] : []; }\n\n  public get selectedEnvIds(): number[] {\n    const val = this.envForm.get('environmentIds')!.value ?? [];\n    return Array.isArray(val) ? val : [];\n  }\n\n  public onNameChange(next: string): void {\n    this.name = next;\n    this.nameError = null;\n    this.cdr.markForCheck();\n  }\n\n  public onDescriptionChange(next: string): void {\n    this.description = next;\n    this.cdr.markForCheck();\n  }\n\n  public onPermissionChange(next: string): void {\n    this.readWriteMode = next === 'RO' ? 'RO' : 'RW';\n    this.cdr.markForCheck();\n  }\n\n  public onRowsChange(next: string): void {\n    this.rowsInput = next ?? '';\n    this.matrixError = null;\n    this.cdr.markForCheck();\n  }\n\n  public onColumnsChange(next: string): void {\n    this.columnsInput = next ?? '';\n    this.matrixError = null;\n    this.cdr.markForCheck();\n  }\n\n  public getValue(): TdpDialogValue | null {\n    const trimmedName = this.name.trim();\n    if (!trimmedName) {\n      this.nameError = 'Name is required.';\n    } else if (this.isDuplicateName(trimmedName)) {\n      this.nameError = 'A profile with this name already exists.';\n    } else {\n      this.nameError = null;\n    }\n\n    const parsed = this.parseMatrix();\n\n    const envIds = this.selectedEnvIds;\n    if (!envIds || envIds.length === 0) {\n      this.envsError = 'Select at least one environment to assign this profile to.';\n    } else {\n      this.envsError = null;\n    }\n\n    if (this.nameError || this.matrixError || this.envsError || !parsed) {\n      this.cdr.markForCheck();\n      return null;\n    }\n\n    const cols = Array.from({ length: parsed.cols }, (_, i) => `col_${i + 1}`);\n    const rows = Array.from({ length: parsed.rows }, (_, i) => `row_${i + 1}`);\n\n    return {\n      name: trimmedName,\n      description: this.description.trim() || null,\n      readWriteMode: this.readWriteMode,\n      columns: cols,\n      rows,\n      environmentIds: this.selectedEnvIds,\n    };\n  }\n\n  private parseMatrix(): { rows: number; cols: number } | null {\n    const rawRows = (this.rowsInput ?? '').trim();\n    const rawCols = (this.columnsInput ?? '').trim();\n    if (!rawRows || !rawCols) {\n      this.matrixError = 'Rows and columns are required.';\n      return null;\n    }\n    const rowsMatch = NewTestDataProfileDialogComponent.INTEGER_REGEX.exec(rawRows);\n    const colsMatch = NewTestDataProfileDialogComponent.INTEGER_REGEX.exec(rawCols);\n    if (!rowsMatch || !colsMatch) {\n      this.matrixError = 'Rows and columns must be whole numbers.';\n      return null;\n    }\n    const rows = parseInt(rowsMatch[1], 10);\n    const cols = parseInt(colsMatch[1], 10);\n    if (rows < 1 || cols < 1) {\n      this.matrixError = 'Rows and columns must be at least 1.';\n      return null;\n    }\n    if (rows > NewTestDataProfileDialogComponent.MAX_DIM\n      || cols > NewTestDataProfileDialogComponent.MAX_DIM) {\n      this.matrixError = `Rows and columns can be at most ${NewTestDataProfileDialogComponent.MAX_DIM}.`;\n      return null;\n    }\n    this.matrixError = null;\n    return { rows, cols };\n  }\n\n  private buildEnvConfig(envs: TdpEnvironmentOption[]): DynamicSelectFieldConfig {\n    const options: SelectOption[] = (envs || []).map(e => ({\n      id: e.id,\n      value: e.id,\n      name: e.name,\n      label: e.name,\n      statusColor: e.color,\n    }));\n    return {\n      key: 'environmentIds',\n      label: '',\n      placeholder: 'Select environments…',\n      multiple: true,\n      searchable: true,\n      optionStyle: 'checkbox',\n      showSelectAll: false,\n      options,\n    };\n  }\n\n  private isDuplicateName(candidate: string): boolean {\n    const existing = (this.existingNames ?? []).map(n => (n ?? '').trim().toLowerCase());\n    const lowered = candidate.toLowerCase();\n    if (this.mode === 'edit') {\n      const original = (this.initialValue?.name ?? '').trim().toLowerCase();\n      return lowered !== original && existing.includes(lowered);\n    }\n    return existing.includes(lowered);\n  }\n}\n","<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n  <!-- Name + Permission -->\n  <div class=\"cqa-grid cqa-grid-cols-[1fr_auto] cqa-gap-4 cqa-items-start\">\n    <div class=\"cqa-flex cqa-flex-col\">\n      <cqa-custom-input\n        label=\"Profile name\"\n        placeholder=\"e.g. Guest Personas\"\n        type=\"text\"\n        [value]=\"name\"\n        [required]=\"true\"\n        [fullWidth]=\"true\"\n        [errors]=\"nameErrorsArray\"\n        (valueChange)=\"onNameChange($event)\">\n      </cqa-custom-input>\n    </div>\n    <div class=\"cqa-flex cqa-flex-col\">\n      <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Permission</label>\n      <cqa-permission-toggle\n        [value]=\"readWriteMode\"\n        (valueChange)=\"onPermissionChange($event)\">\n      </cqa-permission-toggle>\n    </div>\n  </div>\n\n  <!-- Description -->\n  <div class=\"cqa-flex cqa-flex-col\">\n    <cqa-custom-input\n      label=\"Description\"\n      placeholder=\"What data is this profile for?\"\n      type=\"text\"\n      [value]=\"description\"\n      [fullWidth]=\"true\"\n      (valueChange)=\"onDescriptionChange($event)\">\n    </cqa-custom-input>\n  </div>\n\n  <!-- Rows + Columns matrix -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-1\">\n    <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n      <cqa-custom-input\n        label=\"Rows\"\n        placeholder=\"e.g. 2\"\n        type=\"number\"\n        [value]=\"rowsInput\"\n        [fullWidth]=\"true\"\n        (valueChange)=\"onRowsChange($event)\">\n      </cqa-custom-input>\n      <cqa-custom-input\n        label=\"Columns\"\n        placeholder=\"e.g. 3\"\n        type=\"number\"\n        [value]=\"columnsInput\"\n        [fullWidth]=\"true\"\n        (valueChange)=\"onColumnsChange($event)\">\n      </cqa-custom-input>\n    </div>\n    <span *ngIf=\"matrixError\" class=\"cqa-flex cqa-items-center cqa-gap-1 cqa-text-xs cqa-text-red-600\">\n      <mat-icon class=\"cqa-flex-none\" style=\"font-size:14px;width:14px;height:14px;line-height:14px;\">error</mat-icon>\n      <span>{{ matrixError }}</span>\n    </span>\n  </div>\n\n  <!-- Assign to environments -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n    <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">\n      Assign to environments <span class=\"cqa-text-red-600\">*</span>\n    </label>\n    <cqa-dynamic-select\n      [form]=\"envForm\"\n      [config]=\"envConfig\">\n    </cqa-dynamic-select>\n    <span *ngIf=\"envsError\" class=\"cqa-flex cqa-items-center cqa-gap-1 cqa-text-xs cqa-text-red-600\">\n      <mat-icon class=\"cqa-flex-none\" style=\"font-size:14px;width:14px;height:14px;line-height:14px;\">error</mat-icon>\n      <span>{{ envsError }}</span>\n    </span>\n    <span *ngIf=\"!envsError\" class=\"cqa-text-xs cqa-text-gray-500\">\n      Each environment gets an independent copy of the data (same columns, separate rows).\n    </span>\n  </div>\n\n</div>\n"]}
|
|
351
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"new-test-data-profile-dialog.component.js","sourceRoot":"","sources":["../../../../../src/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.component.ts","../../../../../src/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAqB,SAAS,EAAE,KAAK,EAAqB,MAAM,eAAe,CAAC;AAChH,OAAO,EAAE,WAAW,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AACxD,OAAO,EAAc,OAAO,EAAE,EAAE,EAAE,MAAM,MAAM,CAAC;AAC/C,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,oBAAoB,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;;;;;;;AAoBtG,MAAM,OAAO,iCAAiC;IA2C5C,YAA6B,GAAsB;QAAtB,QAAG,GAAH,GAAG,CAAmB;QAvC1C,SAAI,GAAsB,QAAQ,CAAC;QAEnC,kBAAa,GAAa,EAAE,CAAC;QAC7B,iBAAY,GAA2B,EAAE,CAAC;QAI1C,aAAQ,GAAG,EAAE,CAAC;QAEhB,SAAI,GAAG,EAAE,CAAC;QACV,gBAAW,GAAG,EAAE,CAAC;QACjB,kBAAa,GAAwB,IAAI,CAAC;QAC1C,cAAS,GAAG,GAAG,CAAC;QAChB,iBAAY,GAAG,GAAG,CAAC;QAEV,YAAO,GAAG,IAAI,SAAS,CAAC;YACtC,cAAc,EAAE,IAAI,WAAW,CAAC,EAAE,CAAC;SACpC,CAAC,CAAC;QACI,cAAS,GAA6B,IAAI,CAAC,oBAAoB,CAAC,EAAE,CAAC,CAAC;QAEpE,cAAS,GAAkB,IAAI,CAAC;QAChC,gBAAW,GAAkB,IAAI,CAAC;QAClC,cAAS,GAAkB,IAAI,CAAC;QAEvC,oEAAoE;QAC5D,sBAAiB,GAAmB,EAAE,CAAC;QACvC,qBAAgB,GAAG,CAAC,CAAC;QACrB,sBAAiB,GAAG,EAAE,CAAC;QACvB,qBAAgB,GAAG,CAAC,CAAC;QACrB,kBAAa,GAAG,KAAK,CAAC;QAC9B;;;4EAGoE;QACnD,qBAAgB,GAAG,IAAI,GAAG,EAAwB,CAAC;QAEnD,iBAAY,GAAG,IAAI,OAAO,EAAU,CAAC;QACrC,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;IAEM,CAAC;IAEvD,QAAQ;QACN,IAAI,IAAI,CAAC,YAAY,EAAE;YACrB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,IAAI,EAAE,CAAC;YACzC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,YAAY,CAAC,WAAW,IAAI,EAAE,CAAC;YACvD,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,YAAY,CAAC,aAAa,IAAI,IAAI,CAAC;YAC7D,MAAM,QAAQ,GAAG,IAAI,CAAC,YAAY,CAAC,OAAO,EAAE,MAAM,IAAI,CAAC,CAAC;YACxD,MAAM,QAAQ,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,MAAM,IAAI,CAAC,CAAC;YACrD,IAAI,CAAC,SAAS,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC;YAClC,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC;YACrC,MAAM,GAAG,GAAG,IAAI,CAAC,YAAY,CAAC,cAAc,IAAI,EAAE,CAAC;YACnD,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAE,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;SACnD;QAED,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,0BAA0B,EAAE,CAAC;YACnD,yEAAyE;YACzE,4EAA4E;YAC5E,IAAI,CAAC,YAAY;iBACd,IAAI,CACH,YAAY,CAAC,GAAG,CAAC,EACjB,oBAAoB,EAAE,EACtB,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,EAC1C,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CACzB;iBACA,SAAS,CAAC,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,OAAO,EAAE,EAAE,EAAE;gBACrC,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,EAAE,YAAY,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;YAC1D,CAAC,CAAC,CAAC;YACL,+EAA+E;YAC/E,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC,EAAE,YAAY,CAAC,KAAK,CAAC,CAAC;SAC1C;aAAM;YACL,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,YAAY,IAAI,EAAE,CAAC,CAAC;SACrE;QAED,wDAAwD;QACxD,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAE,CAAC,YAAY;aAC7C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;aAC9B,SAAS,CAAC,KAAK,CAAC,EAAE;YACjB,IAAI,IAAI,CAAC,SAAS,EAAE;gBAClB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;gBACtB,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;aACzB;YACD,IAAI,IAAI,CAAC,QAAQ,EAAE;gBACjB,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,CAAC;aAClC;QACH,CAAC,CAAC,CAAC;IACP,CAAC;IAED,WAAW;QACT,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;IAC3B,CAAC;IAED,IAAW,KAAK;QACd,OAAO,IAAI,CAAC,IAAI,KAAK,MAAM,CAAC,CAAC,CAAC,wBAAwB,CAAC,CAAC,CAAC,uBAAuB,CAAC;IACnF,CAAC;IAED,IAAW,QAAQ;QACjB,OAAO,6EAA6E,CAAC;IACvF,CAAC;IAED,IAAW,kBAAkB;QAC3B,OAAO,IAAI,CAAC,IAAI,KAAK,MAAM,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,gBAAgB,CAAC;IAClE,CAAC;IAED,IAAW,eAAe,KAAe,OAAO,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;IAEzF,IAAW,cAAc;QACvB,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAE,CAAC,KAAK,IAAI,EAAE,CAAC;QAC5D,OAAO,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;IACvC,CAAC;IAEM,YAAY,CAAC,IAAY;QAC9B,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAEM,mBAAmB,CAAC,IAAY;QACrC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;QACxB,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAEM,kBAAkB,CAAC,IAAY;QACpC,IAAI,CAAC,aAAa,GAAG,IAAI,KAAK,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC;QACjD,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAEM,YAAY,CAAC,IAAY;QAC9B,IAAI,CAAC,SAAS,GAAG,IAAI,IAAI,EAAE,CAAC;QAC5B,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;QACxB,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAEM,eAAe,CAAC,IAAY;QACjC,IAAI,CAAC,YAAY,GAAG,IAAI,IAAI,EAAE,CAAC;QAC/B,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;QACxB,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAEM,QAAQ;QACb,MAAM,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;QACrC,IAAI,CAAC,WAAW,EAAE;YAChB,IAAI,CAAC,SAAS,GAAG,mBAAmB,CAAC;SACtC;aAAM,IAAI,IAAI,CAAC,eAAe,CAAC,WAAW,CAAC,EAAE;YAC5C,IAAI,CAAC,SAAS,GAAG,0CAA0C,CAAC;SAC7D;aAAM;YACL,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;SACvB;QAED,MAAM,MAAM,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAElC,MAAM,MAAM,GAAG,IAAI,CAAC,cAAc,CAAC;QACnC,IAAI,CAAC,MAAM,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE;YAClC,IAAI,CAAC,SAAS,GAAG,4DAA4D,CAAC;SAC/E;aAAM;YACL,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;SACvB;QAED,IAAI,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,SAAS,IAAI,CAAC,MAAM,EAAE;YACnE,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;YACxB,OAAO,IAAI,CAAC;SACb;QAED,MAAM,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAC3E,MAAM,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAE3E,OAAO;YACL,IAAI,EAAE,WAAW;YACjB,WAAW,EAAE,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,IAAI,IAAI;YAC5C,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,OAAO,EAAE,IAAI;YACb,IAAI;YACJ,cAAc,EAAE,IAAI,CAAC,cAAc;SACpC,CAAC;IACJ,CAAC;IAEO,WAAW;QACjB,MAAM,OAAO,GAAG,CAAC,IAAI,CAAC,SAAS,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;QAC9C,MAAM,OAAO,GAAG,CAAC,IAAI,CAAC,YAAY,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;QACjD,IAAI,CAAC,OAAO,IAAI,CAAC,OAAO,EAAE;YACxB,IAAI,CAAC,WAAW,GAAG,gCAAgC,CAAC;YACpD,OAAO,IAAI,CAAC;SACb;QACD,MAAM,SAAS,GAAG,iCAAiC,CAAC,aAAa,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAChF,MAAM,SAAS,GAAG,iCAAiC,CAAC,aAAa,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAChF,IAAI,CAAC,SAAS,IAAI,CAAC,SAAS,EAAE;YAC5B,IAAI,CAAC,WAAW,GAAG,yCAAyC,CAAC;YAC7D,OAAO,IAAI,CAAC;SACb;QACD,MAAM,IAAI,GAAG,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QACxC,MAAM,IAAI,GAAG,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QACxC,IAAI,IAAI,GAAG,CAAC,IAAI,IAAI,GAAG,CAAC,EAAE;YACxB,IAAI,CAAC,WAAW,GAAG,sCAAsC,CAAC;YAC1D,OAAO,IAAI,CAAC;SACb;QACD,IAAI,IAAI,GAAG,iCAAiC,CAAC,OAAO;eAC/C,IAAI,GAAG,iCAAiC,CAAC,OAAO,EAAE;YACrD,IAAI,CAAC,WAAW,GAAG,mCAAmC,iCAAiC,CAAC,OAAO,GAAG,CAAC;YACnG,OAAO,IAAI,CAAC;SACb;QACD,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;QACxB,OAAO,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC;IACxB,CAAC;IAED,0EAA0E;IAElE,oBAAoB,CAAC,IAA4B;QACvD,MAAM,OAAO,GAAmB,CAAC,IAAI,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC;QAC9E,OAAO;YACL,GAAG,EAAE,gBAAgB;YACrB,KAAK,EAAE,EAAE;YACT,WAAW,EAAE,sBAAsB;YACnC,QAAQ,EAAE,IAAI;YACd,UAAU,EAAE,IAAI;YAChB,WAAW,EAAE,UAAU;YACvB,aAAa,EAAE,KAAK;YACpB,OAAO;SACR,CAAC;IACJ,CAAC;IAED,4EAA4E;IAEpE,0BAA0B;QAChC,MAAM,MAAM,GAAG,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;QACrE,OAAO;YACL,GAAG,EAAE,gBAAgB;YACrB,KAAK,EAAE,EAAE;YACT,WAAW,EAAE,sBAAsB;YACnC,QAAQ,EAAE,IAAI;YACd,UAAU,EAAE,IAAI;YAChB,WAAW,EAAE,UAAU;YACvB,aAAa,EAAE,KAAK;YACpB,YAAY,EAAE,IAAI;YAClB,SAAS,EAAE,IAAI,CAAC,aAAa;YAC7B,OAAO,EAAE,IAAI,CAAC,iBAAiB,CAAC,MAAM,GAAG,IAAI,CAAC,gBAAgB;YAC9D,OAAO,EAAE,MAAM;YACf,QAAQ,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC;YACxD,UAAU,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,aAAa,EAAE;SACvC,CAAC;IACJ,CAAC;IAEO,aAAa;QACnB,IAAI,IAAI,CAAC,aAAa,EAAE;YAAE,OAAO;SAAE;QACnC,IAAI,IAAI,CAAC,iBAAiB,CAAC,MAAM,IAAI,IAAI,CAAC,gBAAgB,EAAE;YAAE,OAAO;SAAE;QACvE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,iBAAiB,EAAE,IAAI,CAAC,gBAAgB,GAAG,CAAC,EAAE,YAAY,CAAC,IAAI,CAAC,CAAC;IACtF,CAAC;IAEO,QAAQ,CAAC,IAAY,EAAE,IAAY,EAAE,MAAe;QAC1D,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;QAC1B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,0BAA0B,EAAE,CAAC;QACnD,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;QAExB,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,CAAC;aACvB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;aAC9B,SAAS,CAAC,CAAC,EAAE,IAAI,EAAE,CAAC,EAAE,IAAI,EAAE,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE;YAC3C,IAAI,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC,EAAE,MAAM,EAAE,OAAO,CAAC,CAAC;QACxC,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,SAAS,CAAC,IAAY,EAAE,IAAY;QAK1C,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;YAClB,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,OAAO,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC;SAC7D;QACD,OAAO,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC,IAAI,CAClD,UAAU,CAAC,GAAG,EAAE,CAAC,EAAE,CAAmB,EAAE,KAAK,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC,EAC/D,SAAS,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC,CAAC,CAClD,CAAC;IACJ,CAAC;IAEO,SAAS,CACf,IAAY,EACZ,IAAY,EACZ,MAAe,EACf,OAA0B;QAE1B,uEAAuE;QACvE,sEAAsE;QACtE,wEAAwE;QACxE,+DAA+D;QAC/D,IAAI,MAAM,IAAI,IAAI,KAAK,IAAI,CAAC,iBAAiB,EAAE;YAAE,OAAO;SAAE;QAE1D,IAAI,CAAC,MAAM,EAAE;YACX,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;SAC/B;QAED,MAAM,KAAK,GAAG,CAAC,OAAO,EAAE,KAAK,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC;QACtE,MAAM,KAAK,GAAG,OAAO,EAAE,KAAK,IAAI,KAAK,CAAC,MAAM,CAAC;QAE7C,IAAI,MAAM,EAAE;YACV,sEAAsE;YACtE,MAAM,IAAI,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YACpE,MAAM,KAAK,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YACzD,IAAI,CAAC,iBAAiB,GAAG,CAAC,GAAG,IAAI,CAAC,iBAAiB,EAAE,GAAG,KAAK,CAAC,CAAC;SAChE;aAAM;YACL,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC;SAChC;QACD,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;QAC9B,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;QAC7B,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAC3B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,0BAA0B,EAAE,CAAC;QACnD,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAED;;;2CAGuC;IAC/B,wBAAwB,CAAC,OAAuB;QACtD,IAAI,IAAI,CAAC,gBAAgB,CAAC,IAAI,KAAK,CAAC,EAAE;YAAE,OAAO,OAAO,CAAC;SAAE;QACzD,MAAM,UAAU,GAAG,IAAI,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QAC3D,MAAM,UAAU,GAAmB,EAAE,CAAC;QACtC,KAAK,MAAM,GAAG,IAAI,IAAI,CAAC,gBAAgB,CAAC,MAAM,EAAE,EAAE;YAChD,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,EAAE;gBAAE,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;aAAE;SAC/D;QACD,OAAO,CAAC,GAAG,UAAU,EAAE,GAAG,OAAO,CAAC,CAAC;IACrC,CAAC;IAED;;;;gDAI4C;IACpC,oBAAoB,CAAC,KAAU;QACrC,MAAM,GAAG,GAAa,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;YACxC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YACzD,CAAC,CAAC,EAAE,CAAC;QACP,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC,EAAE;YAAE,OAAO;SAAE;QACjC,KAAK,MAAM,EAAE,IAAI,GAAG,EAAE;YACpB,IAAI,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE;gBAAE,SAAS;aAAE;YAChD,MAAM,WAAW,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC,CAAC;YAC1E,IAAI,WAAW,EAAE;gBAAE,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,EAAE,EAAE,WAAW,CAAC,CAAC;aAAE;SACjE;IACH,CAAC;IAED,4EAA4E;IAEpE,cAAc,CAAC,CAAuB;QAC5C,OAAO;YACL,EAAE,EAAE,CAAC,CAAC,EAAE;YACR,KAAK,EAAE,CAAC,CAAC,EAAE;YACX,IAAI,EAAE,CAAC,CAAC,IAAI;YACZ,KAAK,EAAE,CAAC,CAAC,IAAI;YACb,WAAW,EAAE,CAAC,CAAC,KAAK;SACrB,CAAC;IACJ,CAAC;IAEO,eAAe,CAAC,SAAiB;QACvC,MAAM,QAAQ,GAAG,CAAC,IAAI,CAAC,aAAa,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC,CAAC;QACrF,MAAM,OAAO,GAAG,SAAS,CAAC,WAAW,EAAE,CAAC;QACxC,IAAI,IAAI,CAAC,IAAI,KAAK,MAAM,EAAE;YACxB,MAAM,QAAQ,GAAG,CAAC,IAAI,CAAC,YAAY,EAAE,IAAI,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;YACtE,OAAO,OAAO,KAAK,QAAQ,IAAI,QAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;SAC3D;QACD,OAAO,QAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;IACpC,CAAC;;AA1WuB,+CAAa,GAAG,eAAgB,CAAA;AAChC,yCAAO,GAAG,GAAI,CAAA;8HAF3B,iCAAiC;kHAAjC,iCAAiC,gUCvB9C,y6FAkFA;2FD3Da,iCAAiC;kBAN7C,SAAS;+BACE,kCAAkC,mBAE3B,uBAAuB,CAAC,MAAM,QACzC,EAAE,KAAK,EAAE,aAAa,EAAE,KAAK,EAAE,2BAA2B,EAAE;wGAMzD,IAAI;sBAAZ,KAAK;gBACG,YAAY;sBAApB,KAAK;gBACG,aAAa;sBAArB,KAAK;gBACG,YAAY;sBAApB,KAAK;gBAGG,QAAQ;sBAAhB,KAAK;gBACG,QAAQ;sBAAhB,KAAK","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Input, OnDestroy, OnInit } from '@angular/core';\nimport { FormControl, FormGroup } from '@angular/forms';\nimport { Observable, Subject, of } from 'rxjs';\nimport { catchError, debounceTime, distinctUntilChanged, switchMap, takeUntil } from 'rxjs/operators';\n\nimport {\n  DynamicSelectFieldConfig,\n  SelectOption,\n} from '../dynamic-select/dynamic-select-field.component';\nimport {\n  TdpDialogPermission,\n  TdpDialogValue,\n  TdpEnvSearchFn,\n  TdpEnvSearchPage,\n  TdpEnvironmentOption,\n} from './new-test-data-profile-dialog.models';\n\n@Component({\n  selector: 'cqa-new-test-data-profile-dialog',\n  templateUrl: './new-test-data-profile-dialog.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' },\n})\nexport class NewTestDataProfileDialogComponent implements OnInit, OnDestroy {\n  private static readonly INTEGER_REGEX = /^\\s*(\\d+)\\s*$/;\n  private static readonly MAX_DIM = 100;\n\n  @Input() mode: 'create' | 'edit' = 'create';\n  @Input() initialValue?: Partial<TdpDialogValue>;\n  @Input() existingNames: string[] = [];\n  @Input() environments: TdpEnvironmentOption[] = [];\n  /** When provided, the env dropdown uses paginated server-search instead of\n   *  the static `environments` list. */\n  @Input() searchFn?: TdpEnvSearchFn;\n  @Input() pageSize = 50;\n\n  public name = '';\n  public description = '';\n  public readWriteMode: TdpDialogPermission = 'RW';\n  public rowsInput = '1';\n  public columnsInput = '1';\n\n  public readonly envForm = new FormGroup({\n    environmentIds: new FormControl([]),\n  });\n  public envConfig: DynamicSelectFieldConfig = this.buildStaticEnvConfig([]);\n\n  public nameError: string | null = null;\n  public matrixError: string | null = null;\n  public envsError: string | null = null;\n\n  // Server-search state — populated only when `searchFn` is provided.\n  private currentEnvOptions: SelectOption[] = [];\n  private totalEnvElements = 0;\n  private currentSearchTerm = '';\n  private currentPageIndex = 0;\n  private isLoadingEnvs = false;\n  /** Cache of every env the user has selected at any point during this dialog\n   *  session — keyed by id. Merged into the visible options on every page\n   *  refresh so the multi-select trigger always shows the right names even\n   *  after the user types a search that filters the selection out. */\n  private readonly selectedEnvCache = new Map<number, SelectOption>();\n\n  private readonly searchInput$ = new Subject<string>();\n  private readonly destroy$ = new Subject<void>();\n\n  constructor(private readonly cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    if (this.initialValue) {\n      this.name = this.initialValue.name ?? '';\n      this.description = this.initialValue.description ?? '';\n      this.readWriteMode = this.initialValue.readWriteMode ?? 'RW';\n      const colCount = this.initialValue.columns?.length ?? 1;\n      const rowCount = this.initialValue.rows?.length ?? 1;\n      this.rowsInput = String(rowCount);\n      this.columnsInput = String(colCount);\n      const ids = this.initialValue.environmentIds ?? [];\n      this.envForm.get('environmentIds')!.setValue(ids);\n    }\n\n    if (this.searchFn) {\n      this.envConfig = this.buildServerSearchEnvConfig();\n      // Debounced typing pipeline. Each new term resets to page 0 and replaces\n      // the visible options. switchMap drops in-flight responses for stale terms.\n      this.searchInput$\n        .pipe(\n          debounceTime(300),\n          distinctUntilChanged(),\n          switchMap(term => this.fetchPage(term, 0)),\n          takeUntil(this.destroy$),\n        )\n        .subscribe(({ term, page, payload }) => {\n          this.applyPage(term, page, /* append */ false, payload);\n        });\n      // Pre-fetch page 0 so the dropdown is populated by the time the user opens it.\n      this.runFetch('', 0, /* append */ false);\n    } else {\n      this.envConfig = this.buildStaticEnvConfig(this.environments ?? []);\n    }\n\n    // Clear envs error as soon as the user picks something.\n    this.envForm.get('environmentIds')!.valueChanges\n      .pipe(takeUntil(this.destroy$))\n      .subscribe(value => {\n        if (this.envsError) {\n          this.envsError = null;\n          this.cdr.markForCheck();\n        }\n        if (this.searchFn) {\n          this.rememberSelectedEnvs(value);\n        }\n      });\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  public get title(): string {\n    return this.mode === 'edit' ? 'Edit test data profile' : 'New test data profile';\n  }\n\n  public get subtitle(): string {\n    return 'Define the matrix; data values can be added per environment after creation.';\n  }\n\n  public get primaryButtonLabel(): string {\n    return this.mode === 'edit' ? 'Save changes' : 'Create profile';\n  }\n\n  public get nameErrorsArray(): string[] { return this.nameError ? [this.nameError] : []; }\n\n  public get selectedEnvIds(): number[] {\n    const val = this.envForm.get('environmentIds')!.value ?? [];\n    return Array.isArray(val) ? val : [];\n  }\n\n  public onNameChange(next: string): void {\n    this.name = next;\n    this.nameError = null;\n    this.cdr.markForCheck();\n  }\n\n  public onDescriptionChange(next: string): void {\n    this.description = next;\n    this.cdr.markForCheck();\n  }\n\n  public onPermissionChange(next: string): void {\n    this.readWriteMode = next === 'RO' ? 'RO' : 'RW';\n    this.cdr.markForCheck();\n  }\n\n  public onRowsChange(next: string): void {\n    this.rowsInput = next ?? '';\n    this.matrixError = null;\n    this.cdr.markForCheck();\n  }\n\n  public onColumnsChange(next: string): void {\n    this.columnsInput = next ?? '';\n    this.matrixError = null;\n    this.cdr.markForCheck();\n  }\n\n  public getValue(): TdpDialogValue | null {\n    const trimmedName = this.name.trim();\n    if (!trimmedName) {\n      this.nameError = 'Name is required.';\n    } else if (this.isDuplicateName(trimmedName)) {\n      this.nameError = 'A profile with this name already exists.';\n    } else {\n      this.nameError = null;\n    }\n\n    const parsed = this.parseMatrix();\n\n    const envIds = this.selectedEnvIds;\n    if (!envIds || envIds.length === 0) {\n      this.envsError = 'Select at least one environment to assign this profile to.';\n    } else {\n      this.envsError = null;\n    }\n\n    if (this.nameError || this.matrixError || this.envsError || !parsed) {\n      this.cdr.markForCheck();\n      return null;\n    }\n\n    const cols = Array.from({ length: parsed.cols }, (_, i) => `col_${i + 1}`);\n    const rows = Array.from({ length: parsed.rows }, (_, i) => `row_${i + 1}`);\n\n    return {\n      name: trimmedName,\n      description: this.description.trim() || null,\n      readWriteMode: this.readWriteMode,\n      columns: cols,\n      rows,\n      environmentIds: this.selectedEnvIds,\n    };\n  }\n\n  private parseMatrix(): { rows: number; cols: number } | null {\n    const rawRows = (this.rowsInput ?? '').trim();\n    const rawCols = (this.columnsInput ?? '').trim();\n    if (!rawRows || !rawCols) {\n      this.matrixError = 'Rows and columns are required.';\n      return null;\n    }\n    const rowsMatch = NewTestDataProfileDialogComponent.INTEGER_REGEX.exec(rawRows);\n    const colsMatch = NewTestDataProfileDialogComponent.INTEGER_REGEX.exec(rawCols);\n    if (!rowsMatch || !colsMatch) {\n      this.matrixError = 'Rows and columns must be whole numbers.';\n      return null;\n    }\n    const rows = parseInt(rowsMatch[1], 10);\n    const cols = parseInt(colsMatch[1], 10);\n    if (rows < 1 || cols < 1) {\n      this.matrixError = 'Rows and columns must be at least 1.';\n      return null;\n    }\n    if (rows > NewTestDataProfileDialogComponent.MAX_DIM\n      || cols > NewTestDataProfileDialogComponent.MAX_DIM) {\n      this.matrixError = `Rows and columns can be at most ${NewTestDataProfileDialogComponent.MAX_DIM}.`;\n      return null;\n    }\n    this.matrixError = null;\n    return { rows, cols };\n  }\n\n  // -- Static-options mode (legacy) ---------------------------------------\n\n  private buildStaticEnvConfig(envs: TdpEnvironmentOption[]): DynamicSelectFieldConfig {\n    const options: SelectOption[] = (envs || []).map(e => this.toSelectOption(e));\n    return {\n      key: 'environmentIds',\n      label: '',\n      placeholder: 'Select environments…',\n      multiple: true,\n      searchable: true,\n      optionStyle: 'checkbox',\n      showSelectAll: false,\n      options,\n    };\n  }\n\n  // -- Server-search mode ---------------------------------------------------\n\n  private buildServerSearchEnvConfig(): DynamicSelectFieldConfig {\n    const merged = this.mergeSelectedIntoOptions(this.currentEnvOptions);\n    return {\n      key: 'environmentIds',\n      label: '',\n      placeholder: 'Select environments…',\n      multiple: true,\n      searchable: true,\n      optionStyle: 'checkbox',\n      showSelectAll: false,\n      serverSearch: true,\n      isLoading: this.isLoadingEnvs,\n      hasMore: this.currentEnvOptions.length < this.totalEnvElements,\n      options: merged,\n      onSearch: (q: string) => this.searchInput$.next(q ?? ''),\n      onLoadMore: () => this.onEnvLoadMore(),\n    };\n  }\n\n  private onEnvLoadMore(): void {\n    if (this.isLoadingEnvs) { return; }\n    if (this.currentEnvOptions.length >= this.totalEnvElements) { return; }\n    this.runFetch(this.currentSearchTerm, this.currentPageIndex + 1, /* append */ true);\n  }\n\n  private runFetch(term: string, page: number, append: boolean): void {\n    this.isLoadingEnvs = true;\n    this.envConfig = this.buildServerSearchEnvConfig();\n    this.cdr.markForCheck();\n\n    this.fetchPage(term, page)\n      .pipe(takeUntil(this.destroy$))\n      .subscribe(({ term: t, page: p, payload }) => {\n        this.applyPage(t, p, append, payload);\n      });\n  }\n\n  private fetchPage(term: string, page: number): Observable<{\n    term: string;\n    page: number;\n    payload: TdpEnvSearchPage;\n  }> {\n    if (!this.searchFn) {\n      return of({ term, page, payload: { items: [], total: 0 } });\n    }\n    return this.searchFn(term, page, this.pageSize).pipe(\n      catchError(() => of<TdpEnvSearchPage>({ items: [], total: 0 })),\n      switchMap(payload => of({ term, page, payload })),\n    );\n  }\n\n  private applyPage(\n    term: string,\n    page: number,\n    append: boolean,\n    payload?: TdpEnvSearchPage,\n  ): void {\n    // Drop stale load-more responses — if the user typed a new search term\n    // while a load-more was in flight, this append is for an old term and\n    // would corrupt the freshly-replaced options list. The search$ pipeline\n    // already drops stale typing-debounce responses via switchMap.\n    if (append && term !== this.currentSearchTerm) { return; }\n\n    if (!append) {\n      this.currentSearchTerm = term;\n    }\n\n    const items = (payload?.items ?? []).map(e => this.toSelectOption(e));\n    const total = payload?.total ?? items.length;\n\n    if (append) {\n      // Dedupe by id so a re-fetch can't double up an already-rendered env.\n      const seen = new Set(this.currentEnvOptions.map(o => Number(o.id)));\n      const fresh = items.filter(o => !seen.has(Number(o.id)));\n      this.currentEnvOptions = [...this.currentEnvOptions, ...fresh];\n    } else {\n      this.currentEnvOptions = items;\n    }\n    this.totalEnvElements = total;\n    this.currentPageIndex = page;\n    this.isLoadingEnvs = false;\n    this.envConfig = this.buildServerSearchEnvConfig();\n    this.cdr.markForCheck();\n  }\n\n  /** Merge cached selected options with the visible page so the multi-select\n   *  trigger always shows the right names — even when the user types a query\n   *  that filters previously selected envs out of view. Selected ones come\n   *  first to keep the state visible. */\n  private mergeSelectedIntoOptions(visible: SelectOption[]): SelectOption[] {\n    if (this.selectedEnvCache.size === 0) { return visible; }\n    const visibleIds = new Set(visible.map(o => Number(o.id)));\n    const onlyCached: SelectOption[] = [];\n    for (const opt of this.selectedEnvCache.values()) {\n      if (!visibleIds.has(Number(opt.id))) { onlyCached.push(opt); }\n    }\n    return [...onlyCached, ...visible];\n  }\n\n  /** Update the selected-env cache from the FormControl value so the trigger\n   *  display + future merges reflect the latest selection. We cache the full\n   *  SelectOption (id + name) by looking up either the visible page or the\n   *  prior cache. Selections are accumulated, never removed — even unselect →\n   *  reselect works without a fresh fetch. */\n  private rememberSelectedEnvs(value: any): void {\n    const ids: number[] = Array.isArray(value)\n      ? value.map(v => Number(v)).filter(n => !Number.isNaN(n))\n      : [];\n    if (ids.length === 0) { return; }\n    for (const id of ids) {\n      if (this.selectedEnvCache.has(id)) { continue; }\n      const fromVisible = this.currentEnvOptions.find(o => Number(o.id) === id);\n      if (fromVisible) { this.selectedEnvCache.set(id, fromVisible); }\n    }\n  }\n\n  // -- Helpers --------------------------------------------------------------\n\n  private toSelectOption(e: TdpEnvironmentOption): SelectOption {\n    return {\n      id: e.id,\n      value: e.id,\n      name: e.name,\n      label: e.name,\n      statusColor: e.color,\n    };\n  }\n\n  private isDuplicateName(candidate: string): boolean {\n    const existing = (this.existingNames ?? []).map(n => (n ?? '').trim().toLowerCase());\n    const lowered = candidate.toLowerCase();\n    if (this.mode === 'edit') {\n      const original = (this.initialValue?.name ?? '').trim().toLowerCase();\n      return lowered !== original && existing.includes(lowered);\n    }\n    return existing.includes(lowered);\n  }\n}\n","<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n  <!-- Name + Permission -->\n  <div class=\"cqa-grid cqa-grid-cols-[1fr_auto] cqa-gap-4 cqa-items-start\">\n    <div class=\"cqa-flex cqa-flex-col\">\n      <cqa-custom-input\n        label=\"Profile name\"\n        placeholder=\"e.g. Guest Personas\"\n        type=\"text\"\n        [value]=\"name\"\n        [required]=\"true\"\n        [fullWidth]=\"true\"\n        [errors]=\"nameErrorsArray\"\n        (valueChange)=\"onNameChange($event)\">\n      </cqa-custom-input>\n    </div>\n    <div class=\"cqa-flex cqa-flex-col\">\n      <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Permission</label>\n      <cqa-permission-toggle\n        [value]=\"readWriteMode\"\n        (valueChange)=\"onPermissionChange($event)\">\n      </cqa-permission-toggle>\n    </div>\n  </div>\n\n  <!-- Description -->\n  <div class=\"cqa-flex cqa-flex-col\">\n    <cqa-custom-input\n      label=\"Description\"\n      placeholder=\"What data is this profile for?\"\n      type=\"text\"\n      [value]=\"description\"\n      [fullWidth]=\"true\"\n      (valueChange)=\"onDescriptionChange($event)\">\n    </cqa-custom-input>\n  </div>\n\n  <!-- Rows + Columns matrix -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-1\">\n    <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n      <cqa-custom-input\n        label=\"Rows\"\n        placeholder=\"e.g. 2\"\n        type=\"number\"\n        [value]=\"rowsInput\"\n        [fullWidth]=\"true\"\n        (valueChange)=\"onRowsChange($event)\">\n      </cqa-custom-input>\n      <cqa-custom-input\n        label=\"Columns\"\n        placeholder=\"e.g. 3\"\n        type=\"number\"\n        [value]=\"columnsInput\"\n        [fullWidth]=\"true\"\n        (valueChange)=\"onColumnsChange($event)\">\n      </cqa-custom-input>\n    </div>\n    <span *ngIf=\"matrixError\" class=\"cqa-flex cqa-items-center cqa-gap-1 cqa-text-xs cqa-text-red-600\">\n      <mat-icon class=\"cqa-flex-none\" style=\"font-size:14px;width:14px;height:14px;line-height:14px;\">error</mat-icon>\n      <span>{{ matrixError }}</span>\n    </span>\n  </div>\n\n  <!-- Assign to environments -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n    <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">\n      Assign to environments <span class=\"cqa-text-red-600\">*</span>\n    </label>\n    <cqa-dynamic-select\n      [form]=\"envForm\"\n      [config]=\"envConfig\">\n    </cqa-dynamic-select>\n    <span *ngIf=\"envsError\" class=\"cqa-flex cqa-items-center cqa-gap-1 cqa-text-xs cqa-text-red-600\">\n      <mat-icon class=\"cqa-flex-none\" style=\"font-size:14px;width:14px;height:14px;line-height:14px;\">error</mat-icon>\n      <span>{{ envsError }}</span>\n    </span>\n    <span *ngIf=\"!envsError\" class=\"cqa-text-xs cqa-text-gray-500\">\n      Each environment gets an independent copy of the data (same columns, separate rows).\n    </span>\n  </div>\n\n</div>\n"]}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmV3LXRlc3QtZGF0YS1wcm9maWxlLWRpYWxvZy5tb2RlbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL25ldy10ZXN0LWRhdGEtcHJvZmlsZS1kaWFsb2cvbmV3LXRlc3QtZGF0YS1wcm9maWxlLWRpYWxvZy5tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE9ic2VydmFibGUgfSBmcm9tICdyeGpzJztcblxuZXhwb3J0IHR5cGUgVGRwRGlhbG9nUGVybWlzc2lvbiA9ICdSTycgfCAnUlcnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFRkcERpYWxvZ1ZhbHVlIHtcbiAgbmFtZTogc3RyaW5nO1xuICBkZXNjcmlwdGlvbjogc3RyaW5nIHwgbnVsbDtcbiAgcmVhZFdyaXRlTW9kZTogVGRwRGlhbG9nUGVybWlzc2lvbjtcbiAgY29sdW1uczogc3RyaW5nW107XG4gIHJvd3M6IHN0cmluZ1tdO1xuICBlbnZpcm9ubWVudElkczogbnVtYmVyW107XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgVGRwRW52aXJvbm1lbnRPcHRpb24ge1xuICBpZDogbnVtYmVyO1xuICBuYW1lOiBzdHJpbmc7XG4gIGNvbG9yPzogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFRkcEVudlNlYXJjaFBhZ2Uge1xuICBpdGVtczogVGRwRW52aXJvbm1lbnRPcHRpb25bXTtcbiAgdG90YWw6IG51bWJlcjtcbn1cblxuLyoqIFBhZ2luYXRlZCwgZGVib3VuY2VkIHNlcnZlci1zZWFyY2ggY2FsbGJhY2sgZm9yIHRoZSBBc3NpZ24tdG8tZW52aXJvbm1lbnRzXG4gKiAgZHJvcGRvd24uIFRoZSBkaWFsb2cgY2FsbHMgYCgnJywgMCwgcGFnZVNpemUpYCBvbiBpbml0IGFuZCBlYWNoIHBhbmVsIG9wZW4sXG4gKiAgYWdhaW4gd2l0aCBhIGZyZXNoIHRlcm0gd2hlbmV2ZXIgdGhlIHVzZXIgdHlwZXMgKGRlYm91bmNlZCAzMDBtcyksIGFuZCBhZ2FpblxuICogIG9uIHNjcm9sbC1uZWFyLWJvdHRvbSBmb3IgcGFnaW5hdGlvbi4gQ2FsbGVyIHdyYXBzIHRoZSB3b3Jrc3BhY2UgZW52IGxpc3RcbiAqICBzZXJ2aWNlLiAqL1xuZXhwb3J0IHR5cGUgVGRwRW52U2VhcmNoRm4gPVxuICAodGVybTogc3RyaW5nLCBwYWdlOiBudW1iZXIsIHNpemU6IG51bWJlcikgPT4gT2JzZXJ2YWJsZTxUZHBFbnZTZWFyY2hQYWdlPjtcblxuZXhwb3J0IGludGVyZmFjZSBOZXdUZXN0RGF0YVByb2ZpbGVEaWFsb2dJbnB1dHMge1xuICBtb2RlPzogJ2NyZWF0ZScgfCAnZWRpdCc7XG4gIGluaXRpYWxWYWx1ZT86IFBhcnRpYWw8VGRwRGlhbG9nVmFsdWU+O1xuICBleGlzdGluZ05hbWVzPzogc3RyaW5nW107XG4gIC8qKiBTdGF0aWMgZW52IGxpc3QuIEtlcHQgZm9yIGJhY2t3YXJkIGNvbXBhdGliaWxpdHkg4oCUIHByZWZlciBgc2VhcmNoRm5gXG4gICAqICBmb3Igd29ya3NwYWNlcyB3aXRoIG1vcmUgdGhhbiBhIGNvdXBsZSBkb3plbiBlbnZpcm9ubWVudHMuICovXG4gIGVudmlyb25tZW50cz86IFRkcEVudmlyb25tZW50T3B0aW9uW107XG4gIC8qKiBTZXJ2ZXItc2VhcmNoIHNvdXJjZS4gV2hlbiBwcm92aWRlZCwgdGhlIGVudiBkcm9wZG93biBzd2l0Y2hlcyB0b1xuICAgKiAgcGFnaW5hdGVkIHNlcnZlci1zaWRlIHNlYXJjaCBhbmQgYGVudmlyb25tZW50c2AgaXMgaWdub3JlZC4gKi9cbiAgc2VhcmNoRm4/OiBUZHBFbnZTZWFyY2hGbjtcbiAgLyoqIFBhZ2Ugc2l6ZSBmb3IgYHNlYXJjaEZuYC4gRGVmYXVsdHMgdG8gNTAuICovXG4gIHBhZ2VTaXplPzogbnVtYmVyO1xufVxuIl19
|
|
@@ -86,7 +86,7 @@ export class QuestionnaireListComponent {
|
|
|
86
86
|
}
|
|
87
87
|
}
|
|
88
88
|
QuestionnaireListComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: QuestionnaireListComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
89
|
-
QuestionnaireListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: QuestionnaireListComponent, selector: "cqa-questionnaire-list", inputs: { items: "items", defaultPageSize: "defaultPageSize", pageSizeOptions: "pageSizeOptions", inputPlaceholder: "inputPlaceholder" }, outputs: { answerChange: "answerChange" }, host: { listeners: { "document:click": "onDocumentClick($event)" }, classAttribute: "cqa-ui-root" }, viewQueries: [{ propertyName: "pageSizeDropdownContainer", first: true, predicate: ["pageSizeDropdownContainer"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-border cqa-border-solid cqa-border-[#E2E8F0] cqa-rounded-xl cqa-shadow-[0_1px_2px_rgba(0,0,0,0.05)] cqa-font-inter\" role=\"region\" aria-label=\"Clarification Questionnaire\">\n\n <!-- Header -->\n <div class=\"cqa-py-3 cqa-px-6 cqa-bg-[#EEF2FF80]\">\n <h2 class=\"cqa-text-[18px] cqa-font-medium cqa-text-[#0F172A] cqa-leading-[22px] cqa-m-0 cqa-mb-1\">Clarification Questionnaire</h2>\n <p class=\"cqa-text-[14px] cqa-text-[#475569] cqa-leading-[18px] cqa-m-0\">\n The AI has generated these questions to resolve ambiguities in the requirement.\n Answering these will regenerate specific test cases.\n </p>\n </div>\n\n <div class=\"cqa-px-6 cqa-py-3\">\n <!-- Question rows -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <ng-container *ngIf=\"visibleItems.length; else emptyState\">\n <div *ngFor=\"let item of visibleItems; last as isLast; trackBy: trackById\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2.5 cqa-mb-2\">\n <span class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-[24px] cqa-h-[24px] cqa-min-w-[24px] cqa-rounded-full cqa-bg-[#EEF2FF] cqa-text-[#4F46E5] cqa-text-[12px] cqa-leading-[18px] cqa-font-bold cqa-shrink-0\">{{ item.index }}</span>\n <label class=\"cqa-text-[12px] cqa-leading-[15px] cqa-text-[#0F172A]\" [for]=\"getInputId(item.id)\">{{ item.question }}</label>\n </div>\n <cqa-custom-input\n [inputId]=\"getInputId(item.id)\"\n [value]=\"item.answer\"\n [placeholder]=\"inputPlaceholder\"\n [fullWidth]=\"true\"\n (valueChange)=\"onAnswerInput(item.id, $event)\"\n class=\"cqa-w-full\"\n ></cqa-custom-input>\n </div>\n </ng-container>\n <ng-template #emptyState>\n <div class=\"cqa-py-6 cqa-text-[13px] cqa-text-[#6B7280] cqa-text-center\">No questions available.</div>\n </ng-template>\n </div>\n\n <!-- Footer: aligned with table pagination design -->\n <div class=\"cqa-mt-3 table-footer-pagination cqa-text-[#717182] cqa-text-[12px] cqa-leading-[15px] cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-flex-wrap cqa-px-[21px] cqa-py-1.5\" aria-live=\"polite\" style=\"border-top: 1px solid #E5E7EB;\">\n <div class=\"pagination-info cqa-flex cqa-items-center cqa-gap-[7px] cqa-relative\">\n <span class=\"rows-label\">Questions per page</span>\n <div class=\"cqa-relative\" #pageSizeDropdownContainer>\n <button\n type=\"button\"\n class=\"cqa-inline-flex cqa-items-center cqa-gap-[24px] cqa-text-[#0A0A0A] cqa-px-[11.5px] cqa-py-[6.75px]\"\n (click)=\"togglePageSizeMenu()\"\n [attr.aria-expanded]=\"pageSizeOpen\"\n aria-haspopup=\"listbox\"\n aria-label=\"Questions per page\"\n >\n {{ pageSize }}\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><g opacity=\"0.5\"><path d=\"M3.5 5.25L7 8.75L10.5 5.25\" stroke=\"#717182\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></g></svg>\n </button>\n <div\n *ngIf=\"pageSizeOpen\"\n class=\"cqa-absolute cqa-z-[100] cqa-bottom-[calc(100%+8px)] cqa-left-0 cqa-w-[75px] cqa-max-h-[170px] cqa-overflow-auto cqa-rounded-lg cqa-border cqa-border-[#E5E7EB] cqa-bg-white cqa-shadow-[0px_4px_6px_-1px_rgba(0,0,0,0.1)] cqa-p-[5px]\"\n role=\"listbox\"\n [attr.aria-activedescendant]=\"'question-pagesize-' + pageSize\"\n >\n <button\n *ngFor=\"let opt of pageSizeOptions\"\n type=\"button\"\n class=\"cqa-w-full cqa-px-2 cqa-py-[6px] hover:cqa-bg-[#F7F8FA] cqa-text-left cqa-rounded-md cqa-text-black-100\"\n [attr.id]=\"'question-pagesize-' + opt\"\n role=\"option\"\n [attr.aria-selected]=\"pageSize === opt\"\n (click)=\"selectPageSize(opt)\"\n >\n {{ opt }}\n </button>\n </div>\n </div>\n </div>\n\n <div class=\"pagination-right cqa-flex cqa-items-center cqa-gap-[21px]\">\n <div class=\"pagination-range\">\n {{ totalItems ? startIndex + 1 : 0 }}–{{ endIndex }} of {{ totalItems }}\n </div>\n <div class=\"pagination-controls cqa-flex cqa-items-stretch cqa-gap-[3.5px]\">\n <button class=\"pagination-btn cqa-w-[28px] cqa-h-[28px] cqa-min-w-[28px] cqa-rounded-[5px] cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-bg-[#F7F8FA] cqa-flex cqa-items-center cqa-justify-center disabled:cqa-opacity-50 disabled:cqa-cursor-not-allowed\" [disabled]=\"isPrevDisabled\" (click)=\"prevPage()\" aria-label=\"Previous page\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M8.75 10.5L5.25 7L8.75 3.5\" stroke=\"#0A0A0A\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </button>\n <button class=\"pagination-btn cqa-w-[28px] cqa-h-[28px] cqa-min-w-[28px] cqa-rounded-[5px] cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-bg-[#F7F8FA] cqa-flex cqa-items-center cqa-justify-center disabled:cqa-opacity-50 disabled:cqa-cursor-not-allowed\" [disabled]=\"isNextDisabled\" (click)=\"nextPage()\" aria-label=\"Next page\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M5.25 10.5L8.75 7L5.25 3.5\" stroke=\"#0A0A0A\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </button>\n </div>\n </div>\n </div>\n </div>\n\n</div>\n", 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"] }], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
|
|
89
|
+
QuestionnaireListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: QuestionnaireListComponent, selector: "cqa-questionnaire-list", inputs: { items: "items", defaultPageSize: "defaultPageSize", pageSizeOptions: "pageSizeOptions", inputPlaceholder: "inputPlaceholder" }, outputs: { answerChange: "answerChange" }, host: { listeners: { "document:click": "onDocumentClick($event)" }, classAttribute: "cqa-ui-root" }, viewQueries: [{ propertyName: "pageSizeDropdownContainer", first: true, predicate: ["pageSizeDropdownContainer"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-border cqa-border-solid cqa-border-[#E2E8F0] cqa-rounded-xl cqa-shadow-[0_1px_2px_rgba(0,0,0,0.05)] cqa-font-inter\" role=\"region\" aria-label=\"Clarification Questionnaire\">\n\n <!-- Header -->\n <div class=\"cqa-py-3 cqa-px-6 cqa-bg-[#EEF2FF80]\">\n <h2 class=\"cqa-text-[18px] cqa-font-medium cqa-text-[#0F172A] cqa-leading-[22px] cqa-m-0 cqa-mb-1\">Clarification Questionnaire</h2>\n <p class=\"cqa-text-[14px] cqa-text-[#475569] cqa-leading-[18px] cqa-m-0\">\n The AI has generated these questions to resolve ambiguities in the requirement.\n Answering these will regenerate specific test cases.\n </p>\n </div>\n\n <div class=\"cqa-px-6 cqa-py-3\">\n <!-- Question rows -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <ng-container *ngIf=\"visibleItems.length; else emptyState\">\n <div *ngFor=\"let item of visibleItems; last as isLast; trackBy: trackById\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2.5 cqa-mb-2\">\n <span class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-[24px] cqa-h-[24px] cqa-min-w-[24px] cqa-rounded-full cqa-bg-[#EEF2FF] cqa-text-[#4F46E5] cqa-text-[12px] cqa-leading-[18px] cqa-font-bold cqa-shrink-0\">{{ item.index }}</span>\n <label class=\"cqa-text-[12px] cqa-leading-[15px] cqa-text-[#0F172A]\" [for]=\"getInputId(item.id)\">{{ item.question }}</label>\n </div>\n <cqa-custom-input\n [inputId]=\"getInputId(item.id)\"\n [value]=\"item.answer\"\n [placeholder]=\"inputPlaceholder\"\n [fullWidth]=\"true\"\n (valueChange)=\"onAnswerInput(item.id, $event)\"\n class=\"cqa-w-full\"\n ></cqa-custom-input>\n </div>\n </ng-container>\n <ng-template #emptyState>\n <div class=\"cqa-py-6 cqa-text-[13px] cqa-text-[#6B7280] cqa-text-center\">No questions available.</div>\n </ng-template>\n </div>\n\n <!-- Footer: aligned with table pagination design -->\n <div class=\"cqa-mt-3 table-footer-pagination cqa-text-[#717182] cqa-text-[12px] cqa-leading-[15px] cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-flex-wrap cqa-px-[21px] cqa-py-1.5\" aria-live=\"polite\" style=\"border-top: 1px solid #E5E7EB;\">\n <div class=\"pagination-info cqa-flex cqa-items-center cqa-gap-[7px] cqa-relative\">\n <span class=\"rows-label\">Questions per page</span>\n <div class=\"cqa-relative\" #pageSizeDropdownContainer>\n <button\n type=\"button\"\n class=\"cqa-inline-flex cqa-items-center cqa-gap-[24px] cqa-text-[#0A0A0A] cqa-px-[11.5px] cqa-py-[6.75px]\"\n (click)=\"togglePageSizeMenu()\"\n [attr.aria-expanded]=\"pageSizeOpen\"\n aria-haspopup=\"listbox\"\n aria-label=\"Questions per page\"\n >\n {{ pageSize }}\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><g opacity=\"0.5\"><path d=\"M3.5 5.25L7 8.75L10.5 5.25\" stroke=\"#717182\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></g></svg>\n </button>\n <div\n *ngIf=\"pageSizeOpen\"\n class=\"cqa-absolute cqa-z-[100] cqa-bottom-[calc(100%+8px)] cqa-left-0 cqa-w-[75px] cqa-max-h-[170px] cqa-overflow-auto cqa-rounded-lg cqa-border cqa-border-[#E5E7EB] cqa-bg-white cqa-shadow-[0px_4px_6px_-1px_rgba(0,0,0,0.1)] cqa-p-[5px]\"\n role=\"listbox\"\n [attr.aria-activedescendant]=\"'question-pagesize-' + pageSize\"\n >\n <button\n *ngFor=\"let opt of pageSizeOptions\"\n type=\"button\"\n class=\"cqa-w-full cqa-px-2 cqa-py-[6px] hover:cqa-bg-[#F7F8FA] cqa-text-left cqa-rounded-md cqa-text-black-100\"\n [attr.id]=\"'question-pagesize-' + opt\"\n role=\"option\"\n [attr.aria-selected]=\"pageSize === opt\"\n (click)=\"selectPageSize(opt)\"\n >\n {{ opt }}\n </button>\n </div>\n </div>\n </div>\n\n <div class=\"pagination-right cqa-flex cqa-items-center cqa-gap-[21px]\">\n <div class=\"pagination-range\">\n {{ totalItems ? startIndex + 1 : 0 }}–{{ endIndex }} of {{ totalItems }}\n </div>\n <div class=\"pagination-controls cqa-flex cqa-items-stretch cqa-gap-[3.5px]\">\n <button class=\"pagination-btn cqa-w-[28px] cqa-h-[28px] cqa-min-w-[28px] cqa-rounded-[5px] cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-bg-[#F7F8FA] cqa-flex cqa-items-center cqa-justify-center disabled:cqa-opacity-50 disabled:cqa-cursor-not-allowed\" [disabled]=\"isPrevDisabled\" (click)=\"prevPage()\" aria-label=\"Previous page\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M8.75 10.5L5.25 7L8.75 3.5\" stroke=\"#0A0A0A\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </button>\n <button class=\"pagination-btn cqa-w-[28px] cqa-h-[28px] cqa-min-w-[28px] cqa-rounded-[5px] cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-bg-[#F7F8FA] cqa-flex cqa-items-center cqa-justify-center disabled:cqa-opacity-50 disabled:cqa-cursor-not-allowed\" [disabled]=\"isNextDisabled\" (click)=\"nextPage()\" aria-label=\"Next page\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M5.25 10.5L8.75 7L5.25 3.5\" stroke=\"#0A0A0A\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </button>\n </div>\n </div>\n </div>\n </div>\n\n</div>\n", components: [{ type: i1.CustomInputComponent, selector: "cqa-custom-input", inputs: ["inputId", "label", "type", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "inputInlineStyle", "labelInlineStyle", "showPasswordToggle"], outputs: ["valueChange", "blurred", "focused", "enterPressed"] }], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
|
|
90
90
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: QuestionnaireListComponent, decorators: [{
|
|
91
91
|
type: Component,
|
|
92
92
|
args: [{ selector: 'cqa-questionnaire-list', host: { class: 'cqa-ui-root' }, template: "<div class=\"cqa-border cqa-border-solid cqa-border-[#E2E8F0] cqa-rounded-xl cqa-shadow-[0_1px_2px_rgba(0,0,0,0.05)] cqa-font-inter\" role=\"region\" aria-label=\"Clarification Questionnaire\">\n\n <!-- Header -->\n <div class=\"cqa-py-3 cqa-px-6 cqa-bg-[#EEF2FF80]\">\n <h2 class=\"cqa-text-[18px] cqa-font-medium cqa-text-[#0F172A] cqa-leading-[22px] cqa-m-0 cqa-mb-1\">Clarification Questionnaire</h2>\n <p class=\"cqa-text-[14px] cqa-text-[#475569] cqa-leading-[18px] cqa-m-0\">\n The AI has generated these questions to resolve ambiguities in the requirement.\n Answering these will regenerate specific test cases.\n </p>\n </div>\n\n <div class=\"cqa-px-6 cqa-py-3\">\n <!-- Question rows -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <ng-container *ngIf=\"visibleItems.length; else emptyState\">\n <div *ngFor=\"let item of visibleItems; last as isLast; trackBy: trackById\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2.5 cqa-mb-2\">\n <span class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-w-[24px] cqa-h-[24px] cqa-min-w-[24px] cqa-rounded-full cqa-bg-[#EEF2FF] cqa-text-[#4F46E5] cqa-text-[12px] cqa-leading-[18px] cqa-font-bold cqa-shrink-0\">{{ item.index }}</span>\n <label class=\"cqa-text-[12px] cqa-leading-[15px] cqa-text-[#0F172A]\" [for]=\"getInputId(item.id)\">{{ item.question }}</label>\n </div>\n <cqa-custom-input\n [inputId]=\"getInputId(item.id)\"\n [value]=\"item.answer\"\n [placeholder]=\"inputPlaceholder\"\n [fullWidth]=\"true\"\n (valueChange)=\"onAnswerInput(item.id, $event)\"\n class=\"cqa-w-full\"\n ></cqa-custom-input>\n </div>\n </ng-container>\n <ng-template #emptyState>\n <div class=\"cqa-py-6 cqa-text-[13px] cqa-text-[#6B7280] cqa-text-center\">No questions available.</div>\n </ng-template>\n </div>\n\n <!-- Footer: aligned with table pagination design -->\n <div class=\"cqa-mt-3 table-footer-pagination cqa-text-[#717182] cqa-text-[12px] cqa-leading-[15px] cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-flex-wrap cqa-px-[21px] cqa-py-1.5\" aria-live=\"polite\" style=\"border-top: 1px solid #E5E7EB;\">\n <div class=\"pagination-info cqa-flex cqa-items-center cqa-gap-[7px] cqa-relative\">\n <span class=\"rows-label\">Questions per page</span>\n <div class=\"cqa-relative\" #pageSizeDropdownContainer>\n <button\n type=\"button\"\n class=\"cqa-inline-flex cqa-items-center cqa-gap-[24px] cqa-text-[#0A0A0A] cqa-px-[11.5px] cqa-py-[6.75px]\"\n (click)=\"togglePageSizeMenu()\"\n [attr.aria-expanded]=\"pageSizeOpen\"\n aria-haspopup=\"listbox\"\n aria-label=\"Questions per page\"\n >\n {{ pageSize }}\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><g opacity=\"0.5\"><path d=\"M3.5 5.25L7 8.75L10.5 5.25\" stroke=\"#717182\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></g></svg>\n </button>\n <div\n *ngIf=\"pageSizeOpen\"\n class=\"cqa-absolute cqa-z-[100] cqa-bottom-[calc(100%+8px)] cqa-left-0 cqa-w-[75px] cqa-max-h-[170px] cqa-overflow-auto cqa-rounded-lg cqa-border cqa-border-[#E5E7EB] cqa-bg-white cqa-shadow-[0px_4px_6px_-1px_rgba(0,0,0,0.1)] cqa-p-[5px]\"\n role=\"listbox\"\n [attr.aria-activedescendant]=\"'question-pagesize-' + pageSize\"\n >\n <button\n *ngFor=\"let opt of pageSizeOptions\"\n type=\"button\"\n class=\"cqa-w-full cqa-px-2 cqa-py-[6px] hover:cqa-bg-[#F7F8FA] cqa-text-left cqa-rounded-md cqa-text-black-100\"\n [attr.id]=\"'question-pagesize-' + opt\"\n role=\"option\"\n [attr.aria-selected]=\"pageSize === opt\"\n (click)=\"selectPageSize(opt)\"\n >\n {{ opt }}\n </button>\n </div>\n </div>\n </div>\n\n <div class=\"pagination-right cqa-flex cqa-items-center cqa-gap-[21px]\">\n <div class=\"pagination-range\">\n {{ totalItems ? startIndex + 1 : 0 }}–{{ endIndex }} of {{ totalItems }}\n </div>\n <div class=\"pagination-controls cqa-flex cqa-items-stretch cqa-gap-[3.5px]\">\n <button class=\"pagination-btn cqa-w-[28px] cqa-h-[28px] cqa-min-w-[28px] cqa-rounded-[5px] cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-bg-[#F7F8FA] cqa-flex cqa-items-center cqa-justify-center disabled:cqa-opacity-50 disabled:cqa-cursor-not-allowed\" [disabled]=\"isPrevDisabled\" (click)=\"prevPage()\" aria-label=\"Previous page\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M8.75 10.5L5.25 7L8.75 3.5\" stroke=\"#0A0A0A\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </button>\n <button class=\"pagination-btn cqa-w-[28px] cqa-h-[28px] cqa-min-w-[28px] cqa-rounded-[5px] cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-bg-[#F7F8FA] cqa-flex cqa-items-center cqa-justify-center disabled:cqa-opacity-50 disabled:cqa-cursor-not-allowed\" [disabled]=\"isNextDisabled\" (click)=\"nextPage()\" aria-label=\"Next page\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M5.25 10.5L8.75 7L5.25 3.5\" stroke=\"#0A0A0A\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </button>\n </div>\n </div>\n </div>\n </div>\n\n</div>\n", styles: [] }]
|
package/esm2020/lib/step-builder/advanced-variables-form/advanced-variables-form.component.mjs
CHANGED
|
@@ -150,7 +150,7 @@ export class AdvancedVariablesFormComponent {
|
|
|
150
150
|
}
|
|
151
151
|
}
|
|
152
152
|
AdvancedVariablesFormComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AdvancedVariablesFormComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
153
|
-
AdvancedVariablesFormComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AdvancedVariablesFormComponent, selector: "cqa-advanced-variables-form", inputs: { advancedVariables: "advancedVariables", advancedVariableForm: "advancedVariableForm" }, outputs: { variableBooleanChange: "variableBooleanChange", variableValueChange: "variableValueChange" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-flex cqa-gap-x-6 cqa-gap-y-4 cqa-flex-wrap advanced-variables-form cqa-mb-4\">\n <ng-container *ngFor=\"let variable of advancedVariables; let i = index; trackBy: trackByVariable\">\n <!-- Boolean variables with mat-slide-toggle -->\n <ng-container *ngIf=\"isBooleanType(variable)\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\" style=\"width: 100%\">\n <mat-slide-toggle \n [checked]=\"getBooleanValue(variable, i)\"\n (change)=\"onVariableBooleanChange(variable.name, $event.checked)\" \n color=\"primary\">\n </mat-slide-toggle>\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 capitalize-first\">\n {{ variable.label }}\n </label>\n </div>\n </ng-container>\n\n <!-- str_list variables with dynamic list -->\n <ng-container *ngIf=\"isStrListType(variable)\">\n <div class=\"cqa-flex cqa-flex-col cqa-w-full\">\n <ng-container *ngIf=\"getStrListFormArray(variable, i) as formArray\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-mb-1\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 capitalize-first\">\n {{ variable.label }}\n </label>\n <!-- Empty-state add button: no rows exist yet, so the usual per-row add icon isn't rendered.\n Placing the add button next to the label is the only way to get the first entry back. -->\n <div\n *ngIf=\"formArray.length === 0\"\n class=\"cqa-cursor-pointer cqa-text-blue-600 cqa-flex cqa-items-center cqa-justify-center\"\n (click)=\"addStrListItem(variable, i)\">\n <mat-icon style=\"display: flex; align-items: center; justify-content: center; font-size: 20px;\">add</mat-icon>\n </div>\n </div>\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <div *ngFor=\"let control of formArray.controls; let itemIndex = index; trackBy: trackByControl\" class=\"cqa-flex cqa-gap-2 cqa-items-center\">\n <cqa-custom-input\n [placeholder]=\"'Enter locator'\"\n [value]=\"control.value\"\n [fullWidth]=\"true\"\n (valueChange)=\"onStrListItemChange(variable, i, itemIndex, $event)\">\n </cqa-custom-input>\n <!-- Delete icon is always available so the user can remove ANY row, including the last one. -->\n <div class=\"cqa-cursor-pointer cqa-text-red-600 cqa-flex cqa-items-center cqa-justify-center\" (click)=\"removeStrListItem(variable, i, itemIndex)\">\n <mat-icon style=\"font-size: 24px;\">delete</mat-icon>\n </div>\n <div *ngIf=\"itemIndex === formArray.length - 1\" class=\"cqa-cursor-pointer cqa-text-blue-600 cqa-flex cqa-items-center cqa-justify-center\" (click)=\"addStrListItem(variable, i)\">\n <mat-icon style=\"font-size: 24px;\">add</mat-icon>\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n </ng-container>\n </ng-container>\n</div>\n\n", components: [{ type: i1.MatSlideToggle, selector: "mat-slide-toggle", inputs: ["disabled", "disableRipple", "color", "tabIndex", "name", "id", "labelPosition", "aria-label", "aria-labelledby", "aria-describedby", "required", "checked"], outputs: ["change", "toggleChange"], exportAs: ["matSlideToggle"] }, { type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i3.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"] }], directives: [{ type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
153
|
+
AdvancedVariablesFormComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AdvancedVariablesFormComponent, selector: "cqa-advanced-variables-form", inputs: { advancedVariables: "advancedVariables", advancedVariableForm: "advancedVariableForm" }, outputs: { variableBooleanChange: "variableBooleanChange", variableValueChange: "variableValueChange" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-flex cqa-gap-x-6 cqa-gap-y-4 cqa-flex-wrap advanced-variables-form cqa-mb-4\">\n <ng-container *ngFor=\"let variable of advancedVariables; let i = index; trackBy: trackByVariable\">\n <!-- Boolean variables with mat-slide-toggle -->\n <ng-container *ngIf=\"isBooleanType(variable)\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\" style=\"width: 100%\">\n <mat-slide-toggle \n [checked]=\"getBooleanValue(variable, i)\"\n (change)=\"onVariableBooleanChange(variable.name, $event.checked)\" \n color=\"primary\">\n </mat-slide-toggle>\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 capitalize-first\">\n {{ variable.label }}\n </label>\n </div>\n </ng-container>\n\n <!-- str_list variables with dynamic list -->\n <ng-container *ngIf=\"isStrListType(variable)\">\n <div class=\"cqa-flex cqa-flex-col cqa-w-full\">\n <ng-container *ngIf=\"getStrListFormArray(variable, i) as formArray\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-mb-1\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 capitalize-first\">\n {{ variable.label }}\n </label>\n <!-- Empty-state add button: no rows exist yet, so the usual per-row add icon isn't rendered.\n Placing the add button next to the label is the only way to get the first entry back. -->\n <div\n *ngIf=\"formArray.length === 0\"\n class=\"cqa-cursor-pointer cqa-text-blue-600 cqa-flex cqa-items-center cqa-justify-center\"\n (click)=\"addStrListItem(variable, i)\">\n <mat-icon style=\"display: flex; align-items: center; justify-content: center; font-size: 20px;\">add</mat-icon>\n </div>\n </div>\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <div *ngFor=\"let control of formArray.controls; let itemIndex = index; trackBy: trackByControl\" class=\"cqa-flex cqa-gap-2 cqa-items-center\">\n <cqa-custom-input\n [placeholder]=\"'Enter locator'\"\n [value]=\"control.value\"\n [fullWidth]=\"true\"\n (valueChange)=\"onStrListItemChange(variable, i, itemIndex, $event)\">\n </cqa-custom-input>\n <!-- Delete icon is always available so the user can remove ANY row, including the last one. -->\n <div class=\"cqa-cursor-pointer cqa-text-red-600 cqa-flex cqa-items-center cqa-justify-center\" (click)=\"removeStrListItem(variable, i, itemIndex)\">\n <mat-icon style=\"font-size: 24px;\">delete</mat-icon>\n </div>\n <div *ngIf=\"itemIndex === formArray.length - 1\" class=\"cqa-cursor-pointer cqa-text-blue-600 cqa-flex cqa-items-center cqa-justify-center\" (click)=\"addStrListItem(variable, i)\">\n <mat-icon style=\"font-size: 24px;\">add</mat-icon>\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n </ng-container>\n </ng-container>\n</div>\n\n", components: [{ type: i1.MatSlideToggle, selector: "mat-slide-toggle", inputs: ["disabled", "disableRipple", "color", "tabIndex", "name", "id", "labelPosition", "aria-label", "aria-labelledby", "aria-describedby", "required", "checked"], outputs: ["change", "toggleChange"], exportAs: ["matSlideToggle"] }, { type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i3.CustomInputComponent, selector: "cqa-custom-input", inputs: ["inputId", "label", "type", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "inputInlineStyle", "labelInlineStyle", "showPasswordToggle"], outputs: ["valueChange", "blurred", "focused", "enterPressed"] }], directives: [{ type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
154
154
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AdvancedVariablesFormComponent, decorators: [{
|
|
155
155
|
type: Component,
|
|
156
156
|
args: [{ selector: 'cqa-advanced-variables-form', host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"cqa-flex cqa-gap-x-6 cqa-gap-y-4 cqa-flex-wrap advanced-variables-form cqa-mb-4\">\n <ng-container *ngFor=\"let variable of advancedVariables; let i = index; trackBy: trackByVariable\">\n <!-- Boolean variables with mat-slide-toggle -->\n <ng-container *ngIf=\"isBooleanType(variable)\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\" style=\"width: 100%\">\n <mat-slide-toggle \n [checked]=\"getBooleanValue(variable, i)\"\n (change)=\"onVariableBooleanChange(variable.name, $event.checked)\" \n color=\"primary\">\n </mat-slide-toggle>\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 capitalize-first\">\n {{ variable.label }}\n </label>\n </div>\n </ng-container>\n\n <!-- str_list variables with dynamic list -->\n <ng-container *ngIf=\"isStrListType(variable)\">\n <div class=\"cqa-flex cqa-flex-col cqa-w-full\">\n <ng-container *ngIf=\"getStrListFormArray(variable, i) as formArray\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-mb-1\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 capitalize-first\">\n {{ variable.label }}\n </label>\n <!-- Empty-state add button: no rows exist yet, so the usual per-row add icon isn't rendered.\n Placing the add button next to the label is the only way to get the first entry back. -->\n <div\n *ngIf=\"formArray.length === 0\"\n class=\"cqa-cursor-pointer cqa-text-blue-600 cqa-flex cqa-items-center cqa-justify-center\"\n (click)=\"addStrListItem(variable, i)\">\n <mat-icon style=\"display: flex; align-items: center; justify-content: center; font-size: 20px;\">add</mat-icon>\n </div>\n </div>\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <div *ngFor=\"let control of formArray.controls; let itemIndex = index; trackBy: trackByControl\" class=\"cqa-flex cqa-gap-2 cqa-items-center\">\n <cqa-custom-input\n [placeholder]=\"'Enter locator'\"\n [value]=\"control.value\"\n [fullWidth]=\"true\"\n (valueChange)=\"onStrListItemChange(variable, i, itemIndex, $event)\">\n </cqa-custom-input>\n <!-- Delete icon is always available so the user can remove ANY row, including the last one. -->\n <div class=\"cqa-cursor-pointer cqa-text-red-600 cqa-flex cqa-items-center cqa-justify-center\" (click)=\"removeStrListItem(variable, i, itemIndex)\">\n <mat-icon style=\"font-size: 24px;\">delete</mat-icon>\n </div>\n <div *ngIf=\"itemIndex === formArray.length - 1\" class=\"cqa-cursor-pointer cqa-text-blue-600 cqa-flex cqa-items-center cqa-justify-center\" (click)=\"addStrListItem(variable, i)\">\n <mat-icon style=\"font-size: 24px;\">add</mat-icon>\n </div>\n </div>\n </div>\n </ng-container>\n </div>\n </ng-container>\n </ng-container>\n</div>\n\n", styles: [] }]
|