@cqa-lib/cqa-ui 1.1.188 → 1.1.190
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/add-prerequisite-cases-section/add-prerequisite-cases-section.component.mjs +1 -1
- package/esm2020/lib/button/button.component.mjs +3 -3
- package/esm2020/lib/compare-runs/compare-runs.component.mjs +1 -1
- package/esm2020/lib/custom-textarea/custom-textarea.component.mjs +3 -6
- package/esm2020/lib/dashboards/dashboard-header/dashboard-header.component.mjs +1 -1
- package/esm2020/lib/dynamic-select/dynamic-select-field.component.mjs +82 -3
- package/esm2020/lib/filters/dynamic-filter/dynamic-filter.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-action/step-builder-action.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-api/step-builder-api.component.mjs +3 -2
- package/esm2020/lib/step-builder/step-builder-condition/step-builder-condition.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-loop/step-builder-loop.component.mjs +1 -1
- package/esm2020/lib/test-case-details/api-edit-step/api-edit-step.component.mjs +1122 -0
- package/esm2020/lib/test-case-details/api-step/api-step.component.mjs +48 -9
- package/esm2020/lib/test-case-details/condition-step/condition-step.component.mjs +1 -1
- package/esm2020/lib/test-case-details/custom-edit-step/custom-edit-step.component.mjs +1 -1
- package/esm2020/lib/test-case-details/element-popup/element-popup.component.mjs +1 -1
- package/esm2020/lib/test-case-details/loop-step/loop-step.component.mjs +1 -1
- package/esm2020/lib/test-case-details/step-details-drawer/step-details-drawer-data.mjs +3 -0
- package/esm2020/lib/test-case-details/step-details-drawer/step-details-drawer-field.config.mjs +188 -0
- package/esm2020/lib/test-case-details/step-details-drawer/step-details-drawer-ref.mjs +28 -0
- package/esm2020/lib/test-case-details/step-details-drawer/step-details-drawer.component.mjs +291 -0
- package/esm2020/lib/test-case-details/step-details-drawer/step-details-drawer.service.mjs +67 -0
- package/esm2020/lib/ui-kit.module.mjs +13 -3
- package/esm2020/lib/utils/tw-overlay-container.mjs +6 -3
- package/esm2020/public-api.mjs +7 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +2066 -227
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +1898 -112
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/custom-textarea/custom-textarea.component.d.ts +1 -2
- package/lib/dynamic-select/dynamic-select-field.component.d.ts +19 -1
- package/lib/test-case-details/api-edit-step/api-edit-step.component.d.ts +346 -0
- package/lib/test-case-details/api-step/api-step.component.d.ts +18 -2
- package/lib/test-case-details/step-details-drawer/step-details-drawer-data.d.ts +7 -0
- package/lib/test-case-details/step-details-drawer/step-details-drawer-field.config.d.ts +51 -0
- package/lib/test-case-details/step-details-drawer/step-details-drawer-ref.d.ts +13 -0
- package/lib/test-case-details/step-details-drawer/step-details-drawer.component.d.ts +60 -0
- package/lib/test-case-details/step-details-drawer/step-details-drawer.service.d.ts +22 -0
- package/lib/ui-kit.module.d.ts +35 -33
- package/package.json +1 -1
- package/public-api.d.ts +6 -0
- package/src/lib/assets/images/delete.svg +3 -0
- package/styles.css +1 -1
|
@@ -0,0 +1,291 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Inject, Input, Optional, Output, } from '@angular/core';
|
|
2
|
+
import { getStepDetailsStepType, STEP_DETAILS_FIELDS_BY_TYPE, STEP_DETAILS_FIELD_META, ADVANCED_TOGGLE_KEYS, ADVANCED_SUBFIELDS_BY_TYPE, } from './step-details-drawer-field.config';
|
|
3
|
+
import { STEP_DETAILS_DRAWER_REF } from './step-details-drawer-ref';
|
|
4
|
+
import { STEP_DETAILS_DRAWER_DATA } from './step-details-drawer-data';
|
|
5
|
+
import { isNormalStepConfig, isApiStepConfig, isAiAgentStepConfig, } from '../test-case-step.models';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/forms";
|
|
8
|
+
import * as i2 from "@angular/material/icon";
|
|
9
|
+
import * as i3 from "../../custom-toggle/custom-toggle.component";
|
|
10
|
+
import * as i4 from "../../dynamic-select/dynamic-select-field.component";
|
|
11
|
+
import * as i5 from "../../custom-input/custom-input.component";
|
|
12
|
+
import * as i6 from "../../custom-textarea/custom-textarea.component";
|
|
13
|
+
import * as i7 from "../../button/button.component";
|
|
14
|
+
import * as i8 from "@angular/common";
|
|
15
|
+
import * as i9 from "./step-details-drawer-ref";
|
|
16
|
+
/**
|
|
17
|
+
* Step Details Drawer (Edit In Depth).
|
|
18
|
+
* Single reusable component; form is dynamic by Step Type via configuration.
|
|
19
|
+
* No Priority section. Uses CQA components only.
|
|
20
|
+
*/
|
|
21
|
+
export class StepDetailsDrawerComponent {
|
|
22
|
+
constructor(ref, fb, cdr, data) {
|
|
23
|
+
this.ref = ref;
|
|
24
|
+
this.fb = fb;
|
|
25
|
+
this.cdr = cdr;
|
|
26
|
+
this.saveChanges = new EventEmitter();
|
|
27
|
+
this.cancel = new EventEmitter();
|
|
28
|
+
this.saveAsTemplate = new EventEmitter();
|
|
29
|
+
/** Optional: when provided (e.g. from Storybook Controls), use instead of STEP_DETAILS_DRAWER_DATA. */
|
|
30
|
+
this.stepData = null;
|
|
31
|
+
this.stepNumberInput = null;
|
|
32
|
+
this.stepType = null;
|
|
33
|
+
this.visibleFields = [];
|
|
34
|
+
this.fieldMeta = STEP_DETAILS_FIELD_META;
|
|
35
|
+
this.advancedToggleKeys = ADVANCED_TOGGLE_KEYS;
|
|
36
|
+
this.advancedExpanded = false;
|
|
37
|
+
this.agentTaskExpanded = true;
|
|
38
|
+
/** Cached select configs (stable references per key) to avoid infinite change detection in template. */
|
|
39
|
+
this.selectConfigMap = this.buildSelectConfigMap();
|
|
40
|
+
this.step = data?.step ?? {};
|
|
41
|
+
this.stepNumber = data?.stepNumber ?? 1;
|
|
42
|
+
this.form = this.fb.group({});
|
|
43
|
+
this.syncStepTypeAndForm();
|
|
44
|
+
}
|
|
45
|
+
/** Subfields to show inside Advanced section for current step type. */
|
|
46
|
+
get advancedSubfields() {
|
|
47
|
+
return this.stepType ? ADVANCED_SUBFIELDS_BY_TYPE[this.stepType] : [];
|
|
48
|
+
}
|
|
49
|
+
ngOnInit() {
|
|
50
|
+
if (this.stepData != null) {
|
|
51
|
+
this.step = this.stepData;
|
|
52
|
+
this.stepNumber = this.stepNumberInput ?? this.stepData.stepNumber ?? 1;
|
|
53
|
+
this.syncStepTypeAndForm();
|
|
54
|
+
this.cdr.markForCheck();
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
ngOnChanges(changes) {
|
|
58
|
+
if (changes['stepData'] || changes['stepNumberInput']) {
|
|
59
|
+
if (this.stepData != null) {
|
|
60
|
+
this.step = this.stepData;
|
|
61
|
+
this.stepNumber = this.stepNumberInput ?? this.stepData.stepNumber ?? 1;
|
|
62
|
+
}
|
|
63
|
+
this.syncStepTypeAndForm();
|
|
64
|
+
this.cdr.detectChanges();
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
syncStepTypeAndForm() {
|
|
68
|
+
this.stepType = getStepDetailsStepType(this.step);
|
|
69
|
+
this.visibleFields = this.stepType ? STEP_DETAILS_FIELDS_BY_TYPE[this.stepType] : [];
|
|
70
|
+
this.buildFormFromStep();
|
|
71
|
+
}
|
|
72
|
+
buildFormFromStep() {
|
|
73
|
+
const group = {};
|
|
74
|
+
for (const key of this.visibleFields) {
|
|
75
|
+
if (key === 'advanced')
|
|
76
|
+
continue;
|
|
77
|
+
group[key] = [this.getStepValue(key)];
|
|
78
|
+
}
|
|
79
|
+
for (const key of ADVANCED_TOGGLE_KEYS) {
|
|
80
|
+
if (!(key in group))
|
|
81
|
+
group[key] = [this.getStepValue(key)];
|
|
82
|
+
}
|
|
83
|
+
this.form = this.fb.group(group);
|
|
84
|
+
}
|
|
85
|
+
getStepValue(key) {
|
|
86
|
+
const step = this.step;
|
|
87
|
+
if (isNormalStepConfig(this.step) && this.step.eventType === 'custom') {
|
|
88
|
+
if (key === 'description') {
|
|
89
|
+
const params = this.step.parameters;
|
|
90
|
+
return params?.[0]?.value ?? this.step.description ?? '';
|
|
91
|
+
}
|
|
92
|
+
if (key === 'metadata') {
|
|
93
|
+
const params = this.step.parameters;
|
|
94
|
+
return params?.[1]?.value ?? '';
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
if (isApiStepConfig(this.step)) {
|
|
98
|
+
const api = this.step;
|
|
99
|
+
if (key === 'method')
|
|
100
|
+
return api.method ?? 'GET';
|
|
101
|
+
if (key === 'url')
|
|
102
|
+
return api.baseUrl ? `${api.baseUrl}${api.endpoint || ''}` : api.endpoint ?? '';
|
|
103
|
+
if (key === 'headers')
|
|
104
|
+
return api.headers ? JSON.stringify(api.headers, null, 2) : '';
|
|
105
|
+
if (key === 'body')
|
|
106
|
+
return api.requestBody ?? '';
|
|
107
|
+
if (key === 'saveOutputAsVariable')
|
|
108
|
+
return api.saveTo ?? '';
|
|
109
|
+
}
|
|
110
|
+
if (isAiAgentStepConfig(this.step)) {
|
|
111
|
+
const ai = this.step;
|
|
112
|
+
if (key === 'agentTask')
|
|
113
|
+
return ai.instructions ?? '';
|
|
114
|
+
if (key === 'description')
|
|
115
|
+
return ai.description ?? '';
|
|
116
|
+
if (key === 'metadata')
|
|
117
|
+
return step.metadata ?? '';
|
|
118
|
+
}
|
|
119
|
+
if (key === 'onlyUseAttachedContext' ||
|
|
120
|
+
key === 'takeScreenshotsWhenVerifying' ||
|
|
121
|
+
key === 'continueOnError' ||
|
|
122
|
+
key === 'disabled') {
|
|
123
|
+
return step[key] === true;
|
|
124
|
+
}
|
|
125
|
+
return step[key] ?? '';
|
|
126
|
+
}
|
|
127
|
+
setStepValue(key, value) {
|
|
128
|
+
const step = this.step;
|
|
129
|
+
if (isNormalStepConfig(this.step) && this.step.eventType === 'custom') {
|
|
130
|
+
if (key === 'description') {
|
|
131
|
+
const params = this.step.parameters;
|
|
132
|
+
if (params?.[0])
|
|
133
|
+
params[0].value = String(value);
|
|
134
|
+
else
|
|
135
|
+
this.step.description = String(value);
|
|
136
|
+
return;
|
|
137
|
+
}
|
|
138
|
+
if (key === 'metadata') {
|
|
139
|
+
const params = this.step.parameters;
|
|
140
|
+
if (params?.[1])
|
|
141
|
+
params[1].value = String(value);
|
|
142
|
+
return;
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
if (isApiStepConfig(this.step)) {
|
|
146
|
+
const api = this.step;
|
|
147
|
+
if (key === 'method') {
|
|
148
|
+
api.method = value;
|
|
149
|
+
return;
|
|
150
|
+
}
|
|
151
|
+
if (key === 'url') {
|
|
152
|
+
api.endpoint = String(value);
|
|
153
|
+
return;
|
|
154
|
+
}
|
|
155
|
+
if (key === 'headers') {
|
|
156
|
+
try {
|
|
157
|
+
api.headers = JSON.parse(String(value) || '{}');
|
|
158
|
+
}
|
|
159
|
+
catch {
|
|
160
|
+
api.headers = {};
|
|
161
|
+
}
|
|
162
|
+
return;
|
|
163
|
+
}
|
|
164
|
+
if (key === 'body') {
|
|
165
|
+
api.requestBody = String(value);
|
|
166
|
+
return;
|
|
167
|
+
}
|
|
168
|
+
if (key === 'saveOutputAsVariable') {
|
|
169
|
+
api.saveTo = String(value);
|
|
170
|
+
return;
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
if (isAiAgentStepConfig(this.step)) {
|
|
174
|
+
const ai = this.step;
|
|
175
|
+
if (key === 'agentTask') {
|
|
176
|
+
ai.instructions = String(value);
|
|
177
|
+
return;
|
|
178
|
+
}
|
|
179
|
+
if (key === 'description') {
|
|
180
|
+
ai.description = String(value);
|
|
181
|
+
return;
|
|
182
|
+
}
|
|
183
|
+
if (key === 'metadata') {
|
|
184
|
+
step.metadata = value;
|
|
185
|
+
return;
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
step[key] = value;
|
|
189
|
+
}
|
|
190
|
+
onBack() {
|
|
191
|
+
this.cancel.emit();
|
|
192
|
+
this.ref.close(undefined);
|
|
193
|
+
}
|
|
194
|
+
onClose() {
|
|
195
|
+
this.cancel.emit();
|
|
196
|
+
this.ref.close(undefined);
|
|
197
|
+
}
|
|
198
|
+
onCancel() {
|
|
199
|
+
this.cancel.emit();
|
|
200
|
+
this.ref.close(undefined);
|
|
201
|
+
}
|
|
202
|
+
onSaveAsTemplate() {
|
|
203
|
+
this.applyFormToStep();
|
|
204
|
+
this.saveAsTemplate.emit(this.step);
|
|
205
|
+
this.ref.close({ action: 'saveAsTemplate', step: this.step });
|
|
206
|
+
}
|
|
207
|
+
onSaveChanges() {
|
|
208
|
+
this.applyFormToStep();
|
|
209
|
+
this.saveChanges.emit(this.step);
|
|
210
|
+
this.ref.close({ action: 'saveChanges', step: this.step });
|
|
211
|
+
}
|
|
212
|
+
applyFormToStep() {
|
|
213
|
+
for (const key of this.visibleFields) {
|
|
214
|
+
if (key === 'advanced')
|
|
215
|
+
continue;
|
|
216
|
+
if (this.form.contains(key))
|
|
217
|
+
this.setStepValue(key, this.form.get(key)?.value);
|
|
218
|
+
}
|
|
219
|
+
for (const key of ADVANCED_TOGGLE_KEYS) {
|
|
220
|
+
if (this.form.contains(key))
|
|
221
|
+
this.setStepValue(key, this.form.get(key)?.value);
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
/** Build cached map of select configs from field meta (stable refs to prevent CD loops). */
|
|
225
|
+
buildSelectConfigMap() {
|
|
226
|
+
const map = {};
|
|
227
|
+
const keys = Object.keys(STEP_DETAILS_FIELD_META);
|
|
228
|
+
for (const key of keys) {
|
|
229
|
+
const meta = STEP_DETAILS_FIELD_META[key];
|
|
230
|
+
if (!meta || meta.controlType !== 'dropdown' || !meta.options?.length)
|
|
231
|
+
continue;
|
|
232
|
+
const options = meta.options.map((o) => ({ value: o.value, label: o.label }));
|
|
233
|
+
map[key] = {
|
|
234
|
+
key: meta.key,
|
|
235
|
+
placeholder: meta.placeholder ?? meta.label,
|
|
236
|
+
options,
|
|
237
|
+
};
|
|
238
|
+
}
|
|
239
|
+
return map;
|
|
240
|
+
}
|
|
241
|
+
/** Return cached select config for key (for template use; same reference every time). */
|
|
242
|
+
getSelectConfig(key) {
|
|
243
|
+
return this.selectConfigMap[key] ?? null;
|
|
244
|
+
}
|
|
245
|
+
/** Whether this key is the first in its section (for showing section headings). */
|
|
246
|
+
isFirstInSection(key, section) {
|
|
247
|
+
const meta = STEP_DETAILS_FIELD_META[key];
|
|
248
|
+
if (!meta || meta.section !== section)
|
|
249
|
+
return false;
|
|
250
|
+
const idx = this.visibleFields.indexOf(key);
|
|
251
|
+
if (idx <= 0)
|
|
252
|
+
return true;
|
|
253
|
+
const prevMeta = STEP_DETAILS_FIELD_META[this.visibleFields[idx - 1]];
|
|
254
|
+
return !prevMeta || prevMeta.section !== section;
|
|
255
|
+
}
|
|
256
|
+
/** Dynamic drawer title by step type (e.g. "AI Agent Step Details"). */
|
|
257
|
+
get drawerTitle() {
|
|
258
|
+
if (this.stepType === 'custom')
|
|
259
|
+
return 'Custom Step Details';
|
|
260
|
+
if (this.stepType === 'api')
|
|
261
|
+
return 'API Step Details';
|
|
262
|
+
if (this.stepType === 'aiAgent')
|
|
263
|
+
return 'AI Agent Step Details';
|
|
264
|
+
return 'Step Details';
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
StepDetailsDrawerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepDetailsDrawerComponent, deps: [{ token: STEP_DETAILS_DRAWER_REF }, { token: i1.FormBuilder }, { token: i0.ChangeDetectorRef }, { token: STEP_DETAILS_DRAWER_DATA, optional: true }], target: i0.ɵɵFactoryTarget.Component });
|
|
268
|
+
StepDetailsDrawerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: StepDetailsDrawerComponent, selector: "cqa-step-details-drawer", inputs: { stepData: "stepData", stepNumberInput: "stepNumberInput" }, outputs: { saveChanges: "saveChanges", cancel: "cancel", saveAsTemplate: "saveAsTemplate" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: "<!-- Step Details Drawer \u2013 Edit In Depth. No Priority. Fully config-driven. -->\n<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-w-full cqa-max-w-[480px] cqa-box-border\">\n <!-- Header: back chevron, dynamic title, Step N pill, close -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-3 cqa-px-4 cqa-py-4 cqa-bg-white cqa-rounded-t-[12px] cqa-border cqa-border-[#E5E5E5]\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <button type=\"button\" (click)=\"onBack()\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-8 cqa-h-8 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0\"\n aria-label=\"Back\">\n <mat-icon class=\"!cqa-w-5 !cqa-h-5 !cqa-text-[20px]\">chevron_left</mat-icon>\n </button>\n <h2 class=\"cqa-text-[16px] cqa-leading-[24px] cqa-font-semibold cqa-text-[#111827] cqa-m-0 cqa-flex-1\">\n {{ drawerTitle }}\n </h2>\n <span class=\"cqa-px-2.5 cqa-py-1 cqa-rounded-lg cqa-bg-[#EDE9FE] cqa-text-[#7C3AED] cqa-text-[12px] cqa-font-medium\">\n Step {{ stepNumber }}\n </span>\n </div>\n <button type=\"button\" (click)=\"onClose()\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-[28px] cqa-min-w-[28px] cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0\"\n title=\"Close\" aria-label=\"Close\">\n <mat-icon class=\"!cqa-w-5 !cqa-h-5 !cqa-text-[20px]\">close</mat-icon>\n </button>\n </div>\n <div class=\"cqa-border cqa-border-solid cqa-border-[#E5E5E5]\"> </div>\n\n <!-- Content: Attributes + dynamic fields from config -->\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-bg-white cqa-px-4 cqa-py-4\">\n <ng-container *ngIf=\"stepType\">\n <p class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#6B7280] cqa-mb-3 cqa-mt-0\">Attributes</p>\n\n <div class=\"cqa-flex cqa-flex-col cqa-gap-4\">\n <ng-container *ngFor=\"let key of visibleFields\">\n <!-- Section heading: Constraints (first constraints field only) -->\n <ng-container *ngIf=\"key !== 'advanced' && isFirstInSection(key, 'constraints')\">\n <p class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#6B7280] cqa-mb-2 cqa-mt-0\">Constraints</p>\n </ng-container>\n\n <!-- Advanced: expandable section with config-driven toggles + subfields -->\n <ng-container *ngIf=\"key === 'advanced'\">\n <button type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-between cqa-w-full cqa-text-left cqa-text-[12px] cqa-font-medium cqa-text-[#111827] cqa-py-2 cqa-border-b cqa-border-[#E5E7EB]\"\n (click)=\"advancedExpanded = !advancedExpanded\">\n <span>Advanced</span>\n <mat-icon class=\"cqa-text-[20px] cqa-transition-transform\" [class.cqa-rotate-180]=\"!advancedExpanded\">\n expand_less\n </mat-icon>\n </button>\n <div *ngIf=\"advancedExpanded\" class=\"cqa-flex cqa-flex-col cqa-gap-3 cqa-pt-2\">\n <!-- Advanced toggles from config -->\n <ng-container *ngFor=\"let toggleKey of advancedToggleKeys\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\" *ngIf=\"fieldMeta[toggleKey]\">\n <cqa-custom-toggle\n [checked]=\"form.get(toggleKey)?.value\"\n (checkedChange)=\"form.get(toggleKey)?.setValue($event)\">\n </cqa-custom-toggle>\n <div class=\"cqa-flex cqa-flex-col\">\n <span class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#737373]\">{{ fieldMeta[toggleKey].label }}</span>\n <span class=\"cqa-text-[12px] cqa-text-[#0A0A0A]\" *ngIf=\"fieldMeta[toggleKey].subLabel\">{{ fieldMeta[toggleKey].subLabel }}</span>\n </div>\n </div>\n </ng-container>\n <!-- Advanced subfields (e.g. Retry Count, Iframe locator, Other Locators for aiAgent) -->\n <ng-container *ngFor=\"let subKey of advancedSubfields\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-step-details-drawer-select\" *ngIf=\"fieldMeta[subKey]?.controlType === 'dropdown' && getSelectConfig(subKey) && form.contains(subKey)\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[subKey].label }}</label>\n <cqa-dynamic-select [form]=\"form\" [config]=\"getSelectConfig(subKey)!\">\n </cqa-dynamic-select>\n </div>\n </ng-container>\n </div>\n </ng-container>\n\n <!-- Skip: advanced key (already rendered) and fields that are rendered inside advanced for this step type -->\n <ng-container *ngIf=\"key !== 'advanced' && !(stepType === 'aiAgent' && advancedSubfields.includes(key))\">\n <!-- text -->\n <div *ngIf=\"fieldMeta[key]?.controlType === 'text'\" class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].label }}</label>\n <cqa-custom-input\n [placeholder]=\"fieldMeta[key].placeholder || ''\"\n [value]=\"form.get(key)?.value\"\n [fullWidth]=\"true\"\n (valueChange)=\"form.get(key)?.setValue($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- textarea (including agentTask with optional required + tip) -->\n <div *ngIf=\"fieldMeta[key]?.controlType === 'textarea'\" class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].label }}</label>\n <span *ngIf=\"fieldMeta[key].required\" class=\"cqa-px-2 cqa-py-0.5 cqa-rounded-full cqa-bg-[#FEE2E2] cqa-text-[#DC2626] cqa-text-[10px] cqa-font-medium\">Required</span>\n </div>\n <p *ngIf=\"key === 'agentTask'\" class=\"cqa-text-[12px] cqa-text-[#6B7280] cqa-m-0\">What should the agent achieve?</p>\n <cqa-custom-textarea\n [placeholder]=\"fieldMeta[key].placeholder || ''\"\n [value]=\"form.get(key)?.value\"\n [fullWidth]=\"true\"\n [rows]=\"fieldMeta[key].rows || 4\"\n customClass=\"cqa-p-2 cqa-text-[14px] cqa-leading-[20px]\"\n (valueChange)=\"form.get(key)?.setValue($event)\">\n </cqa-custom-textarea>\n <p *ngIf=\"fieldMeta[key].tip\" class=\"cqa-text-[12px] cqa-text-[#6B7280] cqa-m-0\">{{ fieldMeta[key].tip }}</p>\n </div>\n\n <!-- code -->\n <div *ngIf=\"fieldMeta[key]?.controlType === 'code'\" class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].label }}</label>\n <cqa-custom-textarea\n [placeholder]=\"fieldMeta[key].placeholder || ''\"\n [value]=\"form.get(key)?.value\"\n [fullWidth]=\"true\"\n [rows]=\"8\"\n resize=\"vertical\"\n customClass=\"cqa-p-2 cqa-text-[14px] cqa-leading-[20px]\"\n (valueChange)=\"form.get(key)?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n\n <!-- dropdown (main + constraints; advanced-only dropdowns rendered inside Advanced section) -->\n <div *ngIf=\"fieldMeta[key]?.controlType === 'dropdown' && form.contains(key) && getSelectConfig(key) && !advancedSubfields.includes(key)\"\n class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-step-details-drawer-select\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].label }}</label>\n <cqa-dynamic-select [form]=\"form\" [config]=\"getSelectConfig(key)!\">\n </cqa-dynamic-select>\n </div>\n\n <!-- toggle (e.g. onlyUseAttachedContext, takeScreenshotsWhenVerifying) -->\n <div *ngIf=\"fieldMeta[key]?.controlType === 'toggle'\" class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2\">\n <div>\n <p class=\"cqa-text-[12px] cqa-text-[#111827] cqa-m-0\">{{ fieldMeta[key].label }}</p>\n <span *ngIf=\"fieldMeta[key].subLabel\" class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].subLabel }}</span>\n </div>\n <cqa-custom-toggle\n [checked]=\"form.get(key)?.value\"\n (checkedChange)=\"form.get(key)?.setValue($event)\">\n </cqa-custom-toggle>\n </div>\n </ng-container>\n </ng-container>\n </div>\n </ng-container>\n\n <p *ngIf=\"!stepType\" class=\"cqa-text-[14px] cqa-text-[#6B7280]\">This step type is not supported in the drawer.</p>\n </div>\n<div class=\"cqa-border cqa-border-solid cqa-border-[#E5E5E5]\"> </div>\n <!-- Footer -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-start cqa-gap-2 cqa-px-4 cqa-py-4 cqa-border-t cqa-border-[#E5E7EB] cqa-bg-[#F5F5F54D]\">\n <cqa-button variant=\"text\" btnSize=\"lg\" [text]=\"'Cancel'\" (clicked)=\"onCancel()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-text-[#374151]'\"></cqa-button>\n <div class=\"cqa-flex-1 cqa-min-w-0\"></div>\n <cqa-button variant=\"outlined\" btnSize=\"lg\" [text]=\"'Save as Template'\" (clicked)=\"onSaveAsTemplate()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"></cqa-button>\n <cqa-button variant=\"filled\" btnSize=\"lg\" [text]=\"'Save Changes'\" (clicked)=\"onSaveChanges()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-bg-[#3F43EE] cqa-border-[#3F43EE]'\"></cqa-button>\n </div>\n</div>\n", components: [{ type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i3.CustomToggleComponent, selector: "cqa-custom-toggle", inputs: ["checked", "disabled", "ariaLabel"], outputs: ["checkedChange", "change"] }, { type: i4.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }, { type: i5.CustomInputComponent, selector: "cqa-custom-input", inputs: ["label", "type", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "inputInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused", "enterPressed"] }, { type: i6.CustomTextareaComponent, selector: "cqa-custom-textarea", inputs: ["label", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "rows", "cols", "resize", "textareaInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused"] }, { type: i7.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }], directives: [{ type: i8.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i8.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
269
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepDetailsDrawerComponent, decorators: [{
|
|
270
|
+
type: Component,
|
|
271
|
+
args: [{ selector: 'cqa-step-details-drawer', host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "<!-- Step Details Drawer \u2013 Edit In Depth. No Priority. Fully config-driven. -->\n<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-w-full cqa-max-w-[480px] cqa-box-border\">\n <!-- Header: back chevron, dynamic title, Step N pill, close -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-3 cqa-px-4 cqa-py-4 cqa-bg-white cqa-rounded-t-[12px] cqa-border cqa-border-[#E5E5E5]\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <button type=\"button\" (click)=\"onBack()\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-8 cqa-h-8 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0\"\n aria-label=\"Back\">\n <mat-icon class=\"!cqa-w-5 !cqa-h-5 !cqa-text-[20px]\">chevron_left</mat-icon>\n </button>\n <h2 class=\"cqa-text-[16px] cqa-leading-[24px] cqa-font-semibold cqa-text-[#111827] cqa-m-0 cqa-flex-1\">\n {{ drawerTitle }}\n </h2>\n <span class=\"cqa-px-2.5 cqa-py-1 cqa-rounded-lg cqa-bg-[#EDE9FE] cqa-text-[#7C3AED] cqa-text-[12px] cqa-font-medium\">\n Step {{ stepNumber }}\n </span>\n </div>\n <button type=\"button\" (click)=\"onClose()\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-[28px] cqa-min-w-[28px] cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0\"\n title=\"Close\" aria-label=\"Close\">\n <mat-icon class=\"!cqa-w-5 !cqa-h-5 !cqa-text-[20px]\">close</mat-icon>\n </button>\n </div>\n <div class=\"cqa-border cqa-border-solid cqa-border-[#E5E5E5]\"> </div>\n\n <!-- Content: Attributes + dynamic fields from config -->\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-bg-white cqa-px-4 cqa-py-4\">\n <ng-container *ngIf=\"stepType\">\n <p class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#6B7280] cqa-mb-3 cqa-mt-0\">Attributes</p>\n\n <div class=\"cqa-flex cqa-flex-col cqa-gap-4\">\n <ng-container *ngFor=\"let key of visibleFields\">\n <!-- Section heading: Constraints (first constraints field only) -->\n <ng-container *ngIf=\"key !== 'advanced' && isFirstInSection(key, 'constraints')\">\n <p class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#6B7280] cqa-mb-2 cqa-mt-0\">Constraints</p>\n </ng-container>\n\n <!-- Advanced: expandable section with config-driven toggles + subfields -->\n <ng-container *ngIf=\"key === 'advanced'\">\n <button type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-between cqa-w-full cqa-text-left cqa-text-[12px] cqa-font-medium cqa-text-[#111827] cqa-py-2 cqa-border-b cqa-border-[#E5E7EB]\"\n (click)=\"advancedExpanded = !advancedExpanded\">\n <span>Advanced</span>\n <mat-icon class=\"cqa-text-[20px] cqa-transition-transform\" [class.cqa-rotate-180]=\"!advancedExpanded\">\n expand_less\n </mat-icon>\n </button>\n <div *ngIf=\"advancedExpanded\" class=\"cqa-flex cqa-flex-col cqa-gap-3 cqa-pt-2\">\n <!-- Advanced toggles from config -->\n <ng-container *ngFor=\"let toggleKey of advancedToggleKeys\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\" *ngIf=\"fieldMeta[toggleKey]\">\n <cqa-custom-toggle\n [checked]=\"form.get(toggleKey)?.value\"\n (checkedChange)=\"form.get(toggleKey)?.setValue($event)\">\n </cqa-custom-toggle>\n <div class=\"cqa-flex cqa-flex-col\">\n <span class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#737373]\">{{ fieldMeta[toggleKey].label }}</span>\n <span class=\"cqa-text-[12px] cqa-text-[#0A0A0A]\" *ngIf=\"fieldMeta[toggleKey].subLabel\">{{ fieldMeta[toggleKey].subLabel }}</span>\n </div>\n </div>\n </ng-container>\n <!-- Advanced subfields (e.g. Retry Count, Iframe locator, Other Locators for aiAgent) -->\n <ng-container *ngFor=\"let subKey of advancedSubfields\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-step-details-drawer-select\" *ngIf=\"fieldMeta[subKey]?.controlType === 'dropdown' && getSelectConfig(subKey) && form.contains(subKey)\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[subKey].label }}</label>\n <cqa-dynamic-select [form]=\"form\" [config]=\"getSelectConfig(subKey)!\">\n </cqa-dynamic-select>\n </div>\n </ng-container>\n </div>\n </ng-container>\n\n <!-- Skip: advanced key (already rendered) and fields that are rendered inside advanced for this step type -->\n <ng-container *ngIf=\"key !== 'advanced' && !(stepType === 'aiAgent' && advancedSubfields.includes(key))\">\n <!-- text -->\n <div *ngIf=\"fieldMeta[key]?.controlType === 'text'\" class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].label }}</label>\n <cqa-custom-input\n [placeholder]=\"fieldMeta[key].placeholder || ''\"\n [value]=\"form.get(key)?.value\"\n [fullWidth]=\"true\"\n (valueChange)=\"form.get(key)?.setValue($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- textarea (including agentTask with optional required + tip) -->\n <div *ngIf=\"fieldMeta[key]?.controlType === 'textarea'\" class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].label }}</label>\n <span *ngIf=\"fieldMeta[key].required\" class=\"cqa-px-2 cqa-py-0.5 cqa-rounded-full cqa-bg-[#FEE2E2] cqa-text-[#DC2626] cqa-text-[10px] cqa-font-medium\">Required</span>\n </div>\n <p *ngIf=\"key === 'agentTask'\" class=\"cqa-text-[12px] cqa-text-[#6B7280] cqa-m-0\">What should the agent achieve?</p>\n <cqa-custom-textarea\n [placeholder]=\"fieldMeta[key].placeholder || ''\"\n [value]=\"form.get(key)?.value\"\n [fullWidth]=\"true\"\n [rows]=\"fieldMeta[key].rows || 4\"\n customClass=\"cqa-p-2 cqa-text-[14px] cqa-leading-[20px]\"\n (valueChange)=\"form.get(key)?.setValue($event)\">\n </cqa-custom-textarea>\n <p *ngIf=\"fieldMeta[key].tip\" class=\"cqa-text-[12px] cqa-text-[#6B7280] cqa-m-0\">{{ fieldMeta[key].tip }}</p>\n </div>\n\n <!-- code -->\n <div *ngIf=\"fieldMeta[key]?.controlType === 'code'\" class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].label }}</label>\n <cqa-custom-textarea\n [placeholder]=\"fieldMeta[key].placeholder || ''\"\n [value]=\"form.get(key)?.value\"\n [fullWidth]=\"true\"\n [rows]=\"8\"\n resize=\"vertical\"\n customClass=\"cqa-p-2 cqa-text-[14px] cqa-leading-[20px]\"\n (valueChange)=\"form.get(key)?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n\n <!-- dropdown (main + constraints; advanced-only dropdowns rendered inside Advanced section) -->\n <div *ngIf=\"fieldMeta[key]?.controlType === 'dropdown' && form.contains(key) && getSelectConfig(key) && !advancedSubfields.includes(key)\"\n class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-step-details-drawer-select\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].label }}</label>\n <cqa-dynamic-select [form]=\"form\" [config]=\"getSelectConfig(key)!\">\n </cqa-dynamic-select>\n </div>\n\n <!-- toggle (e.g. onlyUseAttachedContext, takeScreenshotsWhenVerifying) -->\n <div *ngIf=\"fieldMeta[key]?.controlType === 'toggle'\" class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2\">\n <div>\n <p class=\"cqa-text-[12px] cqa-text-[#111827] cqa-m-0\">{{ fieldMeta[key].label }}</p>\n <span *ngIf=\"fieldMeta[key].subLabel\" class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].subLabel }}</span>\n </div>\n <cqa-custom-toggle\n [checked]=\"form.get(key)?.value\"\n (checkedChange)=\"form.get(key)?.setValue($event)\">\n </cqa-custom-toggle>\n </div>\n </ng-container>\n </ng-container>\n </div>\n </ng-container>\n\n <p *ngIf=\"!stepType\" class=\"cqa-text-[14px] cqa-text-[#6B7280]\">This step type is not supported in the drawer.</p>\n </div>\n<div class=\"cqa-border cqa-border-solid cqa-border-[#E5E5E5]\"> </div>\n <!-- Footer -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-start cqa-gap-2 cqa-px-4 cqa-py-4 cqa-border-t cqa-border-[#E5E7EB] cqa-bg-[#F5F5F54D]\">\n <cqa-button variant=\"text\" btnSize=\"lg\" [text]=\"'Cancel'\" (clicked)=\"onCancel()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-text-[#374151]'\"></cqa-button>\n <div class=\"cqa-flex-1 cqa-min-w-0\"></div>\n <cqa-button variant=\"outlined\" btnSize=\"lg\" [text]=\"'Save as Template'\" (clicked)=\"onSaveAsTemplate()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"></cqa-button>\n <cqa-button variant=\"filled\" btnSize=\"lg\" [text]=\"'Save Changes'\" (clicked)=\"onSaveChanges()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-bg-[#3F43EE] cqa-border-[#3F43EE]'\"></cqa-button>\n </div>\n</div>\n" }]
|
|
272
|
+
}], ctorParameters: function () { return [{ type: i9.StepDetailsDrawerRef, decorators: [{
|
|
273
|
+
type: Inject,
|
|
274
|
+
args: [STEP_DETAILS_DRAWER_REF]
|
|
275
|
+
}] }, { type: i1.FormBuilder }, { type: i0.ChangeDetectorRef }, { type: undefined, decorators: [{
|
|
276
|
+
type: Optional
|
|
277
|
+
}, {
|
|
278
|
+
type: Inject,
|
|
279
|
+
args: [STEP_DETAILS_DRAWER_DATA]
|
|
280
|
+
}] }]; }, propDecorators: { saveChanges: [{
|
|
281
|
+
type: Output
|
|
282
|
+
}], cancel: [{
|
|
283
|
+
type: Output
|
|
284
|
+
}], saveAsTemplate: [{
|
|
285
|
+
type: Output
|
|
286
|
+
}], stepData: [{
|
|
287
|
+
type: Input
|
|
288
|
+
}], stepNumberInput: [{
|
|
289
|
+
type: Input
|
|
290
|
+
}] } });
|
|
291
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"step-details-drawer.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/step-details-drawer/step-details-drawer.component.ts","../../../../../../src/lib/test-case-details/step-details-drawer/step-details-drawer.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EAEvB,SAAS,EACT,YAAY,EACZ,MAAM,EACN,KAAK,EAGL,QAAQ,EACR,MAAM,GAEP,MAAM,eAAe,CAAC;AAEvB,OAAO,EACL,sBAAsB,EAItB,2BAA2B,EAC3B,uBAAuB,EACvB,oBAAoB,EACpB,0BAA0B,GAC3B,MAAM,oCAAoC,CAAC;AAC5C,OAAO,EAAE,uBAAuB,EAAwB,MAAM,2BAA2B,CAAC;AAC1F,OAAO,EAAE,wBAAwB,EAAyB,MAAM,4BAA4B,CAAC;AAC7F,OAAO,EAKL,kBAAkB,EAClB,eAAe,EACf,mBAAmB,GACpB,MAAM,0BAA0B,CAAC;;;;;;;;;;;AAMlC;;;;GAIG;AAOH,MAAM,OAAO,0BAA0B;IA2BrC,YAC2C,GAAyB,EAC1D,EAAe,EACf,GAAsB,EACgB,IAA4B;QAHjC,QAAG,GAAH,GAAG,CAAsB;QAC1D,OAAE,GAAF,EAAE,CAAa;QACf,QAAG,GAAH,GAAG,CAAmB;QA7BtB,gBAAW,GAAG,IAAI,YAAY,EAAsB,CAAC;QACrD,WAAM,GAAG,IAAI,YAAY,EAAQ,CAAC;QAClC,mBAAc,GAAG,IAAI,YAAY,EAAsB,CAAC;QAElE,uGAAuG;QAC9F,aAAQ,GAA8B,IAAI,CAAC;QAC3C,oBAAe,GAA2B,IAAI,CAAC;QAIxD,aAAQ,GAA+B,IAAI,CAAC;QAC5C,kBAAa,GAA0B,EAAE,CAAC;QAC1C,cAAS,GAAG,uBAAuB,CAAC;QACpC,uBAAkB,GAAG,oBAAoB,CAAC;QAC1C,qBAAgB,GAAG,KAAK,CAAC;QACzB,sBAAiB,GAAG,IAAI,CAAC;QAGzB,wGAAwG;QAC/F,oBAAe,GAA6C,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAa/F,IAAI,CAAC,IAAI,GAAG,IAAI,EAAE,IAAI,IAAK,EAAyB,CAAC;QACrD,IAAI,CAAC,UAAU,GAAG,IAAI,EAAE,UAAU,IAAI,CAAC,CAAC;QACxC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QAC9B,IAAI,CAAC,mBAAmB,EAAE,CAAC;IAC7B,CAAC;IAfD,uEAAuE;IACvE,IAAI,iBAAiB;QACnB,OAAO,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,0BAA0B,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;IACxE,CAAC;IAcD,QAAQ;QACN,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,EAAE;YACzB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,QAAQ,CAAC;YAC1B,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,eAAe,IAAK,IAAI,CAAC,QAAgB,CAAC,UAAU,IAAI,CAAC,CAAC;YACjF,IAAI,CAAC,mBAAmB,EAAE,CAAC;YAC3B,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;SACzB;IACH,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,UAAU,CAAC,IAAI,OAAO,CAAC,iBAAiB,CAAC,EAAE;YACrD,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,EAAE;gBACzB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,QAAQ,CAAC;gBAC1B,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,eAAe,IAAK,IAAI,CAAC,QAAgB,CAAC,UAAU,IAAI,CAAC,CAAC;aAClF;YACD,IAAI,CAAC,mBAAmB,EAAE,CAAC;YAC3B,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;SAC1B;IACH,CAAC;IAEO,mBAAmB;QACzB,IAAI,CAAC,QAAQ,GAAG,sBAAsB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAClD,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,2BAA2B,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QACrF,IAAI,CAAC,iBAAiB,EAAE,CAAC;IAC3B,CAAC;IAEO,iBAAiB;QACvB,MAAM,KAAK,GAA4B,EAAE,CAAC;QAC1C,KAAK,MAAM,GAAG,IAAI,IAAI,CAAC,aAAa,EAAE;YACpC,IAAI,GAAG,KAAK,UAAU;gBAAE,SAAS;YACjC,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC;SACvC;QACD,KAAK,MAAM,GAAG,IAAI,oBAAoB,EAAE;YACtC,IAAI,CAAC,CAAC,GAAG,IAAI,KAAK,CAAC;gBAAE,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC;SAC5D;QACD,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IACnC,CAAC;IAED,YAAY,CAAC,GAAwB;QACnC,MAAM,IAAI,GAAG,IAAI,CAAC,IAA0C,CAAC;QAC7D,IAAI,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,IAAK,IAAI,CAAC,IAAyB,CAAC,SAAS,KAAK,QAAQ,EAAE;YAC3F,IAAI,GAAG,KAAK,aAAa,EAAE;gBACzB,MAAM,MAAM,GAAI,IAAI,CAAC,IAAyB,CAAC,UAAU,CAAC;gBAC1D,OAAO,MAAM,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,IAAK,IAAI,CAAC,IAAyB,CAAC,WAAW,IAAI,EAAE,CAAC;aAChF;YACD,IAAI,GAAG,KAAK,UAAU,EAAE;gBACtB,MAAM,MAAM,GAAI,IAAI,CAAC,IAAyB,CAAC,UAAU,CAAC;gBAC1D,OAAO,MAAM,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,IAAI,EAAE,CAAC;aACjC;SACF;QACD,IAAI,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;YAC9B,MAAM,GAAG,GAAG,IAAI,CAAC,IAAqB,CAAC;YACvC,IAAI,GAAG,KAAK,QAAQ;gBAAE,OAAO,GAAG,CAAC,MAAM,IAAI,KAAK,CAAC;YACjD,IAAI,GAAG,KAAK,KAAK;gBAAE,OAAO,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC,OAAO,GAAG,GAAG,CAAC,QAAQ,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,QAAQ,IAAI,EAAE,CAAC;YACnG,IAAI,GAAG,KAAK,SAAS;gBAAE,OAAO,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACtF,IAAI,GAAG,KAAK,MAAM;gBAAE,OAAO,GAAG,CAAC,WAAW,IAAI,EAAE,CAAC;YACjD,IAAI,GAAG,KAAK,sBAAsB;gBAAE,OAAO,GAAG,CAAC,MAAM,IAAI,EAAE,CAAC;SAC7D;QACD,IAAI,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;YAClC,MAAM,EAAE,GAAG,IAAI,CAAC,IAAyB,CAAC;YAC1C,IAAI,GAAG,KAAK,WAAW;gBAAE,OAAO,EAAE,CAAC,YAAY,IAAI,EAAE,CAAC;YACtD,IAAI,GAAG,KAAK,aAAa;gBAAE,OAAQ,EAAU,CAAC,WAAW,IAAI,EAAE,CAAC;YAChE,IAAI,GAAG,KAAK,UAAU;gBAAE,OAAQ,IAAY,CAAC,QAAQ,IAAI,EAAE,CAAC;SAC7D;QACD,IACE,GAAG,KAAK,wBAAwB;YAChC,GAAG,KAAK,8BAA8B;YACtC,GAAG,KAAK,iBAAiB;YACzB,GAAG,KAAK,UAAU,EAClB;YACA,OAAQ,IAAY,CAAC,GAAG,CAAC,KAAK,IAAI,CAAC;SACpC;QACD,OAAQ,IAAY,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC;IAClC,CAAC;IAED,YAAY,CAAC,GAAwB,EAAE,KAAc;QACnD,MAAM,IAAI,GAAG,IAAI,CAAC,IAA0C,CAAC;QAC7D,IAAI,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,IAAK,IAAI,CAAC,IAAyB,CAAC,SAAS,KAAK,QAAQ,EAAE;YAC3F,IAAI,GAAG,KAAK,aAAa,EAAE;gBACzB,MAAM,MAAM,GAAI,IAAI,CAAC,IAAyB,CAAC,UAAU,CAAC;gBAC1D,IAAI,MAAM,EAAE,CAAC,CAAC,CAAC;oBAAE,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;;oBAC3C,IAAI,CAAC,IAAyB,CAAC,WAAW,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;gBACjE,OAAO;aACR;YACD,IAAI,GAAG,KAAK,UAAU,EAAE;gBACtB,MAAM,MAAM,GAAI,IAAI,CAAC,IAAyB,CAAC,UAAU,CAAC;gBAC1D,IAAI,MAAM,EAAE,CAAC,CAAC,CAAC;oBAAE,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;gBACjD,OAAO;aACR;SACF;QACD,IAAI,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;YAC9B,MAAM,GAAG,GAAG,IAAI,CAAC,IAAqB,CAAC;YACvC,IAAI,GAAG,KAAK,QAAQ,EAAE;gBAAE,GAAG,CAAC,MAAM,GAAG,KAAgC,CAAC;gBAAC,OAAO;aAAE;YAChF,IAAI,GAAG,KAAK,KAAK,EAAE;gBAAE,GAAG,CAAC,QAAQ,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;gBAAC,OAAO;aAAE;YAC5D,IAAI,GAAG,KAAK,SAAS,EAAE;gBACrB,IAAI;oBAAE,GAAG,CAAC,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,IAAI,CAAC,CAAC;iBAAE;gBAAC,MAAM;oBAAE,GAAG,CAAC,OAAO,GAAG,EAAE,CAAC;iBAAE;gBACpF,OAAO;aACR;YACD,IAAI,GAAG,KAAK,MAAM,EAAE;gBAAE,GAAG,CAAC,WAAW,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;gBAAC,OAAO;aAAE;YAChE,IAAI,GAAG,KAAK,sBAAsB,EAAE;gBAAE,GAAG,CAAC,MAAM,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;gBAAC,OAAO;aAAE;SAC5E;QACD,IAAI,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;YAClC,MAAM,EAAE,GAAG,IAAI,CAAC,IAAyB,CAAC;YAC1C,IAAI,GAAG,KAAK,WAAW,EAAE;gBAAE,EAAE,CAAC,YAAY,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;gBAAC,OAAO;aAAE;YACrE,IAAI,GAAG,KAAK,aAAa,EAAE;gBAAG,EAAU,CAAC,WAAW,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;gBAAC,OAAO;aAAE;YAC/E,IAAI,GAAG,KAAK,UAAU,EAAE;gBAAG,IAAY,CAAC,QAAQ,GAAG,KAAK,CAAC;gBAAC,OAAO;aAAE;SACpE;QACA,IAAY,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;IAC7B,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;QACnB,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;IAC5B,CAAC;IAED,OAAO;QACL,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;QACnB,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;IAC5B,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;QACnB,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;IAC5B,CAAC;IAED,gBAAgB;QACd,IAAI,CAAC,eAAe,EAAE,CAAC;QACvB,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACpC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,MAAM,EAAE,gBAAgB,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;IAChE,CAAC;IAED,aAAa;QACX,IAAI,CAAC,eAAe,EAAE,CAAC;QACvB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACjC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,MAAM,EAAE,aAAa,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;IAC7D,CAAC;IAEO,eAAe;QACrB,KAAK,MAAM,GAAG,IAAI,IAAI,CAAC,aAAa,EAAE;YACpC,IAAI,GAAG,KAAK,UAAU;gBAAE,SAAS;YACjC,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC;gBAAE,IAAI,CAAC,YAAY,CAAC,GAAG,EAAE,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,KAAK,CAAC,CAAC;SAChF;QACD,KAAK,MAAM,GAAG,IAAI,oBAAoB,EAAE;YACtC,IAAI,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC;gBAAE,IAAI,CAAC,YAAY,CAAC,GAAG,EAAE,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,KAAK,CAAC,CAAC;SAChF;IACH,CAAC;IAED,4FAA4F;IACpF,oBAAoB;QAC1B,MAAM,GAAG,GAA6C,EAAE,CAAC;QACzD,MAAM,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,uBAAuB,CAA0B,CAAC;QAC3E,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE;YACtB,MAAM,IAAI,GAAG,uBAAuB,CAAC,GAAG,CAAC,CAAC;YAC1C,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,WAAW,KAAK,UAAU,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,MAAM;gBAAE,SAAS;YAChF,MAAM,OAAO,GAAmB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;YAC9F,GAAG,CAAC,GAAG,CAAC,GAAG;gBACT,GAAG,EAAE,IAAI,CAAC,GAAG;gBACb,WAAW,EAAE,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,KAAK;gBAC3C,OAAO;aACR,CAAC;SACH;QACD,OAAO,GAAG,CAAC;IACb,CAAC;IAED,yFAAyF;IACzF,eAAe,CAAC,GAAwB;QACtC,OAAO,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC;IAC3C,CAAC;IAED,mFAAmF;IACnF,gBAAgB,CAAC,GAAwB,EAAE,OAAgC;QACzE,MAAM,IAAI,GAAG,uBAAuB,CAAC,GAAG,CAAC,CAAC;QAC1C,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,OAAO,KAAK,OAAO;YAAE,OAAO,KAAK,CAAC;QACpD,MAAM,GAAG,GAAG,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAC5C,IAAI,GAAG,IAAI,CAAC;YAAE,OAAO,IAAI,CAAC;QAC1B,MAAM,QAAQ,GAAG,uBAAuB,CAAC,IAAI,CAAC,aAAa,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;QACtE,OAAO,CAAC,QAAQ,IAAI,QAAQ,CAAC,OAAO,KAAK,OAAO,CAAC;IACnD,CAAC;IAED,wEAAwE;IACxE,IAAI,WAAW;QACb,IAAI,IAAI,CAAC,QAAQ,KAAK,QAAQ;YAAE,OAAO,qBAAqB,CAAC;QAC7D,IAAI,IAAI,CAAC,QAAQ,KAAK,KAAK;YAAE,OAAO,kBAAkB,CAAC;QACvD,IAAI,IAAI,CAAC,QAAQ,KAAK,SAAS;YAAE,OAAO,uBAAuB,CAAC;QAChE,OAAO,cAAc,CAAC;IACxB,CAAC;;uHAhOU,0BAA0B,kBA4B3B,uBAAuB,yEAGX,wBAAwB;2GA/BnC,0BAA0B,gSCnDvC,qxTA2JA;2FDxGa,0BAA0B;kBANtC,SAAS;+BACE,yBAAyB,QAE7B,EAAE,KAAK,EAAE,aAAa,EAAE,mBACb,uBAAuB,CAAC,MAAM;;0BA8B5C,MAAM;2BAAC,uBAAuB;;0BAG9B,QAAQ;;0BAAI,MAAM;2BAAC,wBAAwB;4CA9BpC,WAAW;sBAApB,MAAM;gBACG,MAAM;sBAAf,MAAM;gBACG,cAAc;sBAAvB,MAAM;gBAGE,QAAQ;sBAAhB,KAAK;gBACG,eAAe;sBAAvB,KAAK","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Inject,\n  Input,\n  OnChanges,\n  OnInit,\n  Optional,\n  Output,\n  SimpleChanges,\n} from '@angular/core';\nimport { FormBuilder, FormGroup } from '@angular/forms';\nimport {\n  getStepDetailsStepType,\n  StepDetailsFieldKey,\n  StepDetailsStepType,\n  StepDetailsFieldSection,\n  STEP_DETAILS_FIELDS_BY_TYPE,\n  STEP_DETAILS_FIELD_META,\n  ADVANCED_TOGGLE_KEYS,\n  ADVANCED_SUBFIELDS_BY_TYPE,\n} from './step-details-drawer-field.config';\nimport { STEP_DETAILS_DRAWER_REF, StepDetailsDrawerRef } from './step-details-drawer-ref';\nimport { STEP_DETAILS_DRAWER_DATA, StepDetailsDrawerData } from './step-details-drawer-data';\nimport {\n  TestCaseStepConfig,\n  NormalStepConfig,\n  ApiStepConfig,\n  AiAgentStepConfig,\n  isNormalStepConfig,\n  isApiStepConfig,\n  isAiAgentStepConfig,\n} from '../test-case-step.models';\nimport type {\n  DynamicSelectFieldConfig,\n  SelectOption,\n} from '../../dynamic-select/dynamic-select-field.component';\n\n/**\n * Step Details Drawer (Edit In Depth).\n * Single reusable component; form is dynamic by Step Type via configuration.\n * No Priority section. Uses CQA components only.\n */\n@Component({\n  selector: 'cqa-step-details-drawer',\n  templateUrl: './step-details-drawer.component.html',\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class StepDetailsDrawerComponent implements OnInit, OnChanges {\n  @Output() saveChanges = new EventEmitter<TestCaseStepConfig>();\n  @Output() cancel = new EventEmitter<void>();\n  @Output() saveAsTemplate = new EventEmitter<TestCaseStepConfig>();\n\n  /** Optional: when provided (e.g. from Storybook Controls), use instead of STEP_DETAILS_DRAWER_DATA. */\n  @Input() stepData: TestCaseStepConfig | null = null;\n  @Input() stepNumberInput: number | string | null = null;\n\n  step: TestCaseStepConfig;\n  stepNumber: number | string;\n  stepType: StepDetailsStepType | null = null;\n  visibleFields: StepDetailsFieldKey[] = [];\n  fieldMeta = STEP_DETAILS_FIELD_META;\n  advancedToggleKeys = ADVANCED_TOGGLE_KEYS;\n  advancedExpanded = false;\n  agentTaskExpanded = true;\n  form: FormGroup;\n\n  /** Cached select configs (stable references per key) to avoid infinite change detection in template. */\n  readonly selectConfigMap: Record<string, DynamicSelectFieldConfig> = this.buildSelectConfigMap();\n\n  /** Subfields to show inside Advanced section for current step type. */\n  get advancedSubfields(): StepDetailsFieldKey[] {\n    return this.stepType ? ADVANCED_SUBFIELDS_BY_TYPE[this.stepType] : [];\n  }\n\n  constructor(\n    @Inject(STEP_DETAILS_DRAWER_REF) private ref: StepDetailsDrawerRef,\n    private fb: FormBuilder,\n    private cdr: ChangeDetectorRef,\n    @Optional() @Inject(STEP_DETAILS_DRAWER_DATA) data?: StepDetailsDrawerData\n  ) {\n    this.step = data?.step ?? ({} as TestCaseStepConfig);\n    this.stepNumber = data?.stepNumber ?? 1;\n    this.form = this.fb.group({});\n    this.syncStepTypeAndForm();\n  }\n\n  ngOnInit(): void {\n    if (this.stepData != null) {\n      this.step = this.stepData;\n      this.stepNumber = this.stepNumberInput ?? (this.stepData as any).stepNumber ?? 1;\n      this.syncStepTypeAndForm();\n      this.cdr.markForCheck();\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['stepData'] || changes['stepNumberInput']) {\n      if (this.stepData != null) {\n        this.step = this.stepData;\n        this.stepNumber = this.stepNumberInput ?? (this.stepData as any).stepNumber ?? 1;\n      }\n      this.syncStepTypeAndForm();\n      this.cdr.detectChanges();\n    }\n  }\n\n  private syncStepTypeAndForm(): void {\n    this.stepType = getStepDetailsStepType(this.step);\n    this.visibleFields = this.stepType ? STEP_DETAILS_FIELDS_BY_TYPE[this.stepType] : [];\n    this.buildFormFromStep();\n  }\n\n  private buildFormFromStep(): void {\n    const group: Record<string, unknown> = {};\n    for (const key of this.visibleFields) {\n      if (key === 'advanced') continue;\n      group[key] = [this.getStepValue(key)];\n    }\n    for (const key of ADVANCED_TOGGLE_KEYS) {\n      if (!(key in group)) group[key] = [this.getStepValue(key)];\n    }\n    this.form = this.fb.group(group);\n  }\n\n  getStepValue(key: StepDetailsFieldKey): unknown {\n    const step = this.step as unknown as Record<string, unknown>;\n    if (isNormalStepConfig(this.step) && (this.step as NormalStepConfig).eventType === 'custom') {\n      if (key === 'description') {\n        const params = (this.step as NormalStepConfig).parameters;\n        return params?.[0]?.value ?? (this.step as NormalStepConfig).description ?? '';\n      }\n      if (key === 'metadata') {\n        const params = (this.step as NormalStepConfig).parameters;\n        return params?.[1]?.value ?? '';\n      }\n    }\n    if (isApiStepConfig(this.step)) {\n      const api = this.step as ApiStepConfig;\n      if (key === 'method') return api.method ?? 'GET';\n      if (key === 'url') return api.baseUrl ? `${api.baseUrl}${api.endpoint || ''}` : api.endpoint ?? '';\n      if (key === 'headers') return api.headers ? JSON.stringify(api.headers, null, 2) : '';\n      if (key === 'body') return api.requestBody ?? '';\n      if (key === 'saveOutputAsVariable') return api.saveTo ?? '';\n    }\n    if (isAiAgentStepConfig(this.step)) {\n      const ai = this.step as AiAgentStepConfig;\n      if (key === 'agentTask') return ai.instructions ?? '';\n      if (key === 'description') return (ai as any).description ?? '';\n      if (key === 'metadata') return (step as any).metadata ?? '';\n    }\n    if (\n      key === 'onlyUseAttachedContext' ||\n      key === 'takeScreenshotsWhenVerifying' ||\n      key === 'continueOnError' ||\n      key === 'disabled'\n    ) {\n      return (step as any)[key] === true;\n    }\n    return (step as any)[key] ?? '';\n  }\n\n  setStepValue(key: StepDetailsFieldKey, value: unknown): void {\n    const step = this.step as unknown as Record<string, unknown>;\n    if (isNormalStepConfig(this.step) && (this.step as NormalStepConfig).eventType === 'custom') {\n      if (key === 'description') {\n        const params = (this.step as NormalStepConfig).parameters;\n        if (params?.[0]) params[0].value = String(value);\n        else (this.step as NormalStepConfig).description = String(value);\n        return;\n      }\n      if (key === 'metadata') {\n        const params = (this.step as NormalStepConfig).parameters;\n        if (params?.[1]) params[1].value = String(value);\n        return;\n      }\n    }\n    if (isApiStepConfig(this.step)) {\n      const api = this.step as ApiStepConfig;\n      if (key === 'method') { api.method = value as ApiStepConfig['method']; return; }\n      if (key === 'url') { api.endpoint = String(value); return; }\n      if (key === 'headers') {\n        try { api.headers = JSON.parse(String(value) || '{}'); } catch { api.headers = {}; }\n        return;\n      }\n      if (key === 'body') { api.requestBody = String(value); return; }\n      if (key === 'saveOutputAsVariable') { api.saveTo = String(value); return; }\n    }\n    if (isAiAgentStepConfig(this.step)) {\n      const ai = this.step as AiAgentStepConfig;\n      if (key === 'agentTask') { ai.instructions = String(value); return; }\n      if (key === 'description') { (ai as any).description = String(value); return; }\n      if (key === 'metadata') { (step as any).metadata = value; return; }\n    }\n    (step as any)[key] = value;\n  }\n\n  onBack(): void {\n    this.cancel.emit();\n    this.ref.close(undefined);\n  }\n\n  onClose(): void {\n    this.cancel.emit();\n    this.ref.close(undefined);\n  }\n\n  onCancel(): void {\n    this.cancel.emit();\n    this.ref.close(undefined);\n  }\n\n  onSaveAsTemplate(): void {\n    this.applyFormToStep();\n    this.saveAsTemplate.emit(this.step);\n    this.ref.close({ action: 'saveAsTemplate', step: this.step });\n  }\n\n  onSaveChanges(): void {\n    this.applyFormToStep();\n    this.saveChanges.emit(this.step);\n    this.ref.close({ action: 'saveChanges', step: this.step });\n  }\n\n  private applyFormToStep(): void {\n    for (const key of this.visibleFields) {\n      if (key === 'advanced') continue;\n      if (this.form.contains(key)) this.setStepValue(key, this.form.get(key)?.value);\n    }\n    for (const key of ADVANCED_TOGGLE_KEYS) {\n      if (this.form.contains(key)) this.setStepValue(key, this.form.get(key)?.value);\n    }\n  }\n\n  /** Build cached map of select configs from field meta (stable refs to prevent CD loops). */\n  private buildSelectConfigMap(): Record<string, DynamicSelectFieldConfig> {\n    const map: Record<string, DynamicSelectFieldConfig> = {};\n    const keys = Object.keys(STEP_DETAILS_FIELD_META) as StepDetailsFieldKey[];\n    for (const key of keys) {\n      const meta = STEP_DETAILS_FIELD_META[key];\n      if (!meta || meta.controlType !== 'dropdown' || !meta.options?.length) continue;\n      const options: SelectOption[] = meta.options.map((o) => ({ value: o.value, label: o.label }));\n      map[key] = {\n        key: meta.key,\n        placeholder: meta.placeholder ?? meta.label,\n        options,\n      };\n    }\n    return map;\n  }\n\n  /** Return cached select config for key (for template use; same reference every time). */\n  getSelectConfig(key: StepDetailsFieldKey): DynamicSelectFieldConfig | null {\n    return this.selectConfigMap[key] ?? null;\n  }\n\n  /** Whether this key is the first in its section (for showing section headings). */\n  isFirstInSection(key: StepDetailsFieldKey, section: StepDetailsFieldSection): boolean {\n    const meta = STEP_DETAILS_FIELD_META[key];\n    if (!meta || meta.section !== section) return false;\n    const idx = this.visibleFields.indexOf(key);\n    if (idx <= 0) return true;\n    const prevMeta = STEP_DETAILS_FIELD_META[this.visibleFields[idx - 1]];\n    return !prevMeta || prevMeta.section !== section;\n  }\n\n  /** Dynamic drawer title by step type (e.g. \"AI Agent Step Details\"). */\n  get drawerTitle(): string {\n    if (this.stepType === 'custom') return 'Custom Step Details';\n    if (this.stepType === 'api') return 'API Step Details';\n    if (this.stepType === 'aiAgent') return 'AI Agent Step Details';\n    return 'Step Details';\n  }\n}\n","<!-- Step Details Drawer – Edit In Depth. No Priority. Fully config-driven. -->\n<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-w-full cqa-max-w-[480px] cqa-box-border\">\n  <!-- Header: back chevron, dynamic title, Step N pill, close -->\n  <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-3 cqa-px-4 cqa-py-4 cqa-bg-white cqa-rounded-t-[12px] cqa-border cqa-border-[#E5E5E5]\">\n    <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n      <button type=\"button\" (click)=\"onBack()\"\n        class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-8 cqa-h-8 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0\"\n        aria-label=\"Back\">\n        <mat-icon class=\"!cqa-w-5 !cqa-h-5 !cqa-text-[20px]\">chevron_left</mat-icon>\n      </button>\n      <h2 class=\"cqa-text-[16px] cqa-leading-[24px] cqa-font-semibold cqa-text-[#111827] cqa-m-0 cqa-flex-1\">\n        {{ drawerTitle }}\n      </h2>\n      <span class=\"cqa-px-2.5 cqa-py-1 cqa-rounded-lg cqa-bg-[#EDE9FE] cqa-text-[#7C3AED] cqa-text-[12px] cqa-font-medium\">\n        Step {{ stepNumber }}\n      </span>\n    </div>\n    <button type=\"button\" (click)=\"onClose()\"\n      class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-[28px] cqa-min-w-[28px] cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0\"\n      title=\"Close\" aria-label=\"Close\">\n      <mat-icon class=\"!cqa-w-5 !cqa-h-5 !cqa-text-[20px]\">close</mat-icon>\n    </button>\n  </div>\n  <div class=\"cqa-border cqa-border-solid cqa-border-[#E5E5E5]\"> </div>\n\n  <!-- Content: Attributes + dynamic fields from config -->\n  <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-bg-white cqa-px-4 cqa-py-4\">\n    <ng-container *ngIf=\"stepType\">\n      <p class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#6B7280] cqa-mb-3 cqa-mt-0\">Attributes</p>\n\n      <div class=\"cqa-flex cqa-flex-col cqa-gap-4\">\n        <ng-container *ngFor=\"let key of visibleFields\">\n          <!-- Section heading: Constraints (first constraints field only) -->\n          <ng-container *ngIf=\"key !== 'advanced' && isFirstInSection(key, 'constraints')\">\n            <p class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#6B7280] cqa-mb-2 cqa-mt-0\">Constraints</p>\n          </ng-container>\n\n          <!-- Advanced: expandable section with config-driven toggles + subfields -->\n          <ng-container *ngIf=\"key === 'advanced'\">\n            <button type=\"button\"\n              class=\"cqa-flex cqa-items-center cqa-justify-between cqa-w-full cqa-text-left cqa-text-[12px] cqa-font-medium cqa-text-[#111827] cqa-py-2 cqa-border-b cqa-border-[#E5E7EB]\"\n              (click)=\"advancedExpanded = !advancedExpanded\">\n              <span>Advanced</span>\n              <mat-icon class=\"cqa-text-[20px] cqa-transition-transform\" [class.cqa-rotate-180]=\"!advancedExpanded\">\n                expand_less\n              </mat-icon>\n            </button>\n            <div *ngIf=\"advancedExpanded\" class=\"cqa-flex cqa-flex-col cqa-gap-3 cqa-pt-2\">\n              <!-- Advanced toggles from config -->\n              <ng-container *ngFor=\"let toggleKey of advancedToggleKeys\">\n                <div class=\"cqa-flex cqa-items-center cqa-gap-2\" *ngIf=\"fieldMeta[toggleKey]\">\n                  <cqa-custom-toggle\n                    [checked]=\"form.get(toggleKey)?.value\"\n                    (checkedChange)=\"form.get(toggleKey)?.setValue($event)\">\n                  </cqa-custom-toggle>\n                  <div class=\"cqa-flex cqa-flex-col\">\n                    <span class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#737373]\">{{ fieldMeta[toggleKey].label }}</span>\n                    <span class=\"cqa-text-[12px] cqa-text-[#0A0A0A]\" *ngIf=\"fieldMeta[toggleKey].subLabel\">{{ fieldMeta[toggleKey].subLabel }}</span>\n                  </div>\n                </div>\n              </ng-container>\n              <!-- Advanced subfields (e.g. Retry Count, Iframe locator, Other Locators for aiAgent) -->\n              <ng-container *ngFor=\"let subKey of advancedSubfields\">\n                <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-step-details-drawer-select\" *ngIf=\"fieldMeta[subKey]?.controlType === 'dropdown' && getSelectConfig(subKey) && form.contains(subKey)\">\n                  <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[subKey].label }}</label>\n                  <cqa-dynamic-select [form]=\"form\" [config]=\"getSelectConfig(subKey)!\">\n                  </cqa-dynamic-select>\n                </div>\n              </ng-container>\n            </div>\n          </ng-container>\n\n          <!-- Skip: advanced key (already rendered) and fields that are rendered inside advanced for this step type -->\n          <ng-container *ngIf=\"key !== 'advanced' && !(stepType === 'aiAgent' && advancedSubfields.includes(key))\">\n            <!-- text -->\n            <div *ngIf=\"fieldMeta[key]?.controlType === 'text'\" class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n              <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].label }}</label>\n              <cqa-custom-input\n                [placeholder]=\"fieldMeta[key].placeholder || ''\"\n                [value]=\"form.get(key)?.value\"\n                [fullWidth]=\"true\"\n                (valueChange)=\"form.get(key)?.setValue($event)\">\n              </cqa-custom-input>\n            </div>\n\n            <!-- textarea (including agentTask with optional required + tip) -->\n            <div *ngIf=\"fieldMeta[key]?.controlType === 'textarea'\" class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n              <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n                <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].label }}</label>\n                <span *ngIf=\"fieldMeta[key].required\" class=\"cqa-px-2 cqa-py-0.5 cqa-rounded-full cqa-bg-[#FEE2E2] cqa-text-[#DC2626] cqa-text-[10px] cqa-font-medium\">Required</span>\n              </div>\n              <p *ngIf=\"key === 'agentTask'\" class=\"cqa-text-[12px] cqa-text-[#6B7280] cqa-m-0\">What should the agent achieve?</p>\n              <cqa-custom-textarea\n                [placeholder]=\"fieldMeta[key].placeholder || ''\"\n                [value]=\"form.get(key)?.value\"\n                [fullWidth]=\"true\"\n                [rows]=\"fieldMeta[key].rows || 4\"\n                customClass=\"cqa-p-2 cqa-text-[14px] cqa-leading-[20px]\"\n                (valueChange)=\"form.get(key)?.setValue($event)\">\n              </cqa-custom-textarea>\n              <p *ngIf=\"fieldMeta[key].tip\" class=\"cqa-text-[12px] cqa-text-[#6B7280] cqa-m-0\">{{ fieldMeta[key].tip }}</p>\n            </div>\n\n            <!-- code -->\n            <div *ngIf=\"fieldMeta[key]?.controlType === 'code'\" class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n              <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].label }}</label>\n              <cqa-custom-textarea\n                [placeholder]=\"fieldMeta[key].placeholder || ''\"\n                [value]=\"form.get(key)?.value\"\n                [fullWidth]=\"true\"\n                [rows]=\"8\"\n                resize=\"vertical\"\n                customClass=\"cqa-p-2 cqa-text-[14px] cqa-leading-[20px]\"\n                (valueChange)=\"form.get(key)?.setValue($event)\">\n              </cqa-custom-textarea>\n            </div>\n\n            <!-- dropdown (main + constraints; advanced-only dropdowns rendered inside Advanced section) -->\n            <div *ngIf=\"fieldMeta[key]?.controlType === 'dropdown' && form.contains(key) && getSelectConfig(key) && !advancedSubfields.includes(key)\"\n              class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-step-details-drawer-select\">\n              <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].label }}</label>\n              <cqa-dynamic-select [form]=\"form\" [config]=\"getSelectConfig(key)!\">\n              </cqa-dynamic-select>\n            </div>\n\n            <!-- toggle (e.g. onlyUseAttachedContext, takeScreenshotsWhenVerifying) -->\n            <div *ngIf=\"fieldMeta[key]?.controlType === 'toggle'\" class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2\">\n              <div>\n                <p class=\"cqa-text-[12px] cqa-text-[#111827] cqa-m-0\">{{ fieldMeta[key].label }}</p>\n                <span *ngIf=\"fieldMeta[key].subLabel\" class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#374151]\">{{ fieldMeta[key].subLabel }}</span>\n              </div>\n              <cqa-custom-toggle\n                [checked]=\"form.get(key)?.value\"\n                (checkedChange)=\"form.get(key)?.setValue($event)\">\n              </cqa-custom-toggle>\n            </div>\n          </ng-container>\n        </ng-container>\n      </div>\n    </ng-container>\n\n    <p *ngIf=\"!stepType\" class=\"cqa-text-[14px] cqa-text-[#6B7280]\">This step type is not supported in the drawer.</p>\n  </div>\n<div class=\"cqa-border cqa-border-solid cqa-border-[#E5E5E5]\"> </div>\n  <!-- Footer -->\n  <div class=\"cqa-flex cqa-items-center cqa-justify-start cqa-gap-2 cqa-px-4 cqa-py-4 cqa-border-t cqa-border-[#E5E7EB] cqa-bg-[#F5F5F54D]\">\n    <cqa-button variant=\"text\" btnSize=\"lg\" [text]=\"'Cancel'\" (clicked)=\"onCancel()\"\n      [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-text-[#374151]'\"></cqa-button>\n    <div class=\"cqa-flex-1 cqa-min-w-0\"></div>\n    <cqa-button variant=\"outlined\" btnSize=\"lg\" [text]=\"'Save as Template'\" (clicked)=\"onSaveAsTemplate()\"\n      [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"></cqa-button>\n    <cqa-button variant=\"filled\" btnSize=\"lg\" [text]=\"'Save Changes'\" (clicked)=\"onSaveChanges()\"\n      [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-bg-[#3F43EE] cqa-border-[#3F43EE]'\"></cqa-button>\n  </div>\n</div>\n"]}
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
import { Injectable, Injector } from '@angular/core';
|
|
2
|
+
import { OverlayConfig } from '@angular/cdk/overlay';
|
|
3
|
+
import { ComponentPortal } from '@angular/cdk/portal';
|
|
4
|
+
import { filter } from 'rxjs/operators';
|
|
5
|
+
import { StepDetailsDrawerComponent } from './step-details-drawer.component';
|
|
6
|
+
import { StepDetailsDrawerRef } from './step-details-drawer-ref';
|
|
7
|
+
import { STEP_DETAILS_DRAWER_REF } from './step-details-drawer-ref';
|
|
8
|
+
import { STEP_DETAILS_DRAWER_DATA } from './step-details-drawer-data';
|
|
9
|
+
import * as i0 from "@angular/core";
|
|
10
|
+
import * as i1 from "@angular/cdk/overlay";
|
|
11
|
+
/**
|
|
12
|
+
* Opens the Step Details Drawer (Edit In Depth) from the right.
|
|
13
|
+
*/
|
|
14
|
+
export class StepDetailsDrawerService {
|
|
15
|
+
constructor(overlay, injector) {
|
|
16
|
+
this.overlay = overlay;
|
|
17
|
+
this.injector = injector;
|
|
18
|
+
this.currentRef = null;
|
|
19
|
+
}
|
|
20
|
+
open(data) {
|
|
21
|
+
if (this.currentRef)
|
|
22
|
+
return this.currentRef;
|
|
23
|
+
const drawerWidth = 480;
|
|
24
|
+
const positionStrategy = this.overlay
|
|
25
|
+
.position()
|
|
26
|
+
.global()
|
|
27
|
+
.right('0')
|
|
28
|
+
.top('0')
|
|
29
|
+
.bottom('0')
|
|
30
|
+
.width(`${drawerWidth}px`);
|
|
31
|
+
const overlayRef = this.overlay.create(new OverlayConfig({
|
|
32
|
+
hasBackdrop: true,
|
|
33
|
+
backdropClass: 'cdk-overlay-transparent-backdrop',
|
|
34
|
+
scrollStrategy: this.overlay.scrollStrategies.block(),
|
|
35
|
+
positionStrategy,
|
|
36
|
+
panelClass: ['cqa-step-details-drawer-panel', 'cqa-ui-root'],
|
|
37
|
+
width: `${drawerWidth}px`,
|
|
38
|
+
maxWidth: '100vw',
|
|
39
|
+
height: '100%',
|
|
40
|
+
}));
|
|
41
|
+
const drawerRef = new StepDetailsDrawerRef(overlayRef);
|
|
42
|
+
const injector = Injector.create({
|
|
43
|
+
parent: this.injector,
|
|
44
|
+
providers: [
|
|
45
|
+
{ provide: STEP_DETAILS_DRAWER_REF, useValue: drawerRef },
|
|
46
|
+
{ provide: STEP_DETAILS_DRAWER_DATA, useValue: data },
|
|
47
|
+
],
|
|
48
|
+
});
|
|
49
|
+
const portal = new ComponentPortal(StepDetailsDrawerComponent, undefined, injector);
|
|
50
|
+
overlayRef.attach(portal);
|
|
51
|
+
this.currentRef = drawerRef;
|
|
52
|
+
drawerRef.afterClosed().subscribe(() => { this.currentRef = null; });
|
|
53
|
+
overlayRef.backdropClick().subscribe(() => drawerRef.close({ action: 'cancel' }));
|
|
54
|
+
overlayRef
|
|
55
|
+
.keydownEvents()
|
|
56
|
+
.pipe(filter((e) => e.key === 'Escape' || e.key === 'Esc'))
|
|
57
|
+
.subscribe(() => drawerRef.close({ action: 'cancel' }));
|
|
58
|
+
return drawerRef;
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
StepDetailsDrawerService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepDetailsDrawerService, deps: [{ token: i1.Overlay }, { token: i0.Injector }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
62
|
+
StepDetailsDrawerService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepDetailsDrawerService, providedIn: 'root' });
|
|
63
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepDetailsDrawerService, decorators: [{
|
|
64
|
+
type: Injectable,
|
|
65
|
+
args: [{ providedIn: 'root' }]
|
|
66
|
+
}], ctorParameters: function () { return [{ type: i1.Overlay }, { type: i0.Injector }]; } });
|
|
67
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RlcC1kZXRhaWxzLWRyYXdlci5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi90ZXN0LWNhc2UtZGV0YWlscy9zdGVwLWRldGFpbHMtZHJhd2VyL3N0ZXAtZGV0YWlscy1kcmF3ZXIuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNyRCxPQUFPLEVBQVcsYUFBYSxFQUFFLE1BQU0sc0JBQXNCLENBQUM7QUFDOUQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ3RELE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUV4QyxPQUFPLEVBQUUsMEJBQTBCLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUM3RSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUNqRSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUNwRSxPQUFPLEVBQUUsd0JBQXdCLEVBQXlCLE1BQU0sNEJBQTRCLENBQUM7OztBQU83Rjs7R0FFRztBQUVILE1BQU0sT0FBTyx3QkFBd0I7SUFHbkMsWUFBNkIsT0FBZ0IsRUFBbUIsUUFBa0I7UUFBckQsWUFBTyxHQUFQLE9BQU8sQ0FBUztRQUFtQixhQUFRLEdBQVIsUUFBUSxDQUFVO1FBRjFFLGVBQVUsR0FBeUQsSUFBSSxDQUFDO0lBRUssQ0FBQztJQUV0RixJQUFJLENBQUMsSUFBMkI7UUFDOUIsSUFBSSxJQUFJLENBQUMsVUFBVTtZQUFFLE9BQU8sSUFBSSxDQUFDLFVBQVUsQ0FBQztRQUU1QyxNQUFNLFdBQVcsR0FBRyxHQUFHLENBQUM7UUFDeEIsTUFBTSxnQkFBZ0IsR0FBRyxJQUFJLENBQUMsT0FBTzthQUNsQyxRQUFRLEVBQUU7YUFDVixNQUFNLEVBQUU7YUFDUixLQUFLLENBQUMsR0FBRyxDQUFDO2FBQ1YsR0FBRyxDQUFDLEdBQUcsQ0FBQzthQUNSLE1BQU0sQ0FBQyxHQUFHLENBQUM7YUFDWCxLQUFLLENBQUMsR0FBRyxXQUFXLElBQUksQ0FBQyxDQUFDO1FBRTdCLE1BQU0sVUFBVSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUNwQyxJQUFJLGFBQWEsQ0FBQztZQUNoQixXQUFXLEVBQUUsSUFBSTtZQUNqQixhQUFhLEVBQUUsa0NBQWtDO1lBQ2pELGNBQWMsRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLGdCQUFnQixDQUFDLEtBQUssRUFBRTtZQUNyRCxnQkFBZ0I7WUFDaEIsVUFBVSxFQUFFLENBQUMsK0JBQStCLEVBQUUsYUFBYSxDQUFDO1lBQzVELEtBQUssRUFBRSxHQUFHLFdBQVcsSUFBSTtZQUN6QixRQUFRLEVBQUUsT0FBTztZQUNqQixNQUFNLEVBQUUsTUFBTTtTQUNmLENBQUMsQ0FDSCxDQUFDO1FBRUYsTUFBTSxTQUFTLEdBQUcsSUFBSSxvQkFBb0IsQ0FBMEIsVUFBVSxDQUFDLENBQUM7UUFDaEYsTUFBTSxRQUFRLEdBQUcsUUFBUSxDQUFDLE1BQU0sQ0FBQztZQUMvQixNQUFNLEVBQUUsSUFBSSxDQUFDLFFBQVE7WUFDckIsU0FBUyxFQUFFO2dCQUNULEVBQUUsT0FBTyxFQUFFLHVCQUF1QixFQUFFLFFBQVEsRUFBRSxTQUFTLEVBQUU7Z0JBQ3pELEVBQUUsT0FBTyxFQUFFLHdCQUF3QixFQUFFLFFBQVEsRUFBRSxJQUFJLEVBQUU7YUFDdEQ7U0FDRixDQUFDLENBQUM7UUFFSCxNQUFNLE1BQU0sR0FBRyxJQUFJLGVBQWUsQ0FBQywwQkFBMEIsRUFBRSxTQUFTLEVBQUUsUUFBUSxDQUFDLENBQUM7UUFDcEYsVUFBVSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUUxQixJQUFJLENBQUMsVUFBVSxHQUFHLFNBQVMsQ0FBQztRQUM1QixTQUFTLENBQUMsV0FBVyxFQUFFLENBQUMsU0FBUyxDQUFDLEdBQUcsRUFBRSxHQUFHLElBQUksQ0FBQyxVQUFVLEdBQUcsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDckUsVUFBVSxDQUFDLGFBQWEsRUFBRSxDQUFDLFNBQVMsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLEVBQUUsTUFBTSxFQUFFLFFBQVEsRUFBRSxDQUFDLENBQUMsQ0FBQztRQUNsRixVQUFVO2FBQ1AsYUFBYSxFQUFFO2FBQ2YsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQWdCLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxHQUFHLEtBQUssUUFBUSxJQUFJLENBQUMsQ0FBQyxHQUFHLEtBQUssS0FBSyxDQUFDLENBQUM7YUFDekUsU0FBUyxDQUFDLEdBQUcsRUFBRSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsRUFBRSxNQUFNLEVBQUUsUUFBUSxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBRTFELE9BQU8sU0FBUyxDQUFDO0lBQ25CLENBQUM7O3FIQW5EVSx3QkFBd0I7eUhBQXhCLHdCQUF3QixjQURYLE1BQU07MkZBQ25CLHdCQUF3QjtrQkFEcEMsVUFBVTttQkFBQyxFQUFFLFVBQVUsRUFBRSxNQUFNLEVBQUUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlLCBJbmplY3RvciB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgT3ZlcmxheSwgT3ZlcmxheUNvbmZpZyB9IGZyb20gJ0Bhbmd1bGFyL2Nkay9vdmVybGF5JztcbmltcG9ydCB7IENvbXBvbmVudFBvcnRhbCB9IGZyb20gJ0Bhbmd1bGFyL2Nkay9wb3J0YWwnO1xuaW1wb3J0IHsgZmlsdGVyIH0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuaW1wb3J0IHsgVGVzdENhc2VTdGVwQ29uZmlnIH0gZnJvbSAnLi4vdGVzdC1jYXNlLXN0ZXAubW9kZWxzJztcbmltcG9ydCB7IFN0ZXBEZXRhaWxzRHJhd2VyQ29tcG9uZW50IH0gZnJvbSAnLi9zdGVwLWRldGFpbHMtZHJhd2VyLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBTdGVwRGV0YWlsc0RyYXdlclJlZiB9IGZyb20gJy4vc3RlcC1kZXRhaWxzLWRyYXdlci1yZWYnO1xuaW1wb3J0IHsgU1RFUF9ERVRBSUxTX0RSQVdFUl9SRUYgfSBmcm9tICcuL3N0ZXAtZGV0YWlscy1kcmF3ZXItcmVmJztcbmltcG9ydCB7IFNURVBfREVUQUlMU19EUkFXRVJfREFUQSwgU3RlcERldGFpbHNEcmF3ZXJEYXRhIH0gZnJvbSAnLi9zdGVwLWRldGFpbHMtZHJhd2VyLWRhdGEnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFN0ZXBEZXRhaWxzRHJhd2VyUmVzdWx0IHtcbiAgYWN0aW9uOiAnc2F2ZUNoYW5nZXMnIHwgJ3NhdmVBc1RlbXBsYXRlJyB8ICdjYW5jZWwnO1xuICBzdGVwPzogVGVzdENhc2VTdGVwQ29uZmlnO1xufVxuXG4vKipcbiAqIE9wZW5zIHRoZSBTdGVwIERldGFpbHMgRHJhd2VyIChFZGl0IEluIERlcHRoKSBmcm9tIHRoZSByaWdodC5cbiAqL1xuQEluamVjdGFibGUoeyBwcm92aWRlZEluOiAncm9vdCcgfSlcbmV4cG9ydCBjbGFzcyBTdGVwRGV0YWlsc0RyYXdlclNlcnZpY2Uge1xuICBwcml2YXRlIGN1cnJlbnRSZWY6IFN0ZXBEZXRhaWxzRHJhd2VyUmVmPFN0ZXBEZXRhaWxzRHJhd2VyUmVzdWx0PiB8IG51bGwgPSBudWxsO1xuXG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgcmVhZG9ubHkgb3ZlcmxheTogT3ZlcmxheSwgcHJpdmF0ZSByZWFkb25seSBpbmplY3RvcjogSW5qZWN0b3IpIHt9XG5cbiAgb3BlbihkYXRhOiBTdGVwRGV0YWlsc0RyYXdlckRhdGEpOiBTdGVwRGV0YWlsc0RyYXdlclJlZjxTdGVwRGV0YWlsc0RyYXdlclJlc3VsdD4ge1xuICAgIGlmICh0aGlzLmN1cnJlbnRSZWYpIHJldHVybiB0aGlzLmN1cnJlbnRSZWY7XG5cbiAgICBjb25zdCBkcmF3ZXJXaWR0aCA9IDQ4MDtcbiAgICBjb25zdCBwb3NpdGlvblN0cmF0ZWd5ID0gdGhpcy5vdmVybGF5XG4gICAgICAucG9zaXRpb24oKVxuICAgICAgLmdsb2JhbCgpXG4gICAgICAucmlnaHQoJzAnKVxuICAgICAgLnRvcCgnMCcpXG4gICAgICAuYm90dG9tKCcwJylcbiAgICAgIC53aWR0aChgJHtkcmF3ZXJXaWR0aH1weGApO1xuXG4gICAgY29uc3Qgb3ZlcmxheVJlZiA9IHRoaXMub3ZlcmxheS5jcmVhdGUoXG4gICAgICBuZXcgT3ZlcmxheUNvbmZpZyh7XG4gICAgICAgIGhhc0JhY2tkcm9wOiB0cnVlLFxuICAgICAgICBiYWNrZHJvcENsYXNzOiAnY2RrLW92ZXJsYXktdHJhbnNwYXJlbnQtYmFja2Ryb3AnLFxuICAgICAgICBzY3JvbGxTdHJhdGVneTogdGhpcy5vdmVybGF5LnNjcm9sbFN0cmF0ZWdpZXMuYmxvY2soKSxcbiAgICAgICAgcG9zaXRpb25TdHJhdGVneSxcbiAgICAgICAgcGFuZWxDbGFzczogWydjcWEtc3RlcC1kZXRhaWxzLWRyYXdlci1wYW5lbCcsICdjcWEtdWktcm9vdCddLFxuICAgICAgICB3aWR0aDogYCR7ZHJhd2VyV2lkdGh9cHhgLFxuICAgICAgICBtYXhXaWR0aDogJzEwMHZ3JyxcbiAgICAgICAgaGVpZ2h0OiAnMTAwJScsXG4gICAgICB9KVxuICAgICk7XG5cbiAgICBjb25zdCBkcmF3ZXJSZWYgPSBuZXcgU3RlcERldGFpbHNEcmF3ZXJSZWY8U3RlcERldGFpbHNEcmF3ZXJSZXN1bHQ+KG92ZXJsYXlSZWYpO1xuICAgIGNvbnN0IGluamVjdG9yID0gSW5qZWN0b3IuY3JlYXRlKHtcbiAgICAgIHBhcmVudDogdGhpcy5pbmplY3RvcixcbiAgICAgIHByb3ZpZGVyczogW1xuICAgICAgICB7IHByb3ZpZGU6IFNURVBfREVUQUlMU19EUkFXRVJfUkVGLCB1c2VWYWx1ZTogZHJhd2VyUmVmIH0sXG4gICAgICAgIHsgcHJvdmlkZTogU1RFUF9ERVRBSUxTX0RSQVdFUl9EQVRBLCB1c2VWYWx1ZTogZGF0YSB9LFxuICAgICAgXSxcbiAgICB9KTtcblxuICAgIGNvbnN0IHBvcnRhbCA9IG5ldyBDb21wb25lbnRQb3J0YWwoU3RlcERldGFpbHNEcmF3ZXJDb21wb25lbnQsIHVuZGVmaW5lZCwgaW5qZWN0b3IpO1xuICAgIG92ZXJsYXlSZWYuYXR0YWNoKHBvcnRhbCk7XG5cbiAgICB0aGlzLmN1cnJlbnRSZWYgPSBkcmF3ZXJSZWY7XG4gICAgZHJhd2VyUmVmLmFmdGVyQ2xvc2VkKCkuc3Vic2NyaWJlKCgpID0+IHsgdGhpcy5jdXJyZW50UmVmID0gbnVsbDsgfSk7XG4gICAgb3ZlcmxheVJlZi5iYWNrZHJvcENsaWNrKCkuc3Vic2NyaWJlKCgpID0+IGRyYXdlclJlZi5jbG9zZSh7IGFjdGlvbjogJ2NhbmNlbCcgfSkpO1xuICAgIG92ZXJsYXlSZWZcbiAgICAgIC5rZXlkb3duRXZlbnRzKClcbiAgICAgIC5waXBlKGZpbHRlcigoZTogS2V5Ym9hcmRFdmVudCkgPT4gZS5rZXkgPT09ICdFc2NhcGUnIHx8IGUua2V5ID09PSAnRXNjJykpXG4gICAgICAuc3Vic2NyaWJlKCgpID0+IGRyYXdlclJlZi5jbG9zZSh7IGFjdGlvbjogJ2NhbmNlbCcgfSkpO1xuXG4gICAgcmV0dXJuIGRyYXdlclJlZjtcbiAgfVxufVxuIl19
|