@cqa-lib/cqa-ui 1.1.238 → 1.1.239
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2020/lib/test-case-details/ai-agent-step/ai-agent-step.component.mjs +3 -3
- package/esm2020/lib/test-case-details/ai-verify-step/ai-verify-step.component.mjs +3 -3
- package/esm2020/lib/test-case-details/api-step/api-step.component.mjs +3 -3
- package/esm2020/lib/test-case-details/condition-step/condition-step.component.mjs +38 -3
- package/esm2020/lib/test-case-details/custom-code-step/custom-code-step.component.mjs +3 -3
- package/esm2020/lib/test-case-details/database-step/database-step.component.mjs +3 -3
- package/esm2020/lib/test-case-details/loop-step/loop-step.component.mjs +506 -44
- package/esm2020/lib/test-case-details/normal-step/normal-step.component.mjs +3 -3
- package/esm2020/lib/test-case-details/restore-session-step/restore-session-step.component.mjs +3 -3
- package/esm2020/lib/test-case-details/screenshot-step/screenshot-step.component.mjs +3 -3
- package/esm2020/lib/test-case-details/scroll-step/scroll-step.component.mjs +3 -3
- package/esm2020/lib/test-case-details/step-group/step-group.component.mjs +18 -5
- package/esm2020/lib/test-case-details/test-case-details-renderer/test-case-details-renderer.component.mjs +21 -7
- package/esm2020/lib/test-case-details/test-case-step.models.mjs +1 -1
- package/esm2020/lib/test-case-details/upload-step/upload-step.component.mjs +3 -3
- package/esm2020/lib/test-case-details/verify-url-step/verify-url-step.component.mjs +3 -3
- package/fesm2015/cqa-lib-cqa-ui.mjs +610 -110
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +597 -77
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/test-case-details/condition-step/condition-step.component.d.ts +12 -1
- package/lib/test-case-details/loop-step/loop-step.component.d.ts +70 -8
- package/lib/test-case-details/step-group/step-group.component.d.ts +8 -1
- package/lib/test-case-details/test-case-details-renderer/test-case-details-renderer.component.d.ts +6 -2
- package/lib/test-case-details/test-case-step.models.d.ts +5 -0
- package/package.json +1 -1
- package/styles.css +1 -1
|
@@ -1,16 +1,21 @@
|
|
|
1
1
|
import { Component, Input, Output, EventEmitter, ViewChild } from '@angular/core';
|
|
2
|
+
import { FormControl } from '@angular/forms';
|
|
2
3
|
import { StepTypes } from '../test-case-step.models';
|
|
3
4
|
import { STEP_ROW_ACTIONS_STYLES } from '../step-row-actions.styles';
|
|
4
5
|
import * as i0 from "@angular/core";
|
|
5
6
|
import * as i1 from "@angular/forms";
|
|
6
7
|
import * as i2 from "../../dynamic-select/dynamic-select-field.component";
|
|
7
|
-
import * as i3 from "../../
|
|
8
|
-
import * as i4 from "
|
|
9
|
-
import * as i5 from "
|
|
10
|
-
import * as i6 from "
|
|
8
|
+
import * as i3 from "../../autocomplete/autocomplete.component";
|
|
9
|
+
import * as i4 from "@angular/material/slide-toggle";
|
|
10
|
+
import * as i5 from "../../custom-input/custom-input.component";
|
|
11
|
+
import * as i6 from "../../button/button.component";
|
|
12
|
+
import * as i7 from "../test-case-details-renderer/test-case-details-renderer.component";
|
|
13
|
+
import * as i8 from "@angular/common";
|
|
14
|
+
import * as i9 from "ngx-drag-drop";
|
|
11
15
|
export class TestCaseLoopStepComponent {
|
|
12
|
-
constructor(fb) {
|
|
16
|
+
constructor(fb, cdr) {
|
|
13
17
|
this.fb = fb;
|
|
18
|
+
this.cdr = cdr;
|
|
14
19
|
this.loopType = 'for';
|
|
15
20
|
this.nestedSteps = [];
|
|
16
21
|
this.expanded = true;
|
|
@@ -26,6 +31,8 @@ export class TestCaseLoopStepComponent {
|
|
|
26
31
|
this.isLoadingDataProfiles = false;
|
|
27
32
|
/** Options for natural text actions (for while loop condition) */
|
|
28
33
|
this.naturalTextActionsOptions = [];
|
|
34
|
+
/** Function to process template variables for while condition (same as condition-step) */
|
|
35
|
+
this.setConditionTemplateVariables = () => [];
|
|
29
36
|
/** Show "Add Step Between" button above nested steps */
|
|
30
37
|
this.addStepBetweenAbove = false;
|
|
31
38
|
/** Show "Add Step Between" button below nested steps */
|
|
@@ -39,6 +46,7 @@ export class TestCaseLoopStepComponent {
|
|
|
39
46
|
this.conditionChange = new EventEmitter();
|
|
40
47
|
this.maxIterationsChange = new EventEmitter();
|
|
41
48
|
this.nestedStepChange = new EventEmitter();
|
|
49
|
+
/** Emits void when add is from nested content; emits { loopStepConfig } when empty-state add is clicked (same behaviour as add-step-between). */
|
|
42
50
|
this.addStep = new EventEmitter();
|
|
43
51
|
this.deleteStep = new EventEmitter();
|
|
44
52
|
this.duplicate = new EventEmitter();
|
|
@@ -64,6 +72,10 @@ export class TestCaseLoopStepComponent {
|
|
|
64
72
|
this.numberFn = Number;
|
|
65
73
|
/** When true, header shows inline edit form (dropdowns, inputs, Cancel/Apply) */
|
|
66
74
|
this.isEditing = false;
|
|
75
|
+
/** Selected template for while condition (same pattern as condition-step) */
|
|
76
|
+
this.selectedTemplate = null;
|
|
77
|
+
/** Template variables for while condition */
|
|
78
|
+
this.templateVariables = [];
|
|
67
79
|
/** Options for Loop Start / Loop End dropdowns (1-10) */
|
|
68
80
|
this.loopStepOptions = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
|
|
69
81
|
/** Local copy of values while editing (for Cancel revert) */
|
|
@@ -92,25 +104,179 @@ export class TestCaseLoopStepComponent {
|
|
|
92
104
|
ngOnChanges(changes) {
|
|
93
105
|
if (changes['config'] && this.config) {
|
|
94
106
|
this.selected = this.config.selected || false;
|
|
107
|
+
if (this.loopType === 'while') {
|
|
108
|
+
this.rebuildWhileConditionFromNaturalTextAction();
|
|
109
|
+
}
|
|
110
|
+
// FOR loop: when step config has forLoopTestDataId/forLoopStartIndex/forLoopEndIndex, sync form and refresh select configs so dropdowns show selected
|
|
111
|
+
if (this.loopType === 'for') {
|
|
112
|
+
this.updateForLoopSelectConfigsSelectedValues();
|
|
113
|
+
if (this.editForm)
|
|
114
|
+
this.syncForLoopEditFormFromConfig();
|
|
115
|
+
this.cdr.detectChanges();
|
|
116
|
+
}
|
|
95
117
|
}
|
|
96
118
|
if (changes['dataProfileOptions'] || changes['hasMoreDataProfiles'] || changes['isLoadingDataProfiles']) {
|
|
97
119
|
this.updateTestDataProfileSelectConfig();
|
|
120
|
+
// When data profiles load in edit mode, sync form value so dropdown shows selected (forLoopTestDataId or testDataProfile)
|
|
121
|
+
if (this.loopType === 'for' && this.editForm && this.dataProfileOptions?.length) {
|
|
122
|
+
const cfg = this.config;
|
|
123
|
+
if (cfg?.forLoopTestDataId != null || this.testDataProfile)
|
|
124
|
+
this.syncForLoopEditFormFromConfig();
|
|
125
|
+
this.cdr.detectChanges();
|
|
126
|
+
}
|
|
98
127
|
}
|
|
99
128
|
if (changes['naturalTextActionsOptions']) {
|
|
100
129
|
this.updateConditionSelectConfig();
|
|
130
|
+
// When options load, populate condition + template from naturalTextActionId (same as condition-step)
|
|
131
|
+
if (this.loopType === 'while' && this.editForm && this.naturalTextActionsOptions?.length) {
|
|
132
|
+
this.rebuildWhileConditionFromNaturalTextAction();
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
/** Derive forOptionType for FOR loop from config so dropdown shows existing selection */
|
|
137
|
+
getForOptionTypeFromConfig() {
|
|
138
|
+
if (this.loopType !== 'for')
|
|
139
|
+
return null;
|
|
140
|
+
const cfg = this.config;
|
|
141
|
+
const hasTestDataId = cfg?.forLoopTestDataId != null || cfg?.dataMapJson?.forLoopTestDataId != null || cfg?.dataMapBean?.for_loop?.testDataId != null;
|
|
142
|
+
if (hasTestDataId || this.testDataProfile)
|
|
143
|
+
return 'test-data';
|
|
144
|
+
const hasFixed = cfg?.forLoopStartIndex != null || cfg?.forLoopEndIndex != null
|
|
145
|
+
|| cfg?.dataMapJson?.forLoopStartIndex != null || cfg?.dataMapJson?.forLoopEndIndex != null
|
|
146
|
+
|| cfg?.dataMapBean?.for_loop?.startIndex != null || cfg?.dataMapBean?.for_loop?.endIndex != null;
|
|
147
|
+
if (this.startStep != null || this.endStep != null || hasFixed)
|
|
148
|
+
return 'fixed';
|
|
149
|
+
return null;
|
|
150
|
+
}
|
|
151
|
+
/** Test Data Profile dropdown value: prefer forLoopTestDataId from config (or dataMapJson/dataMapBean). Use option.id when options loaded so type matches select. */
|
|
152
|
+
getTestDataProfileFormValue() {
|
|
153
|
+
const cfg = this.config;
|
|
154
|
+
const idFromConfig = cfg?.forLoopTestDataId ?? cfg?.dataMapJson?.forLoopTestDataId ?? cfg?.dataMapBean?.for_loop?.testDataId;
|
|
155
|
+
// When we have options, resolve by id so form value matches option.id (strict equality in dynamic-select)
|
|
156
|
+
if (this.loopType === 'for' && this.dataProfileOptions?.length) {
|
|
157
|
+
if (idFromConfig != null) {
|
|
158
|
+
const match = this.dataProfileOptions.find(o => o.id === idFromConfig || String(o.id) === String(idFromConfig));
|
|
159
|
+
if (match)
|
|
160
|
+
return match.id;
|
|
161
|
+
return idFromConfig;
|
|
162
|
+
}
|
|
163
|
+
const value = this.testDataProfile ?? '';
|
|
164
|
+
if (value !== '') {
|
|
165
|
+
const match = this.dataProfileOptions.find(o => o.name === value || String(o.id) === String(value));
|
|
166
|
+
if (match)
|
|
167
|
+
return match.id;
|
|
168
|
+
}
|
|
169
|
+
return value;
|
|
101
170
|
}
|
|
171
|
+
if (idFromConfig != null)
|
|
172
|
+
return idFromConfig;
|
|
173
|
+
return (this.testDataProfile ?? '');
|
|
174
|
+
}
|
|
175
|
+
/** Loop Start dropdown value: prefer forLoopStartIndex from config (or dataMapJson/dataMapBean), else startStep */
|
|
176
|
+
getLoopStartFormValue() {
|
|
177
|
+
const cfg = this.config;
|
|
178
|
+
const v = cfg?.forLoopStartIndex ?? cfg?.dataMapJson?.forLoopStartIndex ?? cfg?.dataMapBean?.for_loop?.startIndex;
|
|
179
|
+
if (v != null)
|
|
180
|
+
return Number(v);
|
|
181
|
+
return this.startStep ?? null;
|
|
182
|
+
}
|
|
183
|
+
/** Loop End dropdown value: prefer forLoopEndIndex from config (or dataMapJson/dataMapBean), else endStep */
|
|
184
|
+
getLoopEndFormValue() {
|
|
185
|
+
const cfg = this.config;
|
|
186
|
+
const v = cfg?.forLoopEndIndex ?? cfg?.dataMapJson?.forLoopEndIndex ?? cfg?.dataMapBean?.for_loop?.endIndex;
|
|
187
|
+
if (v != null)
|
|
188
|
+
return Number(v);
|
|
189
|
+
return this.endStep ?? null;
|
|
102
190
|
}
|
|
103
191
|
buildEditForm() {
|
|
192
|
+
// For while loop: if config has naturalTextActionId and options are already loaded, use them for initial condition + template
|
|
193
|
+
let conditionValue = this.condition ?? '';
|
|
194
|
+
if (this.loopType === 'while') {
|
|
195
|
+
const populated = this.populateWhileConditionFromNaturalTextActionId();
|
|
196
|
+
if (populated?.condition != null) {
|
|
197
|
+
conditionValue = populated.condition;
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
// For FOR loop: initial form values so dropdowns show existing selection (forOptionType, forLoopTestDataId/testDataProfile, forLoopStartIndex/forLoopEndIndex)
|
|
201
|
+
const forOptionType = this.getForOptionTypeFromConfig();
|
|
202
|
+
const testDataProfileValue = this.loopType === 'for' ? this.getTestDataProfileFormValue() : (this.testDataProfile ?? '');
|
|
203
|
+
const loopStartValue = this.loopType === 'for' ? this.getLoopStartFormValue() : (this.startStep ?? null);
|
|
204
|
+
const loopEndValue = this.loopType === 'for' ? this.getLoopEndFormValue() : (this.endStep ?? null);
|
|
104
205
|
this.editForm = this.fb.group({
|
|
105
|
-
forOptionType: [
|
|
106
|
-
testDataProfile: [
|
|
107
|
-
startStep: [
|
|
108
|
-
endStep: [
|
|
109
|
-
condition: [
|
|
206
|
+
forOptionType: [forOptionType],
|
|
207
|
+
testDataProfile: [testDataProfileValue],
|
|
208
|
+
startStep: [loopStartValue],
|
|
209
|
+
endStep: [loopEndValue],
|
|
210
|
+
condition: [conditionValue],
|
|
110
211
|
maxIterations: [this.maxIterations ?? 5],
|
|
111
212
|
});
|
|
213
|
+
if (!this.templateVariablesForm) {
|
|
214
|
+
this.templateVariablesForm = this.fb.group({});
|
|
215
|
+
}
|
|
112
216
|
}
|
|
113
|
-
/**
|
|
217
|
+
/**
|
|
218
|
+
* Populate condition, selectedTemplate and template variables from config.naturalTextActionId when options are available.
|
|
219
|
+
* Returns { condition } if a match was found so buildEditForm can set initial form value.
|
|
220
|
+
*/
|
|
221
|
+
populateWhileConditionFromNaturalTextActionId() {
|
|
222
|
+
const configWithId = this.config;
|
|
223
|
+
if (!configWithId?.naturalTextActionId || !this.naturalTextActionsOptions?.length) {
|
|
224
|
+
return null;
|
|
225
|
+
}
|
|
226
|
+
const matchingOption = this.naturalTextActionsOptions.find((option) => String(option.id) === String(configWithId.naturalTextActionId));
|
|
227
|
+
if (!matchingOption) {
|
|
228
|
+
return null;
|
|
229
|
+
}
|
|
230
|
+
const naturalText = matchingOption.naturalText ?? matchingOption.name ?? matchingOption.value ?? '';
|
|
231
|
+
if (!naturalText) {
|
|
232
|
+
return null;
|
|
233
|
+
}
|
|
234
|
+
this.condition = naturalText;
|
|
235
|
+
const fullTemplate = matchingOption.template || matchingOption;
|
|
236
|
+
this.selectedTemplate = fullTemplate;
|
|
237
|
+
if (fullTemplate?.variables) {
|
|
238
|
+
this.templateVariables = this.setConditionTemplateVariables(fullTemplate) || [];
|
|
239
|
+
this.buildTemplateVariablesForm();
|
|
240
|
+
}
|
|
241
|
+
else {
|
|
242
|
+
this.templateVariables = [];
|
|
243
|
+
}
|
|
244
|
+
this.cdr.detectChanges();
|
|
245
|
+
return { condition: naturalText };
|
|
246
|
+
}
|
|
247
|
+
/** Rebuild condition + template state when naturalTextActionsOptions load or config changes (same as condition-step). */
|
|
248
|
+
rebuildWhileConditionFromNaturalTextAction() {
|
|
249
|
+
if (!this.editForm || this.loopType !== 'while') {
|
|
250
|
+
return;
|
|
251
|
+
}
|
|
252
|
+
const configWithId = this.config;
|
|
253
|
+
if (!configWithId?.naturalTextActionId || !this.naturalTextActionsOptions?.length) {
|
|
254
|
+
return;
|
|
255
|
+
}
|
|
256
|
+
const matchingOption = this.naturalTextActionsOptions.find((option) => String(option.id) === String(configWithId.naturalTextActionId));
|
|
257
|
+
if (!matchingOption) {
|
|
258
|
+
return;
|
|
259
|
+
}
|
|
260
|
+
const naturalText = matchingOption.naturalText ?? matchingOption.name ?? matchingOption.value ?? '';
|
|
261
|
+
if (naturalText) {
|
|
262
|
+
const currentValue = this.editForm.get('condition')?.value ?? '';
|
|
263
|
+
if (!currentValue || currentValue !== naturalText) {
|
|
264
|
+
this.editForm.get('condition')?.setValue(naturalText);
|
|
265
|
+
}
|
|
266
|
+
this.condition = naturalText;
|
|
267
|
+
}
|
|
268
|
+
const fullTemplate = matchingOption.template || matchingOption;
|
|
269
|
+
this.selectedTemplate = fullTemplate;
|
|
270
|
+
if (fullTemplate?.variables) {
|
|
271
|
+
this.templateVariables = this.setConditionTemplateVariables(fullTemplate) || [];
|
|
272
|
+
this.buildTemplateVariablesForm();
|
|
273
|
+
}
|
|
274
|
+
else {
|
|
275
|
+
this.templateVariables = [];
|
|
276
|
+
}
|
|
277
|
+
this.cdr.detectChanges();
|
|
278
|
+
}
|
|
279
|
+
/** Build select configs once (same option shape as default stories: id + name); FOR loop adds selectedValue from step */
|
|
114
280
|
buildSelectConfigs() {
|
|
115
281
|
const stepOptions = this.loopStepOptions.map((n) => ({
|
|
116
282
|
id: n,
|
|
@@ -128,21 +294,25 @@ export class TestCaseLoopStepComponent {
|
|
|
128
294
|
],
|
|
129
295
|
};
|
|
130
296
|
this.updateTestDataProfileSelectConfig();
|
|
297
|
+
const startVal = this.loopType === 'for' ? this.getLoopStartFormValue() : null;
|
|
298
|
+
const endVal = this.loopType === 'for' ? this.getLoopEndFormValue() : null;
|
|
131
299
|
this.startStepSelectConfig = {
|
|
132
300
|
key: 'startStep',
|
|
133
301
|
placeholder: 'Loop Start',
|
|
134
302
|
searchable: false,
|
|
135
303
|
options: stepOptions.map((o) => ({ ...o })),
|
|
304
|
+
...(startVal != null && { selectedValue: startVal }),
|
|
136
305
|
};
|
|
137
306
|
this.endStepSelectConfig = {
|
|
138
307
|
key: 'endStep',
|
|
139
308
|
placeholder: 'Loop End',
|
|
140
309
|
searchable: false,
|
|
141
310
|
options: stepOptions.map((o) => ({ ...o })),
|
|
311
|
+
...(endVal != null && { selectedValue: endVal }),
|
|
142
312
|
};
|
|
143
313
|
this.updateConditionSelectConfig();
|
|
144
314
|
}
|
|
145
|
-
/** Update condition select config with natural text actions options */
|
|
315
|
+
/** Update condition select config with natural text actions options (legacy mat-select) */
|
|
146
316
|
updateConditionSelectConfig() {
|
|
147
317
|
this.conditionSelectConfig = {
|
|
148
318
|
key: 'condition',
|
|
@@ -150,9 +320,115 @@ export class TestCaseLoopStepComponent {
|
|
|
150
320
|
searchable: true,
|
|
151
321
|
options: this.naturalTextActionsOptions || [],
|
|
152
322
|
};
|
|
323
|
+
this.updateWhileConditionSelectConfig();
|
|
324
|
+
}
|
|
325
|
+
/** While loop: only show options with stepActionType === "WHILE_LOOP" */
|
|
326
|
+
updateWhileConditionSelectConfig() {
|
|
327
|
+
const filteredOptions = (this.naturalTextActionsOptions || []).filter((option) => {
|
|
328
|
+
return option.stepActionType === 'WHILE_LOOP';
|
|
329
|
+
});
|
|
330
|
+
this.whileConditionSelectConfig = {
|
|
331
|
+
key: 'condition',
|
|
332
|
+
placeholder: 'Select condition',
|
|
333
|
+
searchable: true,
|
|
334
|
+
options: filteredOptions,
|
|
335
|
+
};
|
|
336
|
+
}
|
|
337
|
+
/** Options for the while condition autocomplete (WHILE_LOOP natural text actions only) */
|
|
338
|
+
get whileConditionAutocompleteOptions() {
|
|
339
|
+
const opts = this.whileConditionSelectConfig?.options ?? [];
|
|
340
|
+
return opts.map((option) => {
|
|
341
|
+
const naturalText = option.naturalText ?? option.name ?? option.value ?? String(option.id ?? '');
|
|
342
|
+
return {
|
|
343
|
+
value: naturalText,
|
|
344
|
+
label: naturalText,
|
|
345
|
+
template: option.template || option,
|
|
346
|
+
};
|
|
347
|
+
});
|
|
348
|
+
}
|
|
349
|
+
onConditionSelect(option) {
|
|
350
|
+
this.editForm?.get('condition')?.setValue(option.value);
|
|
351
|
+
const matchingTemplate = option.template;
|
|
352
|
+
let fullTemplate = matchingTemplate;
|
|
353
|
+
if (!fullTemplate || !fullTemplate.variables) {
|
|
354
|
+
const matchingOption = (this.naturalTextActionsOptions || []).find((opt) => opt.naturalText === option.value || opt.name === option.value || opt.value === option.value);
|
|
355
|
+
fullTemplate = matchingOption?.template || matchingOption;
|
|
356
|
+
}
|
|
357
|
+
if (fullTemplate && fullTemplate.variables) {
|
|
358
|
+
this.selectedTemplate = fullTemplate;
|
|
359
|
+
this.templateVariables = this.setConditionTemplateVariables(fullTemplate) || [];
|
|
360
|
+
this.buildTemplateVariablesForm();
|
|
361
|
+
this.cdr.detectChanges();
|
|
362
|
+
}
|
|
363
|
+
else {
|
|
364
|
+
this.selectedTemplate = null;
|
|
365
|
+
this.templateVariables = [];
|
|
366
|
+
if (!this.templateVariablesForm) {
|
|
367
|
+
this.templateVariablesForm = this.fb.group({});
|
|
368
|
+
}
|
|
369
|
+
this.cdr.detectChanges();
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
buildTemplateVariablesForm() {
|
|
373
|
+
if (!this.templateVariablesForm) {
|
|
374
|
+
this.templateVariablesForm = this.fb.group({});
|
|
375
|
+
}
|
|
376
|
+
Object.keys(this.templateVariablesForm.controls).forEach(key => {
|
|
377
|
+
this.templateVariablesForm.removeControl(key);
|
|
378
|
+
});
|
|
379
|
+
const configWithData = this.config;
|
|
380
|
+
const existingData = configWithData.templateVariablesData || {};
|
|
381
|
+
this.templateVariables.forEach(variable => {
|
|
382
|
+
let defaultValue;
|
|
383
|
+
if (existingData.hasOwnProperty(variable.name)) {
|
|
384
|
+
defaultValue = existingData[variable.name];
|
|
385
|
+
}
|
|
386
|
+
else {
|
|
387
|
+
defaultValue = variable.type === 'boolean'
|
|
388
|
+
? (variable.value === true || variable.value === 'true' || variable.value === 1)
|
|
389
|
+
: (variable.value || '');
|
|
390
|
+
}
|
|
391
|
+
if (variable.type === 'boolean' && typeof defaultValue === 'string') {
|
|
392
|
+
defaultValue = defaultValue === 'true' || defaultValue === '1';
|
|
393
|
+
}
|
|
394
|
+
this.templateVariablesForm.addControl(variable.name, new FormControl(defaultValue));
|
|
395
|
+
});
|
|
396
|
+
}
|
|
397
|
+
getSelectConfigForVariable(variable) {
|
|
398
|
+
const options = (variable.options || []).map((opt) => ({
|
|
399
|
+
id: opt,
|
|
400
|
+
value: opt,
|
|
401
|
+
name: opt,
|
|
402
|
+
label: opt
|
|
403
|
+
}));
|
|
404
|
+
return {
|
|
405
|
+
key: variable.name,
|
|
406
|
+
placeholder: `Select ${variable.label}`,
|
|
407
|
+
multiple: false,
|
|
408
|
+
searchable: false,
|
|
409
|
+
options: options,
|
|
410
|
+
onChange: (value) => {
|
|
411
|
+
if (this.templateVariablesForm && this.templateVariablesForm.get(variable.name)) {
|
|
412
|
+
this.templateVariablesForm.get(variable.name)?.setValue(value);
|
|
413
|
+
}
|
|
414
|
+
}
|
|
415
|
+
};
|
|
153
416
|
}
|
|
154
|
-
/** Update test data profile select config with current options and
|
|
417
|
+
/** Update test data profile select config with current options and selectedValue (forLoopTestDataId). Ensure selected id is in options so mat-select can display it. */
|
|
155
418
|
updateTestDataProfileSelectConfig() {
|
|
419
|
+
let options = this.convertDataProfileOptionsToSelectOptions(this.dataProfileOptions);
|
|
420
|
+
const cfg = this.config;
|
|
421
|
+
const forLoopTestDataId = cfg?.forLoopTestDataId ?? cfg?.dataMapJson?.forLoopTestDataId ?? cfg?.dataMapBean?.for_loop?.testDataId;
|
|
422
|
+
const selectedVal = this.loopType === 'for' ? this.getTestDataProfileFormValue() : undefined;
|
|
423
|
+
// If we have a preselected id but it's not in the current options list (e.g. options not loaded yet), add a placeholder option so the dropdown can show the selection
|
|
424
|
+
if (this.loopType === 'for' && forLoopTestDataId != null) {
|
|
425
|
+
const idStr = String(forLoopTestDataId);
|
|
426
|
+
const alreadyInList = options.some((o) => String(o.id ?? o.value) === idStr);
|
|
427
|
+
if (!alreadyInList) {
|
|
428
|
+
const displayName = this.testDataProfile || 'Selected profile';
|
|
429
|
+
options = [{ id: forLoopTestDataId, value: forLoopTestDataId, name: displayName, label: displayName }, ...options];
|
|
430
|
+
}
|
|
431
|
+
}
|
|
156
432
|
this.testDataProfileSelectConfig = {
|
|
157
433
|
key: 'testDataProfile',
|
|
158
434
|
placeholder: 'Testdataprofile',
|
|
@@ -160,7 +436,8 @@ export class TestCaseLoopStepComponent {
|
|
|
160
436
|
serverSearch: true,
|
|
161
437
|
hasMore: this.hasMoreDataProfiles,
|
|
162
438
|
isLoading: this.isLoadingDataProfiles,
|
|
163
|
-
options
|
|
439
|
+
options,
|
|
440
|
+
...(selectedVal !== undefined && selectedVal !== '' && { selectedValue: selectedVal }),
|
|
164
441
|
onChange: (value) => {
|
|
165
442
|
this.onDataProfileChange(value);
|
|
166
443
|
},
|
|
@@ -172,6 +449,36 @@ export class TestCaseLoopStepComponent {
|
|
|
172
449
|
},
|
|
173
450
|
};
|
|
174
451
|
}
|
|
452
|
+
/** Update Loop Start / Loop End select configs with selectedValue (forLoopStartIndex, forLoopEndIndex) so dropdowns show selected */
|
|
453
|
+
updateStartEndStepSelectConfigsSelectedValues() {
|
|
454
|
+
if (this.loopType !== 'for')
|
|
455
|
+
return;
|
|
456
|
+
const startVal = this.getLoopStartFormValue();
|
|
457
|
+
const endVal = this.getLoopEndFormValue();
|
|
458
|
+
const stepOptions = this.loopStepOptions.map((n) => ({
|
|
459
|
+
id: n,
|
|
460
|
+
value: n,
|
|
461
|
+
name: String(n),
|
|
462
|
+
label: String(n),
|
|
463
|
+
}));
|
|
464
|
+
this.startStepSelectConfig = {
|
|
465
|
+
...this.startStepSelectConfig,
|
|
466
|
+
options: stepOptions.map((o) => ({ ...o })),
|
|
467
|
+
...(startVal != null && { selectedValue: startVal }),
|
|
468
|
+
};
|
|
469
|
+
this.endStepSelectConfig = {
|
|
470
|
+
...this.endStepSelectConfig,
|
|
471
|
+
options: stepOptions.map((o) => ({ ...o })),
|
|
472
|
+
...(endVal != null && { selectedValue: endVal }),
|
|
473
|
+
};
|
|
474
|
+
}
|
|
475
|
+
/** Update all FOR loop select configs (testDataProfile, startStep, endStep) with selectedValue from step so dropdowns show selected */
|
|
476
|
+
updateForLoopSelectConfigsSelectedValues() {
|
|
477
|
+
if (this.loopType !== 'for')
|
|
478
|
+
return;
|
|
479
|
+
this.updateTestDataProfileSelectConfig();
|
|
480
|
+
this.updateStartEndStepSelectConfigsSelectedValues();
|
|
481
|
+
}
|
|
175
482
|
/** Convert DataProfileOption[] to SelectOption[] */
|
|
176
483
|
convertDataProfileOptionsToSelectOptions(options) {
|
|
177
484
|
return options.map(option => ({
|
|
@@ -208,33 +515,64 @@ export class TestCaseLoopStepComponent {
|
|
|
208
515
|
getEndLabel() {
|
|
209
516
|
return this.loopType === 'for' ? 'END FOR' : 'END WHILE';
|
|
210
517
|
}
|
|
518
|
+
/** Resolve display start index: prefer forLoopStartIndex from step (or dataMapJson/dataMapBean), else startStep, else from nested steps */
|
|
519
|
+
getDisplayStartIndex() {
|
|
520
|
+
const cfg = this.config;
|
|
521
|
+
const v = cfg?.forLoopStartIndex ?? cfg?.dataMapJson?.forLoopStartIndex ?? cfg?.dataMapBean?.for_loop?.startIndex;
|
|
522
|
+
if (v != null)
|
|
523
|
+
return Number(v);
|
|
524
|
+
if (this.startStep != null)
|
|
525
|
+
return this.startStep;
|
|
526
|
+
return this.nestedSteps[0]?.stepNumber;
|
|
527
|
+
}
|
|
528
|
+
/** Resolve display end index: prefer forLoopEndIndex from step (or dataMapJson/dataMapBean), else endStep, else from nested steps */
|
|
529
|
+
getDisplayEndIndex() {
|
|
530
|
+
const cfg = this.config;
|
|
531
|
+
const v = cfg?.forLoopEndIndex ?? cfg?.dataMapJson?.forLoopEndIndex ?? cfg?.dataMapBean?.for_loop?.endIndex;
|
|
532
|
+
if (v != null)
|
|
533
|
+
return Number(v);
|
|
534
|
+
if (this.endStep != null)
|
|
535
|
+
return this.endStep;
|
|
536
|
+
const last = this.nestedSteps[this.nestedSteps.length - 1];
|
|
537
|
+
return last?.stepNumber ?? this.nestedSteps.length;
|
|
538
|
+
}
|
|
539
|
+
/** Resolve display profile name: map forLoopTestDataId to profile name from dataProfileOptions, else use testDataProfile */
|
|
540
|
+
getDisplayTestDataProfileName() {
|
|
541
|
+
const cfg = this.config;
|
|
542
|
+
const id = cfg?.forLoopTestDataId ?? cfg?.dataMapJson?.forLoopTestDataId ?? cfg?.dataMapBean?.for_loop?.testDataId;
|
|
543
|
+
if (id != null && this.dataProfileOptions?.length) {
|
|
544
|
+
const match = this.dataProfileOptions.find(o => o.id === id || String(o.id) === String(id));
|
|
545
|
+
if (match)
|
|
546
|
+
return match.name;
|
|
547
|
+
}
|
|
548
|
+
return this.testDataProfile ?? '';
|
|
549
|
+
}
|
|
211
550
|
getStepsSummary() {
|
|
212
551
|
if (this.nestedSteps.length === 0) {
|
|
213
552
|
return '0 steps';
|
|
214
553
|
}
|
|
215
|
-
const start = this.
|
|
216
|
-
const end = this.
|
|
554
|
+
const start = this.getDisplayStartIndex() ?? 1;
|
|
555
|
+
const end = this.getDisplayEndIndex() ?? this.nestedSteps.length;
|
|
217
556
|
const count = this.nestedSteps.length;
|
|
218
557
|
return `Steps ${start}-${end} (${count} step${count !== 1 ? 's' : ''})`;
|
|
219
558
|
}
|
|
220
559
|
getStartStepLabel() {
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
return `Start: ${defaultStart}`;
|
|
560
|
+
const start = this.getDisplayStartIndex();
|
|
561
|
+
if (start != null)
|
|
562
|
+
return `Start: ${start}`;
|
|
563
|
+
return `Start: ${this.nestedSteps[0]?.stepNumber ?? 1}`;
|
|
226
564
|
}
|
|
227
565
|
getEndStepLabel() {
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
const defaultEnd = this.nestedSteps[this.nestedSteps.length - 1]?.stepNumber
|
|
566
|
+
const end = this.getDisplayEndIndex();
|
|
567
|
+
if (end != null)
|
|
568
|
+
return `End: ${end}`;
|
|
569
|
+
const defaultEnd = this.nestedSteps[this.nestedSteps.length - 1]?.stepNumber ?? this.nestedSteps.length;
|
|
232
570
|
return `End: ${defaultEnd}`;
|
|
233
571
|
}
|
|
234
572
|
getTestDataProfileLabel() {
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
573
|
+
const name = this.getDisplayTestDataProfileName();
|
|
574
|
+
if (name)
|
|
575
|
+
return `Test Data profile: ${name}`;
|
|
238
576
|
return 'Test Data profile';
|
|
239
577
|
}
|
|
240
578
|
getConditionDisplay() {
|
|
@@ -289,8 +627,25 @@ export class TestCaseLoopStepComponent {
|
|
|
289
627
|
onAddStep() {
|
|
290
628
|
this.addStep.emit();
|
|
291
629
|
}
|
|
292
|
-
|
|
293
|
-
|
|
630
|
+
/** Called from empty-state plus only; same behaviour as add-step-between (parentId = loop step id, position = parent + 1). */
|
|
631
|
+
onAddStepEmpty() {
|
|
632
|
+
this.addStep.emit({ loopStepConfig: this.config });
|
|
633
|
+
}
|
|
634
|
+
onDeleteStep(payloadOrIndex, index) {
|
|
635
|
+
const idx = typeof index === 'number' ? index : payloadOrIndex;
|
|
636
|
+
const payload = typeof index === 'number' ? payloadOrIndex : undefined;
|
|
637
|
+
this.deleteStep.emit(idx);
|
|
638
|
+
if (payload?.step)
|
|
639
|
+
this.delete.emit(payload);
|
|
640
|
+
}
|
|
641
|
+
/** Re-emit edit from nested step so the portal can open edit modal (same as root level). */
|
|
642
|
+
onNestedEdit(event) {
|
|
643
|
+
if (event?.step != null)
|
|
644
|
+
this.edit.emit(event);
|
|
645
|
+
}
|
|
646
|
+
/** Re-emit duplicate from nested step so the portal can duplicate (same as root level). */
|
|
647
|
+
onNestedDuplicate(event, step) {
|
|
648
|
+
this.duplicate.emit((event && event.step != null) ? event : { step });
|
|
294
649
|
}
|
|
295
650
|
isNormalStep(step) {
|
|
296
651
|
return 'eventType' in step;
|
|
@@ -337,6 +692,31 @@ export class TestCaseLoopStepComponent {
|
|
|
337
692
|
onOpenExternal() {
|
|
338
693
|
// Not used in loop step, but needed for recursive step groups
|
|
339
694
|
}
|
|
695
|
+
/** Sync FOR loop edit form so dropdowns show existing config (forOptionType, forLoopTestDataId/testDataProfile, forLoopStartIndex/forLoopEndIndex) */
|
|
696
|
+
syncForLoopEditFormFromConfig() {
|
|
697
|
+
if (!this.editForm || this.loopType !== 'for')
|
|
698
|
+
return;
|
|
699
|
+
const testDataVal = this.getTestDataProfileFormValue();
|
|
700
|
+
const startVal = this.getLoopStartFormValue();
|
|
701
|
+
const endVal = this.getLoopEndFormValue();
|
|
702
|
+
this.editForm.patchValue({
|
|
703
|
+
forOptionType: this.getForOptionTypeFromConfig() ?? null,
|
|
704
|
+
testDataProfile: testDataVal,
|
|
705
|
+
startStep: startVal,
|
|
706
|
+
endStep: endVal,
|
|
707
|
+
});
|
|
708
|
+
// Explicitly set each control so mat-select picks up the value (patchValue can be ignored by child in some timing cases)
|
|
709
|
+
const testDataCtrl = this.editForm.get('testDataProfile');
|
|
710
|
+
const startCtrl = this.editForm.get('startStep');
|
|
711
|
+
const endCtrl = this.editForm.get('endStep');
|
|
712
|
+
if (testDataCtrl && testDataVal !== undefined && testDataVal !== '')
|
|
713
|
+
testDataCtrl.setValue(testDataVal, { emitEvent: false });
|
|
714
|
+
if (startCtrl && startVal != null)
|
|
715
|
+
startCtrl.setValue(startVal, { emitEvent: false });
|
|
716
|
+
if (endCtrl && endVal != null)
|
|
717
|
+
endCtrl.setValue(endVal, { emitEvent: false });
|
|
718
|
+
this.cdr.detectChanges();
|
|
719
|
+
}
|
|
340
720
|
onEdit() {
|
|
341
721
|
this.editSnapshot = {
|
|
342
722
|
testDataProfile: this.testDataProfile,
|
|
@@ -344,15 +724,48 @@ export class TestCaseLoopStepComponent {
|
|
|
344
724
|
endStep: this.endStep,
|
|
345
725
|
condition: this.condition,
|
|
346
726
|
maxIterations: this.maxIterations,
|
|
727
|
+
selectedTemplate: this.selectedTemplate,
|
|
728
|
+
templateVariables: this.templateVariables ? [...this.templateVariables] : [],
|
|
729
|
+
conditionTemplateFormValue: this.templateVariablesForm ? { ...this.templateVariablesForm.value } : undefined,
|
|
347
730
|
};
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
731
|
+
// FOR loop: refresh select configs (so Test Data Profile options include selected id if needed) then sync form so dropdowns show selected
|
|
732
|
+
if (this.loopType === 'for') {
|
|
733
|
+
this.updateForLoopSelectConfigsSelectedValues();
|
|
734
|
+
this.syncForLoopEditFormFromConfig();
|
|
735
|
+
setTimeout(() => {
|
|
736
|
+
this.syncForLoopEditFormFromConfig();
|
|
737
|
+
this.cdr.detectChanges();
|
|
738
|
+
}, 0);
|
|
739
|
+
}
|
|
740
|
+
else {
|
|
741
|
+
this.editForm.patchValue({
|
|
742
|
+
testDataProfile: this.testDataProfile ?? '',
|
|
743
|
+
startStep: this.startStep ?? null,
|
|
744
|
+
endStep: this.endStep ?? null,
|
|
745
|
+
condition: this.condition ?? '',
|
|
746
|
+
maxIterations: this.maxIterations ?? 5,
|
|
747
|
+
});
|
|
748
|
+
}
|
|
749
|
+
if (this.loopType === 'while') {
|
|
750
|
+
this.editForm.patchValue({
|
|
751
|
+
condition: this.condition ?? '',
|
|
752
|
+
maxIterations: this.maxIterations ?? 5,
|
|
753
|
+
});
|
|
754
|
+
}
|
|
755
|
+
// While loop: restore selected template and template variables from config when opening edit
|
|
756
|
+
if (this.loopType === 'while' && (this.condition || this.config.templateVariablesData)) {
|
|
757
|
+
const configWithData = this.config;
|
|
758
|
+
const conditionStr = this.condition || configWithData.condition || '';
|
|
759
|
+
if (conditionStr && (this.naturalTextActionsOptions || []).length > 0) {
|
|
760
|
+
const matchingOption = this.naturalTextActionsOptions.find((opt) => (opt.naturalText || opt.name || opt.value) === conditionStr);
|
|
761
|
+
const fullTemplate = matchingOption?.template || matchingOption;
|
|
762
|
+
if (fullTemplate && fullTemplate.variables) {
|
|
763
|
+
this.selectedTemplate = fullTemplate;
|
|
764
|
+
this.templateVariables = this.setConditionTemplateVariables(fullTemplate) || [];
|
|
765
|
+
this.buildTemplateVariablesForm();
|
|
766
|
+
}
|
|
767
|
+
}
|
|
768
|
+
}
|
|
356
769
|
this.isEditing = true;
|
|
357
770
|
this.edit.emit();
|
|
358
771
|
}
|
|
@@ -384,7 +797,22 @@ export class TestCaseLoopStepComponent {
|
|
|
384
797
|
else {
|
|
385
798
|
this.testDataProfile = testDataProfileValue;
|
|
386
799
|
}
|
|
387
|
-
//
|
|
800
|
+
// While loop: build templateVariablesData and naturalTextActionId from selected template (same as condition-step)
|
|
801
|
+
let templateVariablesData;
|
|
802
|
+
let naturalTextActionId;
|
|
803
|
+
if (this.loopType === 'while' && this.templateVariablesForm && this.templateVariables?.length) {
|
|
804
|
+
templateVariablesData = {};
|
|
805
|
+
this.templateVariables.forEach((variable) => {
|
|
806
|
+
const value = this.templateVariablesForm.get(variable.name)?.value;
|
|
807
|
+
if (value !== undefined && value !== null && value !== '') {
|
|
808
|
+
templateVariablesData[variable.name] = value;
|
|
809
|
+
}
|
|
810
|
+
});
|
|
811
|
+
if (this.selectedTemplate?.id != null) {
|
|
812
|
+
naturalTextActionId = this.selectedTemplate.id;
|
|
813
|
+
}
|
|
814
|
+
}
|
|
815
|
+
// Update config with new values (include testDataProfileId, templateVariablesData, naturalTextActionId for while; forLoopTestDataId, forLoopStartIndex/forLoopEndIndex for for)
|
|
388
816
|
const updatedConfig = {
|
|
389
817
|
...this.config,
|
|
390
818
|
testDataProfile: this.testDataProfile,
|
|
@@ -393,6 +821,11 @@ export class TestCaseLoopStepComponent {
|
|
|
393
821
|
endStep: this.endStep,
|
|
394
822
|
condition: this.condition,
|
|
395
823
|
maxIterations: this.maxIterations,
|
|
824
|
+
...(this.loopType === 'while' && Object.keys(templateVariablesData || {}).length > 0 && { templateVariablesData }),
|
|
825
|
+
...(this.loopType === 'while' && naturalTextActionId != null && { naturalTextActionId }),
|
|
826
|
+
...(this.loopType === 'for' && testDataProfileId != null && { forLoopTestDataId: Number(testDataProfileId) }),
|
|
827
|
+
...(this.loopType === 'for' && this.startStep != null && { forLoopStartIndex: this.startStep }),
|
|
828
|
+
...(this.loopType === 'for' && this.endStep != null && { forLoopEndIndex: this.endStep }),
|
|
396
829
|
};
|
|
397
830
|
// Emit individual change events
|
|
398
831
|
this.testDataProfileChange.emit(this.testDataProfile);
|
|
@@ -413,6 +846,11 @@ export class TestCaseLoopStepComponent {
|
|
|
413
846
|
this.endStep = this.editSnapshot.endStep;
|
|
414
847
|
this.condition = this.editSnapshot.condition;
|
|
415
848
|
this.maxIterations = this.editSnapshot.maxIterations;
|
|
849
|
+
this.selectedTemplate = this.editSnapshot.selectedTemplate;
|
|
850
|
+
this.templateVariables = this.editSnapshot.templateVariables || [];
|
|
851
|
+
if (this.templateVariablesForm && this.editSnapshot.conditionTemplateFormValue) {
|
|
852
|
+
this.templateVariablesForm.patchValue(this.editSnapshot.conditionTemplateFormValue);
|
|
853
|
+
}
|
|
416
854
|
this.isEditing = false;
|
|
417
855
|
}
|
|
418
856
|
onEditInDepth() {
|
|
@@ -454,12 +892,34 @@ export class TestCaseLoopStepComponent {
|
|
|
454
892
|
}
|
|
455
893
|
}
|
|
456
894
|
}
|
|
457
|
-
TestCaseLoopStepComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TestCaseLoopStepComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component });
|
|
458
|
-
TestCaseLoopStepComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: TestCaseLoopStepComponent, selector: "cqa-test-case-loop-step", inputs: { config: "config", id: "id", loopType: "loopType", stepNumber: "stepNumber", condition: "condition", maxIterations: "maxIterations", testDataProfile: "testDataProfile", startStep: "startStep", endStep: "endStep", nestedSteps: "nestedSteps", expanded: "expanded", isNested: "isNested", isInsideLoop: "isInsideLoop", isReorder: "isReorder", selected: "selected", dataProfileOptions: "dataProfileOptions", hasMoreDataProfiles: "hasMoreDataProfiles", isLoadingDataProfiles: "isLoadingDataProfiles", naturalTextActionsOptions: "naturalTextActionsOptions", addStepBetweenAbove: "addStepBetweenAbove", addStepBetweenBelow: "addStepBetweenBelow", addStepBetween: "addStepBetween" }, outputs: { toggleExpanded: "toggleExpanded", testDataProfileChange: "testDataProfileChange", startStepChange: "startStepChange", endStepChange: "endStepChange", conditionChange: "conditionChange", maxIterationsChange: "maxIterationsChange", nestedStepChange: "nestedStepChange", addStep: "addStep", deleteStep: "deleteStep", duplicate: "duplicate", delete: "delete", moreOptions: "moreOptions", viewDetails: "viewDetails", editInDepth: "editInDepth", edit: "edit", dndDropInZone: "dndDropInZone", loadMoreDataProfiles: "loadMoreDataProfiles", searchDataProfiles: "searchDataProfiles", stepUpdate: "stepUpdate", clickAction: "clickAction", addStepBetweenClick: "addStepBetweenClick", selectionChange: "selectionChange" }, host: { classAttribute: "cqa-ui-root" }, viewQueries: [{ propertyName: "viewDetailsTrigger", first: true, predicate: ["viewDetailsTrigger"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col\">\n <!-- Inline Edit Mode: Frame 2147224910 - flex row, 1056\u00D736, gap 12px, flex-none -->\n <div *ngIf=\"isEditing\" class=\"cqa-py-2.5 cqa-px-4 cqa-flex cqa-items-center cqa-gap-3 cqa-justify-between\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-flex-grow\">\n <!-- Loop Type Label -->\n <div class=\"cqa-font-semibold cqa-text-[#1C398E] cqa-text-[12px] cqa-leading-[15px]\">\n {{ getLoopTypeLabel() }}\n </div>\n\n <!-- FOR LOOP inline edit: cqa-dynamic-select (same usage as default stories) -->\n <ng-container *ngIf=\"loopType === 'for'\">\n <!-- First select: width 216.25px, height 36px, gap 4px, opacity 1 -->\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"forOptionTypeSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"testDataProfileSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"startStepSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"endStepSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n </ng-container>\n\n <!-- WHILE LOOP inline edit -->\n <ng-container *ngIf=\"isWhileLoop()\">\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"conditionSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n </ng-container>\n\n <!-- Edit In depth link -->\n <a href=\"#\" (click)=\"onEditInDepth(); $event.preventDefault()\"\n class=\"cqa-text-[#3F43EE] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-[2px] cqa-no-underline\">\n Edit In depth\n <svg width=\"8\" height=\"8\" viewBox=\"0 0 8 8\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M2.03809 6.74329L2.62809 7.33329L5.96142 3.99996L2.62809 0.666626L2.03809 1.25663L4.78142 3.99996L2.03809 6.74329Z\"\n fill=\"#3F43EE\" />\n </svg>\n </a>\n </div>\n\n <div class=\"cqa-flex cqa-items-center cqa-gap-1.5\">\n <!-- Cancel / Apply -->\n <cqa-button variant=\"outlined\" btnSize=\"lg\" [customClass]=\"'cqa-text-[14px] cqa-py-[9px]'\" [text]=\"'Cancel'\"\n (clicked)=\"onEditCancel()\"></cqa-button>\n <cqa-button variant=\"filled\" btnSize=\"lg\" [customClass]=\"'cqa-text-[14px] cqa-py-[9px]'\" [text]=\"'Apply'\"\n (clicked)=\"onEditApply()\"></cqa-button>\n </div>\n </div>\n\n <!-- Loop Header (normal view when not editing) -->\n <div *ngIf=\"!isEditing\"\n [class]=\"'step-row cqa-flex cqa-items-center cqa-gap-3 cqa-py-2 ' + (isInsideLoop ? 'cqa-pl-10 cqa-pr-4' : 'cqa-px-4')\">\n <div class=\"cqa-inline-flex cqa-items-center\">\n <!-- Drag Handle Icon (when isReorder is true) -->\n <div *ngIf=\"isReorder\" class=\"cqa-mr-2 cqa-cursor-move cqa-text-[#6B7280] hover:cqa-text-[#111827]\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"3\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"3\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"3\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n </svg>\n </div>\n <!-- Checkbox (when isReorder is false) -->\n <label *ngIf=\"!isReorder\" class=\"cqa-flex cqa-items-center cqa-cursor-pointer cqa-relative cqa-mr-2\">\n <input type=\"checkbox\"\n [ngModel]=\"selected\"\n (ngModelChange)=\"onSelectionChange($event)\"\n class=\"cqa-h-4 cqa-w-4 cqa-cursor-pointer cqa-transition-all cqa-appearance-none cqa-rounded shadow hover:cqa-shadow-md cqa-border cqa-border-solid cqa-border-slate-300 cqa-flex-shrink-0\"\n [class.cqa-bg-[#3F43EE]]=\"selected\"\n [class.cqa-border-[#3F43EE]]=\"selected\"\n id=\"check-loop-step\" />\n <span class=\"cqa-absolute cqa-text-white cqa-top-1/2 cqa-left-1/2 cqa--translate-x-1/2 cqa--translate-y-1/2 cqa-pointer-events-none cqa-flex cqa-items-center cqa-justify-center\"\n [class.cqa-opacity-0]=\"!selected\"\n [class.cqa-opacity-100]=\"selected\">\n <svg width=\"12\" height=\"13\" viewBox=\"0 0 12 13\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M10 3.125L4.5 8.625L2 6.125\" stroke=\"white\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </span>\n </label>\n </div>\n <span class=\"cqa-text-[#6B7280] cqa-text-[14px] cqa-leading-[18px] cqa-min-w-[32px]\">{{ stepNumber }}</span>\n <!-- Expand/Collapse Icon -->\n <button type=\"button\" (click)=\"onToggleExpanded()\" class=\"cqa-p-0\">\n <svg [class.cqa-rotate-180]=\"!expanded\" class=\"cqa-transition-transform\" width=\"16\" height=\"16\"\n viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M12 10L8 6L4 10\" stroke=\"#6B7280\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </button>\n\n <!-- Loop Icon -->\n <div *ngIf=\"loopType === 'for'\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M11.334 1.33325L14.0007 3.99992L11.334 6.66659\" stroke=\"#1C398E\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M2 7.33333V6.66667C2 5.95942 2.28095 5.28115 2.78105 4.78105C3.28115 4.28095 3.95942 4 4.66667 4H14\"\n stroke=\"#1C398E\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M4.66667 14.6666L2 11.9999L4.66667 9.33325\" stroke=\"#1C398E\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M14 8.66675V9.33341C14 10.0407 13.719 10.7189 13.219 11.219C12.7189 11.7191 12.0406 12.0001 11.3333 12.0001H2\"\n stroke=\"#1C398E\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </div>\n <div *ngIf=\"loopType === 'while'\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M11.334 1.33325L14.0007 3.99992L11.334 6.66659\" stroke=\"#59168B\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M2 7.33333V6.66667C2 5.95942 2.28095 5.28115 2.78105 4.78105C3.28115 4.28095 3.95942 4 4.66667 4H14\"\n stroke=\"#59168B\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M4.66667 14.6666L2 11.9999L4.66667 9.33325\" stroke=\"#59168B\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M14 8.66675V9.33341C14 10.0407 13.719 10.7189 13.219 11.219C12.7189 11.7191 12.0406 12.0001 11.3333 12.0001H2\"\n stroke=\"#59168B\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </div>\n\n <!-- Loop Type Label -->\n <span class=\"cqa-font-semibold cqa-text-[#1C398E] cqa-text-[12px] cqa-leading-none\">\n {{ getLoopTypeLabel() }}\n </span>\n\n <!-- For Loop Controls -->\n <div *ngIf=\"loopType === 'for'\" class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-ml-2\">\n <!-- Test Data Profile Button -->\n <button type=\"button\" [class.cqa-bg-primary-100]=\"testDataProfile\" [class.cqa-text-primary]=\"testDataProfile\"\n class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n {{ getTestDataProfileLabel() }}\n </button>\n\n <!-- Start Button -->\n <button type=\"button\" [class.cqa-bg-primary-100]=\"startStep\" [class.cqa-text-primary]=\"startStep\"\n class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n {{ getStartStepLabel() }}\n </button>\n\n <!-- End Button -->\n <button type=\"button\" [class.cqa-bg-primary-100]=\"endStep\" [class.cqa-text-primary]=\"endStep\"\n class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n {{ getEndStepLabel() }}\n </button>\n </div>\n\n <!-- While Loop Condition (only show if condition exists) -->\n <div *ngIf=\"isWhileLoop() && condition\" class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-ml-2 cqa-flex-1\">\n <span class=\"cqa-text-gray-700 cqa-text-sm\">\n While\n </span>\n <span class=\"cqa-text-gray-900 cqa-text-sm\" [innerHTML]=\"getConditionDisplay()\"></span>\n </div>\n\n <!-- Steps Summary -->\n <div\n class=\"cqa-ml-auto cqa-border cqa-border-solid cqa-border-[#E5E5E5] cqa-rounded-lg cqa-py-0.5 cqa-px-2 cqa-text-[#0A0A0A] cqa-text-[12px] cqa-leading-[15px]\">\n {{ getStepsSummary() }}\n </div>\n\n <!-- View Details Link (show only if description is not empty) -->\n <a *ngIf=\"config.description && config.description.trim() !== ''\" #viewDetailsTrigger href=\"#\" (click)=\"onViewDetails($event)\" class=\"cqa-text-[#3F43EE] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-2 cqa-no-underline cqa-ml-auto\">View Details<svg width=\"8\" height=\"8\" viewBox=\"0 0 8 8\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M2.03809 6.74329L2.62809 7.33329L5.96142 3.99996L2.62809 0.666626L2.03809 1.25663L4.78142 3.99996L2.03809 6.74329Z\" fill=\"#3F43EE\"/></svg></a>\n\n <!-- Action Icons: Edit, Link, Duplicate, Delete (show on hover) -->\n <div class=\"step-actions cqa-flex cqa-items-center cqa-gap-3 cqa-px-[7px]\">\n <button type=\"button\" (click)=\"onEdit(); $event.stopPropagation()\" title=\"Edit\"\n class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M7 11.6666H12.25\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path\n d=\"M9.55208 2.1128C9.7843 1.88058 10.0993 1.75012 10.4277 1.75012C10.7561 1.75012 11.071 1.88058 11.3033 2.1128C11.5355 2.34502 11.6659 2.65998 11.6659 2.98838C11.6659 3.31679 11.5355 3.63175 11.3033 3.86397L4.29742 10.8704C4.15864 11.0092 3.9871 11.1107 3.79867 11.1656L2.12333 11.6544C2.07314 11.669 2.01993 11.6699 1.96928 11.6569C1.91863 11.6439 1.8724 11.6176 1.83543 11.5806C1.79846 11.5437 1.7721 11.4974 1.75913 11.4468C1.74615 11.3961 1.74703 11.3429 1.76167 11.2927L2.2505 9.61738C2.30546 9.42916 2.40698 9.25783 2.54567 9.11922L9.55208 2.1128Z\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </button>\n <button type=\"button\" (click)=\"onLink(); $event.stopPropagation()\" title=\"Link\"\n class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M5.00065 9.91671H3.66732C2.78326 9.91671 1.93542 9.60942 1.3103 9.06244C0.685174 8.51545 0.333984 7.77359 0.333984 7.00004C0.333984 6.22649 0.685174 5.48463 1.3103 4.93765C1.93542 4.39066 2.78326 4.08337 3.66732 4.08337H5.00065\"\n stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M9 4.08337H10.3333C11.2174 4.08337 12.0652 4.39066 12.6904 4.93765C13.3155 5.48463 13.6667 6.22649 13.6667 7.00004C13.6667 7.77359 13.3155 8.51545 12.6904 9.06244C12.0652 9.60942 11.2174 9.91671 10.3333 9.91671H9\"\n stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M4.33398 7H9.66732\" stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </button>\n <button type=\"button\" (click)=\"onDuplicate(); $event.stopPropagation()\" title=\"Duplicate\"\n class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M11.666 4.66663H5.83268C5.18835 4.66663 4.66602 5.18896 4.66602 5.83329V11.6666C4.66602 12.311 5.18835 12.8333 5.83268 12.8333H11.666C12.3103 12.8333 12.8327 12.311 12.8327 11.6666V5.83329C12.8327 5.18896 12.3103 4.66663 11.666 4.66663Z\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M2.33268 9.33329C1.69102 9.33329 1.16602 8.80829 1.16602 8.16663V2.33329C1.16602 1.69163 1.69102 1.16663 2.33268 1.16663H8.16602C8.80768 1.16663 9.33268 1.69163 9.33268 2.33329\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </button>\n <button type=\"button\" (click)=\"onDelete(); $event.stopPropagation()\" title=\"Delete\"\n class=\"cqa-p-0 cqa-text-[#ff6467] hover:cqa-text-[#C63535]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M1.75 3.5H12.25\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path\n d=\"M11.0827 3.5V11.6667C11.0827 12.25 10.4993 12.8333 9.91602 12.8333H4.08268C3.49935 12.8333 2.91602 12.25 2.91602 11.6667V3.5\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M4.66602 3.49996V2.33329C4.66602 1.74996 5.24935 1.16663 5.83268 1.16663H8.16602C8.74935 1.16663 9.33268 1.74996 9.33268 2.33329V3.49996\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M5.83398 6.41663V9.91663\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path d=\"M8.16602 6.41663V9.91663\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </button>\n </div>\n </div>\n\n <!-- Expanded Content with Nested Steps (renderer dispatches by step type, n-level nesting) -->\n <div *ngIf=\"expanded\" class=\"cqa-flex cqa-flex-col\">\n <div *ngIf=\"!isReorder\" class=\"cqa-flex cqa-flex-col\">\n <cqa-test-case-details-renderer *ngFor=\"let step of nestedSteps; let i = index\" [step]=\"step\" [index]=\"i\"\n [selected]=\"$any(step).selected\"\n [isNested]=\"true\" [isInsideLoop]=\"true\" [isReorder]=\"isReorder\"\n [addStepBetween]=\"addStepBetween\"\n [dataProfileOptions]=\"dataProfileOptions\" [hasMoreDataProfiles]=\"hasMoreDataProfiles\" [isLoadingDataProfiles]=\"isLoadingDataProfiles\"\n [naturalTextActionsOptions]=\"naturalTextActionsOptions\"\n (nestedStepChange)=\"onNestedStepChange($event.step, $event.index)\"\n (addStep)=\"onAddStep()\" (deleteStep)=\"onDeleteStep($event)\"\n (toggleExpanded)=\"onNestedToggleExpanded($event, step, i)\"\n (groupNameChange)=\"$any(step).groupName = $event; onNestedStepChange(step, i)\"\n (descriptionChange)=\"$any(step).description = $event; onNestedStepChange(step, i)\"\n (reusableChange)=\"$any(step).reusable = $event; onNestedStepChange(step, i)\"\n (openExternal)=\"onNestedStepChange(step, i)\" (edit)=\"onNestedStepChange(step, i)\"\n (link)=\"onNestedStepChange(step, i)\" (duplicate)=\"onNestedStepChange(step, i)\" (delete)=\"onDeleteStep(i)\"\n (viewDetails)=\"viewDetails.emit($event)\"\n (conditionChange)=\"$any(step).condition = $event; onNestedStepChange(step, i)\"\n (branchStepChange)=\"onNestedStepChange(step, i)\" (addBranch)=\"onNestedConditionAddBranch($any(step), i)\"\n (deleteBranch)=\"onNestedConditionDeleteBranch($any(step), $event, i)\"\n (testDataProfileChange)=\"$any(step).testDataProfile = $event; onNestedStepChange(step, i)\"\n (startStepChange)=\"$any(step).startStep = $event; onNestedStepChange(step, i)\"\n (endStepChange)=\"$any(step).endStep = $event; onNestedStepChange(step, i)\"\n (maxIterationsChange)=\"$any(step).maxIterations = $event; onNestedStepChange(step, i)\"\n (eventTypeChange)=\"$any(step).eventType = $event; onNestedStepChange(step, i)\"\n (parameterChange)=\"onNestedStepChange(step, i)\"\n (selectionChange)=\"$any(step).selected = $event; onNestedStepChange(step, i)\"\n (loadMoreDataProfiles)=\"loadMoreDataProfiles.emit($event)\"\n (searchDataProfiles)=\"searchDataProfiles.emit($event)\"\n (dndDropInZone)=\"dndDropInZone.emit($event)\"\n (clickAction)=\"clickAction.emit($event)\"\n (editInDepth)=\"editInDepth.emit($event)\"\n (addStepBetweenClick)=\"addStepBetweenClick.emit($event)\">\n </cqa-test-case-details-renderer>\n </div>\n <div *ngIf=\"isReorder\" class=\"cqa-flex cqa-flex-col nested-step-drop-list\"\n [dndDropzone]=\"['step']\"\n dndEffectAllowed=\"move\"\n dndDragoverClass=\"dndDragover\"\n (dndDrop)=\"onDndDrop($event)\">\n <div dndPlaceholderRef class=\"step-drag-placeholder-nested cqa-my-1 cqa-min-h-[50px] cqa-border-2 cqa-border-dashed cqa-border-[#3F43EE] cqa-rounded cqa-bg-[rgba(63,67,238,0.08)] cqa-flex cqa-items-center cqa-justify-center cqa-text-[#3F43EE] cqa-text-xs\">Drop here</div>\n <div *ngFor=\"let step of nestedSteps; let i = index\" class=\"nested-step-drag-item\"\n [dndDraggable]=\"step\"\n dndEffectAllowed=\"move\"\n dndType=\"step\">\n <cqa-test-case-details-renderer [step]=\"step\" [index]=\"i\"\n [selected]=\"$any(step).selected\"\n [isNested]=\"true\" [isInsideLoop]=\"true\" [isReorder]=\"isReorder\"\n [addStepBetweenAbove]=\"addStepBetweenAbove\"\n [addStepBetweenBelow]=\"addStepBetweenBelow\"\n [addStepBetween]=\"addStepBetween\"\n [dataProfileOptions]=\"dataProfileOptions\" [hasMoreDataProfiles]=\"hasMoreDataProfiles\" [isLoadingDataProfiles]=\"isLoadingDataProfiles\"\n [naturalTextActionsOptions]=\"naturalTextActionsOptions\"\n (nestedStepChange)=\"onNestedStepChange($event.step, $event.index)\"\n (addStep)=\"onAddStep()\" (deleteStep)=\"onDeleteStep($event)\"\n (toggleExpanded)=\"onNestedToggleExpanded($event, step, i)\"\n (groupNameChange)=\"$any(step).groupName = $event; onNestedStepChange(step, i)\"\n (descriptionChange)=\"$any(step).description = $event; onNestedStepChange(step, i)\"\n (reusableChange)=\"$any(step).reusable = $event; onNestedStepChange(step, i)\"\n (openExternal)=\"onNestedStepChange(step, i)\" (edit)=\"onNestedStepChange(step, i)\"\n (link)=\"onNestedStepChange(step, i)\" (duplicate)=\"onNestedStepChange(step, i)\" (delete)=\"onDeleteStep(i)\"\n (viewDetails)=\"viewDetails.emit($event)\"\n (conditionChange)=\"$any(step).condition = $event; onNestedStepChange(step, i)\"\n (branchStepChange)=\"onNestedStepChange(step, i)\" (addBranch)=\"onNestedConditionAddBranch($any(step), i)\"\n (deleteBranch)=\"onNestedConditionDeleteBranch($any(step), $event, i)\"\n (testDataProfileChange)=\"$any(step).testDataProfile = $event; onNestedStepChange(step, i)\"\n (startStepChange)=\"$any(step).startStep = $event; onNestedStepChange(step, i)\"\n (endStepChange)=\"$any(step).endStep = $event; onNestedStepChange(step, i)\"\n (maxIterationsChange)=\"$any(step).maxIterations = $event; onNestedStepChange(step, i)\"\n (eventTypeChange)=\"$any(step).eventType = $event; onNestedStepChange(step, i)\"\n (parameterChange)=\"onNestedStepChange(step, i)\"\n (selectionChange)=\"$any(step).selected = $event; onNestedStepChange(step, i)\"\n (loadMoreDataProfiles)=\"loadMoreDataProfiles.emit($event)\"\n (searchDataProfiles)=\"searchDataProfiles.emit($event)\"\n (dndDropInZone)=\"dndDropInZone.emit($event)\"\n (clickAction)=\"clickAction.emit($event)\"\n (editInDepth)=\"editInDepth.emit($event)\"\n (addStepBetweenClick)=\"addStepBetweenClick.emit($event)\">\n </cqa-test-case-details-renderer>\n </div>\n </div>\n\n <!-- END Marker -->\n <div\n [class]=\"'cqa-pl-4 cqa-py-1 cqa-bg-[#DBEAFE80] cqa-text-[#1C398E] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium'\"\n style=\"border-top: 1px solid #BEDBFF;\">\n {{ getEndLabel() }}\n </div>\n </div>\n</div>", styles: [".step-actions{opacity:0;transition:opacity .15s ease}.step-row:hover .step-actions{opacity:1}\n"], components: [{ type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }, { type: i3.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }, { type: i4.TestCaseDetailsRendererComponent, selector: "cqa-test-case-details-renderer", inputs: ["step", "index", "isNested", "isInsideLoop", "branch", "isReorder", "selected", "addStepBetweenAbove", "addStepBetweenBelow", "addStepBetween", "dataProfileOptions", "hasMoreDataProfiles", "isLoadingDataProfiles", "naturalTextActionsOptions", "setConditionTemplateVariables"], outputs: ["nestedStepChange", "addStep", "deleteStep", "toggleExpanded", "groupNameChange", "descriptionChange", "reusableChange", "openExternal", "edit", "link", "duplicate", "delete", "viewDetails", "selectionChange", "conditionChange", "branchStepChange", "addStepForBranch", "deleteStepWithBranch", "addBranch", "addElse", "deleteBranch", "testDataProfileChange", "startStepChange", "endStepChange", "maxIterationsChange", "eventTypeChange", "parameterChange", "clickAction", "dndDropInZone", "loadMoreDataProfiles", "searchDataProfiles", "stepUpdate", "addStepBetweenClick", "editInDepth"] }], directives: [{ type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i1.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i6.DndDropzoneDirective, selector: "[dndDropzone]", inputs: ["dndDropzone", "dndEffectAllowed", "dndAllowExternal", "dndHorizontal", "dndDragoverClass", "dndDropzoneDisabledClass", "dndDisableIf", "dndDisableDropIf"], outputs: ["dndDragover", "dndDrop"] }, { type: i6.DndPlaceholderRefDirective, selector: "[dndPlaceholderRef]" }, { type: i6.DndDraggableDirective, selector: "[dndDraggable]", inputs: ["dndDraggable", "dndEffectAllowed", "dndType", "dndDraggingClass", "dndDraggingSourceClass", "dndDraggableDisabledClass", "dndDragImageOffsetFunction", "dndDisableIf", "dndDisableDragIf"], outputs: ["dndStart", "dndDrag", "dndEnd", "dndMoved", "dndCopied", "dndLinked", "dndCanceled"] }] });
|
|
895
|
+
TestCaseLoopStepComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TestCaseLoopStepComponent, deps: [{ token: i1.FormBuilder }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
896
|
+
TestCaseLoopStepComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: TestCaseLoopStepComponent, selector: "cqa-test-case-loop-step", inputs: { config: "config", id: "id", loopType: "loopType", stepNumber: "stepNumber", condition: "condition", maxIterations: "maxIterations", testDataProfile: "testDataProfile", startStep: "startStep", endStep: "endStep", nestedSteps: "nestedSteps", expanded: "expanded", isNested: "isNested", isInsideLoop: "isInsideLoop", isReorder: "isReorder", selected: "selected", dataProfileOptions: "dataProfileOptions", hasMoreDataProfiles: "hasMoreDataProfiles", isLoadingDataProfiles: "isLoadingDataProfiles", naturalTextActionsOptions: "naturalTextActionsOptions", setConditionTemplateVariables: "setConditionTemplateVariables", addStepBetweenAbove: "addStepBetweenAbove", addStepBetweenBelow: "addStepBetweenBelow", addStepBetween: "addStepBetween" }, outputs: { toggleExpanded: "toggleExpanded", testDataProfileChange: "testDataProfileChange", startStepChange: "startStepChange", endStepChange: "endStepChange", conditionChange: "conditionChange", maxIterationsChange: "maxIterationsChange", nestedStepChange: "nestedStepChange", addStep: "addStep", deleteStep: "deleteStep", duplicate: "duplicate", delete: "delete", moreOptions: "moreOptions", viewDetails: "viewDetails", editInDepth: "editInDepth", edit: "edit", dndDropInZone: "dndDropInZone", loadMoreDataProfiles: "loadMoreDataProfiles", searchDataProfiles: "searchDataProfiles", stepUpdate: "stepUpdate", clickAction: "clickAction", addStepBetweenClick: "addStepBetweenClick", selectionChange: "selectionChange" }, host: { classAttribute: "cqa-ui-root" }, viewQueries: [{ propertyName: "viewDetailsTrigger", first: true, predicate: ["viewDetailsTrigger"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"loop-step-row cqa-flex cqa-flex-col\">\n <!-- Inline Edit Mode: Frame 2147224910 - flex row, 1056\u00D736, gap 12px, flex-none -->\n <div *ngIf=\"isEditing\" class=\"cqa-py-2.5 cqa-px-4 cqa-flex cqa-flex-col cqa-gap-3\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-justify-between\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-flex-grow\">\n <!-- Loop Type Label -->\n <div class=\"cqa-font-semibold cqa-text-[#1C398E] cqa-text-[12px] cqa-leading-[15px]\">\n {{ getLoopTypeLabel() }}\n </div>\n\n <!-- FOR LOOP inline edit: cqa-dynamic-select (same usage as default stories) -->\n <ng-container *ngIf=\"loopType === 'for'\">\n <!-- First select: width 216.25px, height 36px, gap 4px, opacity 1 -->\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"forOptionTypeSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"testDataProfileSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"startStepSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"endStepSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n </ng-container>\n\n <!-- WHILE LOOP inline edit: cqa-autocomplete + template variables (same as condition-step) -->\n <ng-container *ngIf=\"isWhileLoop()\">\n <cqa-autocomplete\n *ngIf=\"editForm\"\n [options]=\"whileConditionAutocompleteOptions\"\n [value]=\"editForm.get('condition')?.value ?? ''\"\n (valueChange)=\"onEditFormFieldChange('condition', $event)\"\n (optionSelect)=\"onConditionSelect($event)\"\n placeholder=\"Select condition\"\n [fullWidth]=\"true\"\n class=\"cqa-w-full cqa-max-w-[216px]\">\n </cqa-autocomplete>\n <!-- Template variables (shown when template is selected, same as condition-step) -->\n <div *ngIf=\"selectedTemplate && templateVariables && templateVariables.length > 0\" class=\"cqa-flex cqa-flex-row cqa-flex-wrap cqa-gap-3\">\n <ng-container *ngFor=\"let variable of templateVariables\">\n <ng-container *ngIf=\"variable.type === 'boolean'\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-gray-700\">{{ variable.label }}</label>\n <mat-slide-toggle\n [checked]=\"templateVariablesForm.get(variable.name)?.value || variable.value || false\"\n (change)=\"templateVariablesForm.get(variable.name)?.setValue($event.checked)\"\n color=\"primary\">\n </mat-slide-toggle>\n </div>\n </ng-container>\n <ng-container *ngIf=\"variable.type !== 'boolean' && variable.name !== 'custom_code'\">\n <ng-container *ngIf=\"variable.name === 'type' || variable.name === 'scrollTo' || variable.options\">\n <div class=\"cqa-flex cqa-flex-col\" style=\"min-width: 150px;\">\n <cqa-dynamic-select\n [form]=\"templateVariablesForm\"\n [config]=\"getSelectConfigForVariable(variable)\">\n </cqa-dynamic-select>\n </div>\n </ng-container>\n <ng-container *ngIf=\"variable.name !== 'type' && variable.name !== 'scrollTo' && !variable.options\">\n <div class=\"cqa-flex cqa-flex-col\" style=\"min-width: 150px;\">\n <cqa-custom-input\n [placeholder]=\"'Enter ' + variable.label\"\n [value]=\"templateVariablesForm.get(variable.name)?.value || variable.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"templateVariablesForm.get(variable.name)?.setValue($event)\">\n </cqa-custom-input>\n </div>\n </ng-container>\n </ng-container>\n </ng-container>\n </div>\n </ng-container>\n\n <!-- Edit In depth link -->\n <a href=\"#\" (click)=\"onEditInDepth(); $event.preventDefault()\"\n class=\"cqa-text-[#3F43EE] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-[2px] cqa-no-underline\">\n Edit In depth\n <svg width=\"8\" height=\"8\" viewBox=\"0 0 8 8\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M2.03809 6.74329L2.62809 7.33329L5.96142 3.99996L2.62809 0.666626L2.03809 1.25663L4.78142 3.99996L2.03809 6.74329Z\"\n fill=\"#3F43EE\" />\n </svg>\n </a>\n </div>\n\n <div class=\"cqa-flex cqa-items-center cqa-gap-1.5\">\n <!-- Cancel / Apply -->\n <cqa-button variant=\"outlined\" btnSize=\"lg\" [customClass]=\"'cqa-text-[14px] cqa-py-[9px]'\" [text]=\"'Cancel'\"\n (clicked)=\"onEditCancel()\"></cqa-button>\n <cqa-button variant=\"filled\" btnSize=\"lg\" [customClass]=\"'cqa-text-[14px] cqa-py-[9px]'\" [text]=\"'Apply'\"\n (clicked)=\"onEditApply()\"></cqa-button>\n </div>\n </div>\n </div>\n\n <!-- Loop Header (normal view when not editing) -->\n <div *ngIf=\"!isEditing\"\n [class]=\"'step-row cqa-flex cqa-items-center cqa-gap-3 cqa-py-2 ' + (isInsideLoop ? 'cqa-pl-10 cqa-pr-4' : 'cqa-px-4')\">\n <div class=\"cqa-inline-flex cqa-items-center\">\n <!-- Drag Handle Icon (when isReorder is true) -->\n <div *ngIf=\"isReorder\" class=\"cqa-mr-2 cqa-cursor-move cqa-text-[#6B7280] hover:cqa-text-[#111827]\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"3\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"3\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"3\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n </svg>\n </div>\n <!-- Checkbox (when isReorder is false) -->\n <label *ngIf=\"!isReorder\" class=\"cqa-flex cqa-items-center cqa-cursor-pointer cqa-relative cqa-mr-2\">\n <input type=\"checkbox\"\n [ngModel]=\"selected\"\n (ngModelChange)=\"onSelectionChange($event)\"\n class=\"cqa-h-4 cqa-w-4 cqa-cursor-pointer cqa-transition-all cqa-appearance-none cqa-rounded shadow hover:cqa-shadow-md cqa-border cqa-border-solid cqa-border-slate-300 cqa-flex-shrink-0\"\n [class.cqa-bg-[#3F43EE]]=\"selected\"\n [class.cqa-border-[#3F43EE]]=\"selected\"\n id=\"check-loop-step\" />\n <span class=\"cqa-absolute cqa-text-white cqa-top-1/2 cqa-left-1/2 cqa--translate-x-1/2 cqa--translate-y-1/2 cqa-pointer-events-none cqa-flex cqa-items-center cqa-justify-center\"\n [class.cqa-opacity-0]=\"!selected\"\n [class.cqa-opacity-100]=\"selected\">\n <svg width=\"12\" height=\"13\" viewBox=\"0 0 12 13\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M10 3.125L4.5 8.625L2 6.125\" stroke=\"white\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </span>\n </label>\n </div>\n <span class=\"cqa-text-[#6B7280] cqa-text-[14px] cqa-leading-[18px] cqa-min-w-[32px]\">{{ stepNumber }}</span>\n <!-- Expand/Collapse Icon -->\n <button type=\"button\" (click)=\"onToggleExpanded()\" class=\"cqa-p-0\">\n <svg [class.cqa-rotate-180]=\"!expanded\" class=\"cqa-transition-transform\" width=\"16\" height=\"16\"\n viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M12 10L8 6L4 10\" stroke=\"#6B7280\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </button>\n\n <!-- Step Number -->\n <span class=\"cqa-text-[#6B7280] cqa-text-[14px] cqa-leading-[18px] cqa-min-w-[32px] cqa-flex-shrink-0\">{{ stepNumber }}</span>\n\n <!-- Loop Icon -->\n <div *ngIf=\"loopType === 'for'\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M11.334 1.33325L14.0007 3.99992L11.334 6.66659\" stroke=\"#1C398E\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M2 7.33333V6.66667C2 5.95942 2.28095 5.28115 2.78105 4.78105C3.28115 4.28095 3.95942 4 4.66667 4H14\"\n stroke=\"#1C398E\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M4.66667 14.6666L2 11.9999L4.66667 9.33325\" stroke=\"#1C398E\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M14 8.66675V9.33341C14 10.0407 13.719 10.7189 13.219 11.219C12.7189 11.7191 12.0406 12.0001 11.3333 12.0001H2\"\n stroke=\"#1C398E\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </div>\n <div *ngIf=\"loopType === 'while'\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M11.334 1.33325L14.0007 3.99992L11.334 6.66659\" stroke=\"#59168B\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M2 7.33333V6.66667C2 5.95942 2.28095 5.28115 2.78105 4.78105C3.28115 4.28095 3.95942 4 4.66667 4H14\"\n stroke=\"#59168B\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M4.66667 14.6666L2 11.9999L4.66667 9.33325\" stroke=\"#59168B\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M14 8.66675V9.33341C14 10.0407 13.719 10.7189 13.219 11.219C12.7189 11.7191 12.0406 12.0001 11.3333 12.0001H2\"\n stroke=\"#59168B\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </div>\n\n <!-- Loop Type Label -->\n <span class=\"cqa-font-semibold cqa-text-[#1C398E] cqa-text-[12px] cqa-leading-none\">\n {{ getLoopTypeLabel() }}\n </span>\n\n <!-- For Loop Controls -->\n <div *ngIf=\"loopType === 'for'\" class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-ml-2\">\n <!-- Test Data Profile Button -->\n <button type=\"button\" [class.cqa-bg-primary-100]=\"testDataProfile\" [class.cqa-text-primary]=\"testDataProfile\"\n class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n {{ getTestDataProfileLabel() }}\n </button>\n\n <!-- Start Button -->\n <button type=\"button\" [class.cqa-bg-primary-100]=\"startStep\" [class.cqa-text-primary]=\"startStep\"\n class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n {{ getStartStepLabel() }}\n </button>\n\n <!-- End Button -->\n <button type=\"button\" [class.cqa-bg-primary-100]=\"endStep\" [class.cqa-text-primary]=\"endStep\"\n class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n {{ getEndStepLabel() }}\n </button>\n </div>\n\n <!-- While Loop Condition (only show if condition exists) -->\n <div *ngIf=\"isWhileLoop() && condition\" class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-ml-2 cqa-flex-1\">\n <span class=\"cqa-text-gray-700 cqa-text-sm\">\n While\n </span>\n <span class=\"cqa-text-gray-900 cqa-text-sm\" [innerHTML]=\"getConditionDisplay()\"></span>\n </div>\n\n <!-- Steps Summary -->\n <div\n class=\"cqa-ml-auto cqa-border cqa-border-solid cqa-border-[#E5E5E5] cqa-rounded-lg cqa-py-0.5 cqa-px-2 cqa-text-[#0A0A0A] cqa-text-[12px] cqa-leading-[15px]\">\n {{ getStepsSummary() }}\n </div>\n\n <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-ml-auto\">\n <!-- View Details Link (show only if description is not empty) -->\n <a *ngIf=\"config.description && config.description.trim() !== ''\" #viewDetailsTrigger href=\"#\" (click)=\"onViewDetails($event)\" class=\"cqa-text-[#3F43EE] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-2 cqa-no-underline cqa-ml-auto\">View Details<svg width=\"8\" height=\"8\" viewBox=\"0 0 8 8\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M2.03809 6.74329L2.62809 7.33329L5.96142 3.99996L2.62809 0.666626L2.03809 1.25663L4.78142 3.99996L2.03809 6.74329Z\" fill=\"#3F43EE\"/></svg></a>\n\n <!-- Action Icons: Edit, Link, Duplicate, Delete (show on hover) -->\n <div class=\"step-actions cqa-flex cqa-items-center cqa-gap-3 cqa-px-[7px]\">\n <button type=\"button\" (click)=\"onEdit(); $event.stopPropagation()\" title=\"Edit\"\n class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M7 11.6666H12.25\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path\n d=\"M9.55208 2.1128C9.7843 1.88058 10.0993 1.75012 10.4277 1.75012C10.7561 1.75012 11.071 1.88058 11.3033 2.1128C11.5355 2.34502 11.6659 2.65998 11.6659 2.98838C11.6659 3.31679 11.5355 3.63175 11.3033 3.86397L4.29742 10.8704C4.15864 11.0092 3.9871 11.1107 3.79867 11.1656L2.12333 11.6544C2.07314 11.669 2.01993 11.6699 1.96928 11.6569C1.91863 11.6439 1.8724 11.6176 1.83543 11.5806C1.79846 11.5437 1.7721 11.4974 1.75913 11.4468C1.74615 11.3961 1.74703 11.3429 1.76167 11.2927L2.2505 9.61738C2.30546 9.42916 2.40698 9.25783 2.54567 9.11922L9.55208 2.1128Z\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </button>\n <!-- <button type=\"button\" (click)=\"onLink(); $event.stopPropagation()\" title=\"Link\"\n class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M5.00065 9.91671H3.66732C2.78326 9.91671 1.93542 9.60942 1.3103 9.06244C0.685174 8.51545 0.333984 7.77359 0.333984 7.00004C0.333984 6.22649 0.685174 5.48463 1.3103 4.93765C1.93542 4.39066 2.78326 4.08337 3.66732 4.08337H5.00065\"\n stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M9 4.08337H10.3333C11.2174 4.08337 12.0652 4.39066 12.6904 4.93765C13.3155 5.48463 13.6667 6.22649 13.6667 7.00004C13.6667 7.77359 13.3155 8.51545 12.6904 9.06244C12.0652 9.60942 11.2174 9.91671 10.3333 9.91671H9\"\n stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M4.33398 7H9.66732\" stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </button> -->\n <button type=\"button\" (click)=\"onDuplicate(); $event.stopPropagation()\" title=\"Duplicate\"\n class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M11.666 4.66663H5.83268C5.18835 4.66663 4.66602 5.18896 4.66602 5.83329V11.6666C4.66602 12.311 5.18835 12.8333 5.83268 12.8333H11.666C12.3103 12.8333 12.8327 12.311 12.8327 11.6666V5.83329C12.8327 5.18896 12.3103 4.66663 11.666 4.66663Z\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M2.33268 9.33329C1.69102 9.33329 1.16602 8.80829 1.16602 8.16663V2.33329C1.16602 1.69163 1.69102 1.16663 2.33268 1.16663H8.16602C8.80768 1.16663 9.33268 1.69163 9.33268 2.33329\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </button>\n <button type=\"button\" (click)=\"onDelete(); $event.stopPropagation()\" title=\"Delete\"\n class=\"cqa-p-0 cqa-text-[#ff6467] hover:cqa-text-[#C63535]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M1.75 3.5H12.25\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path\n d=\"M11.0827 3.5V11.6667C11.0827 12.25 10.4993 12.8333 9.91602 12.8333H4.08268C3.49935 12.8333 2.91602 12.25 2.91602 11.6667V3.5\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M4.66602 3.49996V2.33329C4.66602 1.74996 5.24935 1.16663 5.83268 1.16663H8.16602C8.74935 1.16663 9.33268 1.74996 9.33268 2.33329V3.49996\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M5.83398 6.41663V9.91663\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path d=\"M8.16602 6.41663V9.91663\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </button>\n </div>\n </div>\n </div>\n\n <!-- Expanded Content with Nested Steps (renderer dispatches by step type, n-level nesting) -->\n <div *ngIf=\"expanded\" class=\"cqa-flex cqa-flex-col\">\n <div *ngIf=\"!isReorder\" class=\"cqa-flex cqa-flex-col\">\n <cqa-test-case-details-renderer *ngFor=\"let step of nestedSteps; let i = index\" [step]=\"step\" [index]=\"i\"\n [selected]=\"$any(step).selected\"\n [isNested]=\"true\" [isInsideLoop]=\"true\" [isReorder]=\"isReorder\"\n [addStepBetween]=\"addStepBetween\"\n [dataProfileOptions]=\"dataProfileOptions\" [hasMoreDataProfiles]=\"hasMoreDataProfiles\" [isLoadingDataProfiles]=\"isLoadingDataProfiles\"\n [naturalTextActionsOptions]=\"naturalTextActionsOptions\"\n (nestedStepChange)=\"onNestedStepChange($event.step, $event.index)\"\n (addStep)=\"onAddStep()\" (deleteStep)=\"onDeleteStep($event)\"\n (toggleExpanded)=\"onNestedToggleExpanded($event, step, i)\"\n (groupNameChange)=\"$any(step).groupName = $event; onNestedStepChange(step, i)\"\n (descriptionChange)=\"$any(step).description = $event; onNestedStepChange(step, i)\"\n (reusableChange)=\"$any(step).reusable = $event; onNestedStepChange(step, i)\"\n (openExternal)=\"onNestedStepChange(step, i)\" (edit)=\"onNestedEdit($event)\"\n (link)=\"onNestedStepChange(step, i)\" (duplicate)=\"onNestedDuplicate($event, step)\" (delete)=\"onDeleteStep($event, i)\"\n (viewDetails)=\"viewDetails.emit($event)\"\n (conditionChange)=\"$any(step).condition = $event; onNestedStepChange(step, i)\"\n (branchStepChange)=\"onNestedStepChange(step, i)\" (addBranch)=\"onNestedConditionAddBranch($any(step), i)\"\n (deleteBranch)=\"onNestedConditionDeleteBranch($any(step), $event, i)\"\n (testDataProfileChange)=\"$any(step).testDataProfile = $event; onNestedStepChange(step, i)\"\n (startStepChange)=\"$any(step).startStep = $event; onNestedStepChange(step, i)\"\n (endStepChange)=\"$any(step).endStep = $event; onNestedStepChange(step, i)\"\n (maxIterationsChange)=\"$any(step).maxIterations = $event; onNestedStepChange(step, i)\"\n (eventTypeChange)=\"$any(step).eventType = $event; onNestedStepChange(step, i)\"\n (parameterChange)=\"onNestedStepChange(step, i)\"\n (selectionChange)=\"$any(step).selected = $event; onNestedStepChange(step, i)\"\n (loadMoreDataProfiles)=\"loadMoreDataProfiles.emit($event)\"\n (searchDataProfiles)=\"searchDataProfiles.emit($event)\"\n (dndDropInZone)=\"dndDropInZone.emit($event)\"\n (clickAction)=\"clickAction.emit($event)\"\n (editInDepth)=\"editInDepth.emit($event)\"\n (addStepBetweenClick)=\"addStepBetweenClick.emit($event)\">\n </cqa-test-case-details-renderer>\n </div>\n <div *ngIf=\"isReorder\" class=\"cqa-flex cqa-flex-col nested-step-drop-list\"\n [dndDropzone]=\"['step']\"\n dndEffectAllowed=\"move\"\n dndDragoverClass=\"dndDragover\"\n (dndDrop)=\"onDndDrop($event)\">\n <div dndPlaceholderRef class=\"step-drag-placeholder-nested cqa-my-1 cqa-min-h-[50px] cqa-border-2 cqa-border-dashed cqa-border-[#3F43EE] cqa-rounded cqa-bg-[rgba(63,67,238,0.08)] cqa-flex cqa-items-center cqa-justify-center cqa-text-[#3F43EE] cqa-text-xs\">Drop here</div>\n <div *ngFor=\"let step of nestedSteps; let i = index\" class=\"nested-step-drag-item\"\n [dndDraggable]=\"step\"\n dndEffectAllowed=\"move\"\n dndType=\"step\">\n <cqa-test-case-details-renderer [step]=\"step\" [index]=\"i\"\n [selected]=\"$any(step).selected\"\n [isNested]=\"true\" [isInsideLoop]=\"true\" [isReorder]=\"isReorder\"\n [addStepBetweenAbove]=\"addStepBetweenAbove\"\n [addStepBetweenBelow]=\"addStepBetweenBelow\"\n [addStepBetween]=\"addStepBetween\"\n [dataProfileOptions]=\"dataProfileOptions\" [hasMoreDataProfiles]=\"hasMoreDataProfiles\" [isLoadingDataProfiles]=\"isLoadingDataProfiles\"\n [naturalTextActionsOptions]=\"naturalTextActionsOptions\"\n (nestedStepChange)=\"onNestedStepChange($event.step, $event.index)\"\n (addStep)=\"onAddStep()\" (deleteStep)=\"onDeleteStep($event)\"\n (toggleExpanded)=\"onNestedToggleExpanded($event, step, i)\"\n (groupNameChange)=\"$any(step).groupName = $event; onNestedStepChange(step, i)\"\n (descriptionChange)=\"$any(step).description = $event; onNestedStepChange(step, i)\"\n (reusableChange)=\"$any(step).reusable = $event; onNestedStepChange(step, i)\"\n (openExternal)=\"onNestedStepChange(step, i)\" (edit)=\"onNestedStepChange(step, i)\"\n (link)=\"onNestedStepChange(step, i)\" (duplicate)=\"onNestedStepChange(step, i)\" (delete)=\"onDeleteStep($event, i)\"\n (viewDetails)=\"viewDetails.emit($event)\"\n (conditionChange)=\"$any(step).condition = $event; onNestedStepChange(step, i)\"\n (branchStepChange)=\"onNestedStepChange(step, i)\" (addBranch)=\"onNestedConditionAddBranch($any(step), i)\"\n (deleteBranch)=\"onNestedConditionDeleteBranch($any(step), $event, i)\"\n (testDataProfileChange)=\"$any(step).testDataProfile = $event; onNestedStepChange(step, i)\"\n (startStepChange)=\"$any(step).startStep = $event; onNestedStepChange(step, i)\"\n (endStepChange)=\"$any(step).endStep = $event; onNestedStepChange(step, i)\"\n (maxIterationsChange)=\"$any(step).maxIterations = $event; onNestedStepChange(step, i)\"\n (eventTypeChange)=\"$any(step).eventType = $event; onNestedStepChange(step, i)\"\n (parameterChange)=\"onNestedStepChange(step, i)\"\n (selectionChange)=\"$any(step).selected = $event; onNestedStepChange(step, i)\"\n (loadMoreDataProfiles)=\"loadMoreDataProfiles.emit($event)\"\n (searchDataProfiles)=\"searchDataProfiles.emit($event)\"\n (dndDropInZone)=\"dndDropInZone.emit($event)\"\n (clickAction)=\"clickAction.emit($event)\"\n (editInDepth)=\"editInDepth.emit($event)\"\n (addStepBetweenClick)=\"addStepBetweenClick.emit($event)\">\n </cqa-test-case-details-renderer>\n </div>\n </div>\n\n <!-- Empty loop: show plus icon on hover when expanded and no steps -->\n <div *ngIf=\"nestedSteps.length === 0\" class=\"loop-step-empty cqa-min-h-[52px] cqa-flex cqa-items-center cqa-justify-center cqa-pl-4 cqa-pr-4\">\n <button type=\"button\" (click)=\"onAddStepEmpty(); $event.stopPropagation()\" title=\"Add step\"\n class=\"loop-step-empty-add\">\n <svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <rect x=\"1\" y=\"1\" width=\"21.1822\" height=\"22\" rx=\"10.5911\" fill=\"white\" />\n <rect x=\"1\" y=\"1\" width=\"21.1822\" height=\"22\" rx=\"10.5911\" stroke=\"#3F43EE\" stroke-width=\"2\" />\n <path d=\"M8.5 12H15.5\" stroke=\"#3F43EE\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M12 8.5V15.5\" stroke=\"#3F43EE\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </button>\n </div>\n\n <!-- END Marker -->\n <div\n [class]=\"'cqa-pl-4 cqa-py-1 cqa-bg-[#DBEAFE80] cqa-text-[#1C398E] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium'\"\n style=\"border-top: 1px solid #BEDBFF;\">\n {{ getEndLabel() }}\n </div>\n </div>\n</div>", styles: [".step-actions{opacity:0;transition:opacity .15s ease}.step-row:hover .step-actions{opacity:1}\n", ".loop-step-empty{opacity:0;transition:opacity .1s ease}.loop-step-row:hover .loop-step-empty{opacity:1}.loop-step-empty-add{padding:0;background:transparent;border:0;cursor:pointer;color:#3f43ee;transition:opacity .15s ease}.loop-step-empty-add:hover{opacity:.9}\n"], components: [{ type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }, { type: i3.AutocompleteComponent, selector: "cqa-autocomplete", inputs: ["placeholder", "options", "value", "disabled", "showClear", "ariaLabel", "autoFocus", "size", "fullWidth"], outputs: ["valueChange", "optionSelect", "cleared"] }, { type: i4.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: 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.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }, { type: i7.TestCaseDetailsRendererComponent, selector: "cqa-test-case-details-renderer", inputs: ["step", "index", "isNested", "isInsideLoop", "branch", "isReorder", "selected", "addStepBetweenAbove", "addStepBetweenBelow", "addStepBetween", "dataProfileOptions", "hasMoreDataProfiles", "isLoadingDataProfiles", "naturalTextActionsOptions", "setConditionTemplateVariables"], outputs: ["nestedStepChange", "addStep", "deleteStep", "toggleExpanded", "groupNameChange", "descriptionChange", "reusableChange", "openExternal", "edit", "link", "duplicate", "delete", "viewDetails", "selectionChange", "conditionChange", "branchStepChange", "addStepForBranch", "addStepForLoop", "deleteStepWithBranch", "addBranch", "addElse", "deleteBranch", "testDataProfileChange", "startStepChange", "endStepChange", "maxIterationsChange", "eventTypeChange", "parameterChange", "clickAction", "dndDropInZone", "loadMoreDataProfiles", "searchDataProfiles", "stepUpdate", "addStepBetweenClick", "editInDepth"] }], directives: [{ type: i8.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i8.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i1.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { type: i9.DndDropzoneDirective, selector: "[dndDropzone]", inputs: ["dndDropzone", "dndEffectAllowed", "dndAllowExternal", "dndHorizontal", "dndDragoverClass", "dndDropzoneDisabledClass", "dndDisableIf", "dndDisableDropIf"], outputs: ["dndDragover", "dndDrop"] }, { type: i9.DndPlaceholderRefDirective, selector: "[dndPlaceholderRef]" }, { type: i9.DndDraggableDirective, selector: "[dndDraggable]", inputs: ["dndDraggable", "dndEffectAllowed", "dndType", "dndDraggingClass", "dndDraggingSourceClass", "dndDraggableDisabledClass", "dndDragImageOffsetFunction", "dndDisableIf", "dndDisableDragIf"], outputs: ["dndStart", "dndDrag", "dndEnd", "dndMoved", "dndCopied", "dndLinked", "dndCanceled"] }] });
|
|
459
897
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TestCaseLoopStepComponent, decorators: [{
|
|
460
898
|
type: Component,
|
|
461
|
-
args: [{ selector: 'cqa-test-case-loop-step', host: { class: 'cqa-ui-root' }, styles: [STEP_ROW_ACTIONS_STYLES], template: "<div class=\"cqa-flex cqa-flex-col\">\n <!-- Inline Edit Mode: Frame 2147224910 - flex row, 1056\u00D736, gap 12px, flex-none -->\n <div *ngIf=\"isEditing\" class=\"cqa-py-2.5 cqa-px-4 cqa-flex cqa-items-center cqa-gap-3 cqa-justify-between\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-flex-grow\">\n <!-- Loop Type Label -->\n <div class=\"cqa-font-semibold cqa-text-[#1C398E] cqa-text-[12px] cqa-leading-[15px]\">\n {{ getLoopTypeLabel() }}\n </div>\n\n <!-- FOR LOOP inline edit: cqa-dynamic-select (same usage as default stories) -->\n <ng-container *ngIf=\"loopType === 'for'\">\n <!-- First select: width 216.25px, height 36px, gap 4px, opacity 1 -->\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"forOptionTypeSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"testDataProfileSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"startStepSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"endStepSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n </ng-container>\n\n <!-- WHILE LOOP inline edit -->\n <ng-container *ngIf=\"isWhileLoop()\">\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"conditionSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n </ng-container>\n\n <!-- Edit In depth link -->\n <a href=\"#\" (click)=\"onEditInDepth(); $event.preventDefault()\"\n class=\"cqa-text-[#3F43EE] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-[2px] cqa-no-underline\">\n Edit In depth\n <svg width=\"8\" height=\"8\" viewBox=\"0 0 8 8\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M2.03809 6.74329L2.62809 7.33329L5.96142 3.99996L2.62809 0.666626L2.03809 1.25663L4.78142 3.99996L2.03809 6.74329Z\"\n fill=\"#3F43EE\" />\n </svg>\n </a>\n </div>\n\n <div class=\"cqa-flex cqa-items-center cqa-gap-1.5\">\n <!-- Cancel / Apply -->\n <cqa-button variant=\"outlined\" btnSize=\"lg\" [customClass]=\"'cqa-text-[14px] cqa-py-[9px]'\" [text]=\"'Cancel'\"\n (clicked)=\"onEditCancel()\"></cqa-button>\n <cqa-button variant=\"filled\" btnSize=\"lg\" [customClass]=\"'cqa-text-[14px] cqa-py-[9px]'\" [text]=\"'Apply'\"\n (clicked)=\"onEditApply()\"></cqa-button>\n </div>\n </div>\n\n <!-- Loop Header (normal view when not editing) -->\n <div *ngIf=\"!isEditing\"\n [class]=\"'step-row cqa-flex cqa-items-center cqa-gap-3 cqa-py-2 ' + (isInsideLoop ? 'cqa-pl-10 cqa-pr-4' : 'cqa-px-4')\">\n <div class=\"cqa-inline-flex cqa-items-center\">\n <!-- Drag Handle Icon (when isReorder is true) -->\n <div *ngIf=\"isReorder\" class=\"cqa-mr-2 cqa-cursor-move cqa-text-[#6B7280] hover:cqa-text-[#111827]\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"3\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"3\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"3\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n </svg>\n </div>\n <!-- Checkbox (when isReorder is false) -->\n <label *ngIf=\"!isReorder\" class=\"cqa-flex cqa-items-center cqa-cursor-pointer cqa-relative cqa-mr-2\">\n <input type=\"checkbox\"\n [ngModel]=\"selected\"\n (ngModelChange)=\"onSelectionChange($event)\"\n class=\"cqa-h-4 cqa-w-4 cqa-cursor-pointer cqa-transition-all cqa-appearance-none cqa-rounded shadow hover:cqa-shadow-md cqa-border cqa-border-solid cqa-border-slate-300 cqa-flex-shrink-0\"\n [class.cqa-bg-[#3F43EE]]=\"selected\"\n [class.cqa-border-[#3F43EE]]=\"selected\"\n id=\"check-loop-step\" />\n <span class=\"cqa-absolute cqa-text-white cqa-top-1/2 cqa-left-1/2 cqa--translate-x-1/2 cqa--translate-y-1/2 cqa-pointer-events-none cqa-flex cqa-items-center cqa-justify-center\"\n [class.cqa-opacity-0]=\"!selected\"\n [class.cqa-opacity-100]=\"selected\">\n <svg width=\"12\" height=\"13\" viewBox=\"0 0 12 13\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M10 3.125L4.5 8.625L2 6.125\" stroke=\"white\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </span>\n </label>\n </div>\n <span class=\"cqa-text-[#6B7280] cqa-text-[14px] cqa-leading-[18px] cqa-min-w-[32px]\">{{ stepNumber }}</span>\n <!-- Expand/Collapse Icon -->\n <button type=\"button\" (click)=\"onToggleExpanded()\" class=\"cqa-p-0\">\n <svg [class.cqa-rotate-180]=\"!expanded\" class=\"cqa-transition-transform\" width=\"16\" height=\"16\"\n viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M12 10L8 6L4 10\" stroke=\"#6B7280\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </button>\n\n <!-- Loop Icon -->\n <div *ngIf=\"loopType === 'for'\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M11.334 1.33325L14.0007 3.99992L11.334 6.66659\" stroke=\"#1C398E\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M2 7.33333V6.66667C2 5.95942 2.28095 5.28115 2.78105 4.78105C3.28115 4.28095 3.95942 4 4.66667 4H14\"\n stroke=\"#1C398E\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M4.66667 14.6666L2 11.9999L4.66667 9.33325\" stroke=\"#1C398E\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M14 8.66675V9.33341C14 10.0407 13.719 10.7189 13.219 11.219C12.7189 11.7191 12.0406 12.0001 11.3333 12.0001H2\"\n stroke=\"#1C398E\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </div>\n <div *ngIf=\"loopType === 'while'\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M11.334 1.33325L14.0007 3.99992L11.334 6.66659\" stroke=\"#59168B\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M2 7.33333V6.66667C2 5.95942 2.28095 5.28115 2.78105 4.78105C3.28115 4.28095 3.95942 4 4.66667 4H14\"\n stroke=\"#59168B\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M4.66667 14.6666L2 11.9999L4.66667 9.33325\" stroke=\"#59168B\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M14 8.66675V9.33341C14 10.0407 13.719 10.7189 13.219 11.219C12.7189 11.7191 12.0406 12.0001 11.3333 12.0001H2\"\n stroke=\"#59168B\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </div>\n\n <!-- Loop Type Label -->\n <span class=\"cqa-font-semibold cqa-text-[#1C398E] cqa-text-[12px] cqa-leading-none\">\n {{ getLoopTypeLabel() }}\n </span>\n\n <!-- For Loop Controls -->\n <div *ngIf=\"loopType === 'for'\" class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-ml-2\">\n <!-- Test Data Profile Button -->\n <button type=\"button\" [class.cqa-bg-primary-100]=\"testDataProfile\" [class.cqa-text-primary]=\"testDataProfile\"\n class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n {{ getTestDataProfileLabel() }}\n </button>\n\n <!-- Start Button -->\n <button type=\"button\" [class.cqa-bg-primary-100]=\"startStep\" [class.cqa-text-primary]=\"startStep\"\n class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n {{ getStartStepLabel() }}\n </button>\n\n <!-- End Button -->\n <button type=\"button\" [class.cqa-bg-primary-100]=\"endStep\" [class.cqa-text-primary]=\"endStep\"\n class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n {{ getEndStepLabel() }}\n </button>\n </div>\n\n <!-- While Loop Condition (only show if condition exists) -->\n <div *ngIf=\"isWhileLoop() && condition\" class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-ml-2 cqa-flex-1\">\n <span class=\"cqa-text-gray-700 cqa-text-sm\">\n While\n </span>\n <span class=\"cqa-text-gray-900 cqa-text-sm\" [innerHTML]=\"getConditionDisplay()\"></span>\n </div>\n\n <!-- Steps Summary -->\n <div\n class=\"cqa-ml-auto cqa-border cqa-border-solid cqa-border-[#E5E5E5] cqa-rounded-lg cqa-py-0.5 cqa-px-2 cqa-text-[#0A0A0A] cqa-text-[12px] cqa-leading-[15px]\">\n {{ getStepsSummary() }}\n </div>\n\n <!-- View Details Link (show only if description is not empty) -->\n <a *ngIf=\"config.description && config.description.trim() !== ''\" #viewDetailsTrigger href=\"#\" (click)=\"onViewDetails($event)\" class=\"cqa-text-[#3F43EE] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-2 cqa-no-underline cqa-ml-auto\">View Details<svg width=\"8\" height=\"8\" viewBox=\"0 0 8 8\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M2.03809 6.74329L2.62809 7.33329L5.96142 3.99996L2.62809 0.666626L2.03809 1.25663L4.78142 3.99996L2.03809 6.74329Z\" fill=\"#3F43EE\"/></svg></a>\n\n <!-- Action Icons: Edit, Link, Duplicate, Delete (show on hover) -->\n <div class=\"step-actions cqa-flex cqa-items-center cqa-gap-3 cqa-px-[7px]\">\n <button type=\"button\" (click)=\"onEdit(); $event.stopPropagation()\" title=\"Edit\"\n class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M7 11.6666H12.25\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path\n d=\"M9.55208 2.1128C9.7843 1.88058 10.0993 1.75012 10.4277 1.75012C10.7561 1.75012 11.071 1.88058 11.3033 2.1128C11.5355 2.34502 11.6659 2.65998 11.6659 2.98838C11.6659 3.31679 11.5355 3.63175 11.3033 3.86397L4.29742 10.8704C4.15864 11.0092 3.9871 11.1107 3.79867 11.1656L2.12333 11.6544C2.07314 11.669 2.01993 11.6699 1.96928 11.6569C1.91863 11.6439 1.8724 11.6176 1.83543 11.5806C1.79846 11.5437 1.7721 11.4974 1.75913 11.4468C1.74615 11.3961 1.74703 11.3429 1.76167 11.2927L2.2505 9.61738C2.30546 9.42916 2.40698 9.25783 2.54567 9.11922L9.55208 2.1128Z\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </button>\n <button type=\"button\" (click)=\"onLink(); $event.stopPropagation()\" title=\"Link\"\n class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M5.00065 9.91671H3.66732C2.78326 9.91671 1.93542 9.60942 1.3103 9.06244C0.685174 8.51545 0.333984 7.77359 0.333984 7.00004C0.333984 6.22649 0.685174 5.48463 1.3103 4.93765C1.93542 4.39066 2.78326 4.08337 3.66732 4.08337H5.00065\"\n stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M9 4.08337H10.3333C11.2174 4.08337 12.0652 4.39066 12.6904 4.93765C13.3155 5.48463 13.6667 6.22649 13.6667 7.00004C13.6667 7.77359 13.3155 8.51545 12.6904 9.06244C12.0652 9.60942 11.2174 9.91671 10.3333 9.91671H9\"\n stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M4.33398 7H9.66732\" stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </button>\n <button type=\"button\" (click)=\"onDuplicate(); $event.stopPropagation()\" title=\"Duplicate\"\n class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M11.666 4.66663H5.83268C5.18835 4.66663 4.66602 5.18896 4.66602 5.83329V11.6666C4.66602 12.311 5.18835 12.8333 5.83268 12.8333H11.666C12.3103 12.8333 12.8327 12.311 12.8327 11.6666V5.83329C12.8327 5.18896 12.3103 4.66663 11.666 4.66663Z\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M2.33268 9.33329C1.69102 9.33329 1.16602 8.80829 1.16602 8.16663V2.33329C1.16602 1.69163 1.69102 1.16663 2.33268 1.16663H8.16602C8.80768 1.16663 9.33268 1.69163 9.33268 2.33329\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </button>\n <button type=\"button\" (click)=\"onDelete(); $event.stopPropagation()\" title=\"Delete\"\n class=\"cqa-p-0 cqa-text-[#ff6467] hover:cqa-text-[#C63535]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M1.75 3.5H12.25\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path\n d=\"M11.0827 3.5V11.6667C11.0827 12.25 10.4993 12.8333 9.91602 12.8333H4.08268C3.49935 12.8333 2.91602 12.25 2.91602 11.6667V3.5\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M4.66602 3.49996V2.33329C4.66602 1.74996 5.24935 1.16663 5.83268 1.16663H8.16602C8.74935 1.16663 9.33268 1.74996 9.33268 2.33329V3.49996\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M5.83398 6.41663V9.91663\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path d=\"M8.16602 6.41663V9.91663\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </button>\n </div>\n </div>\n\n <!-- Expanded Content with Nested Steps (renderer dispatches by step type, n-level nesting) -->\n <div *ngIf=\"expanded\" class=\"cqa-flex cqa-flex-col\">\n <div *ngIf=\"!isReorder\" class=\"cqa-flex cqa-flex-col\">\n <cqa-test-case-details-renderer *ngFor=\"let step of nestedSteps; let i = index\" [step]=\"step\" [index]=\"i\"\n [selected]=\"$any(step).selected\"\n [isNested]=\"true\" [isInsideLoop]=\"true\" [isReorder]=\"isReorder\"\n [addStepBetween]=\"addStepBetween\"\n [dataProfileOptions]=\"dataProfileOptions\" [hasMoreDataProfiles]=\"hasMoreDataProfiles\" [isLoadingDataProfiles]=\"isLoadingDataProfiles\"\n [naturalTextActionsOptions]=\"naturalTextActionsOptions\"\n (nestedStepChange)=\"onNestedStepChange($event.step, $event.index)\"\n (addStep)=\"onAddStep()\" (deleteStep)=\"onDeleteStep($event)\"\n (toggleExpanded)=\"onNestedToggleExpanded($event, step, i)\"\n (groupNameChange)=\"$any(step).groupName = $event; onNestedStepChange(step, i)\"\n (descriptionChange)=\"$any(step).description = $event; onNestedStepChange(step, i)\"\n (reusableChange)=\"$any(step).reusable = $event; onNestedStepChange(step, i)\"\n (openExternal)=\"onNestedStepChange(step, i)\" (edit)=\"onNestedStepChange(step, i)\"\n (link)=\"onNestedStepChange(step, i)\" (duplicate)=\"onNestedStepChange(step, i)\" (delete)=\"onDeleteStep(i)\"\n (viewDetails)=\"viewDetails.emit($event)\"\n (conditionChange)=\"$any(step).condition = $event; onNestedStepChange(step, i)\"\n (branchStepChange)=\"onNestedStepChange(step, i)\" (addBranch)=\"onNestedConditionAddBranch($any(step), i)\"\n (deleteBranch)=\"onNestedConditionDeleteBranch($any(step), $event, i)\"\n (testDataProfileChange)=\"$any(step).testDataProfile = $event; onNestedStepChange(step, i)\"\n (startStepChange)=\"$any(step).startStep = $event; onNestedStepChange(step, i)\"\n (endStepChange)=\"$any(step).endStep = $event; onNestedStepChange(step, i)\"\n (maxIterationsChange)=\"$any(step).maxIterations = $event; onNestedStepChange(step, i)\"\n (eventTypeChange)=\"$any(step).eventType = $event; onNestedStepChange(step, i)\"\n (parameterChange)=\"onNestedStepChange(step, i)\"\n (selectionChange)=\"$any(step).selected = $event; onNestedStepChange(step, i)\"\n (loadMoreDataProfiles)=\"loadMoreDataProfiles.emit($event)\"\n (searchDataProfiles)=\"searchDataProfiles.emit($event)\"\n (dndDropInZone)=\"dndDropInZone.emit($event)\"\n (clickAction)=\"clickAction.emit($event)\"\n (editInDepth)=\"editInDepth.emit($event)\"\n (addStepBetweenClick)=\"addStepBetweenClick.emit($event)\">\n </cqa-test-case-details-renderer>\n </div>\n <div *ngIf=\"isReorder\" class=\"cqa-flex cqa-flex-col nested-step-drop-list\"\n [dndDropzone]=\"['step']\"\n dndEffectAllowed=\"move\"\n dndDragoverClass=\"dndDragover\"\n (dndDrop)=\"onDndDrop($event)\">\n <div dndPlaceholderRef class=\"step-drag-placeholder-nested cqa-my-1 cqa-min-h-[50px] cqa-border-2 cqa-border-dashed cqa-border-[#3F43EE] cqa-rounded cqa-bg-[rgba(63,67,238,0.08)] cqa-flex cqa-items-center cqa-justify-center cqa-text-[#3F43EE] cqa-text-xs\">Drop here</div>\n <div *ngFor=\"let step of nestedSteps; let i = index\" class=\"nested-step-drag-item\"\n [dndDraggable]=\"step\"\n dndEffectAllowed=\"move\"\n dndType=\"step\">\n <cqa-test-case-details-renderer [step]=\"step\" [index]=\"i\"\n [selected]=\"$any(step).selected\"\n [isNested]=\"true\" [isInsideLoop]=\"true\" [isReorder]=\"isReorder\"\n [addStepBetweenAbove]=\"addStepBetweenAbove\"\n [addStepBetweenBelow]=\"addStepBetweenBelow\"\n [addStepBetween]=\"addStepBetween\"\n [dataProfileOptions]=\"dataProfileOptions\" [hasMoreDataProfiles]=\"hasMoreDataProfiles\" [isLoadingDataProfiles]=\"isLoadingDataProfiles\"\n [naturalTextActionsOptions]=\"naturalTextActionsOptions\"\n (nestedStepChange)=\"onNestedStepChange($event.step, $event.index)\"\n (addStep)=\"onAddStep()\" (deleteStep)=\"onDeleteStep($event)\"\n (toggleExpanded)=\"onNestedToggleExpanded($event, step, i)\"\n (groupNameChange)=\"$any(step).groupName = $event; onNestedStepChange(step, i)\"\n (descriptionChange)=\"$any(step).description = $event; onNestedStepChange(step, i)\"\n (reusableChange)=\"$any(step).reusable = $event; onNestedStepChange(step, i)\"\n (openExternal)=\"onNestedStepChange(step, i)\" (edit)=\"onNestedStepChange(step, i)\"\n (link)=\"onNestedStepChange(step, i)\" (duplicate)=\"onNestedStepChange(step, i)\" (delete)=\"onDeleteStep(i)\"\n (viewDetails)=\"viewDetails.emit($event)\"\n (conditionChange)=\"$any(step).condition = $event; onNestedStepChange(step, i)\"\n (branchStepChange)=\"onNestedStepChange(step, i)\" (addBranch)=\"onNestedConditionAddBranch($any(step), i)\"\n (deleteBranch)=\"onNestedConditionDeleteBranch($any(step), $event, i)\"\n (testDataProfileChange)=\"$any(step).testDataProfile = $event; onNestedStepChange(step, i)\"\n (startStepChange)=\"$any(step).startStep = $event; onNestedStepChange(step, i)\"\n (endStepChange)=\"$any(step).endStep = $event; onNestedStepChange(step, i)\"\n (maxIterationsChange)=\"$any(step).maxIterations = $event; onNestedStepChange(step, i)\"\n (eventTypeChange)=\"$any(step).eventType = $event; onNestedStepChange(step, i)\"\n (parameterChange)=\"onNestedStepChange(step, i)\"\n (selectionChange)=\"$any(step).selected = $event; onNestedStepChange(step, i)\"\n (loadMoreDataProfiles)=\"loadMoreDataProfiles.emit($event)\"\n (searchDataProfiles)=\"searchDataProfiles.emit($event)\"\n (dndDropInZone)=\"dndDropInZone.emit($event)\"\n (clickAction)=\"clickAction.emit($event)\"\n (editInDepth)=\"editInDepth.emit($event)\"\n (addStepBetweenClick)=\"addStepBetweenClick.emit($event)\">\n </cqa-test-case-details-renderer>\n </div>\n </div>\n\n <!-- END Marker -->\n <div\n [class]=\"'cqa-pl-4 cqa-py-1 cqa-bg-[#DBEAFE80] cqa-text-[#1C398E] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium'\"\n style=\"border-top: 1px solid #BEDBFF;\">\n {{ getEndLabel() }}\n </div>\n </div>\n</div>" }]
|
|
462
|
-
|
|
899
|
+
args: [{ selector: 'cqa-test-case-loop-step', host: { class: 'cqa-ui-root' }, styles: [
|
|
900
|
+
STEP_ROW_ACTIONS_STYLES,
|
|
901
|
+
`
|
|
902
|
+
.loop-step-empty {
|
|
903
|
+
opacity: 0;
|
|
904
|
+
transition: opacity 0.1s ease;
|
|
905
|
+
}
|
|
906
|
+
.loop-step-row:hover .loop-step-empty {
|
|
907
|
+
opacity: 1;
|
|
908
|
+
}
|
|
909
|
+
.loop-step-empty-add {
|
|
910
|
+
padding: 0;
|
|
911
|
+
background: transparent;
|
|
912
|
+
border: 0;
|
|
913
|
+
cursor: pointer;
|
|
914
|
+
color: #3F43EE;
|
|
915
|
+
transition: opacity 0.15s ease;
|
|
916
|
+
}
|
|
917
|
+
.loop-step-empty-add:hover {
|
|
918
|
+
opacity: 0.9;
|
|
919
|
+
}
|
|
920
|
+
`,
|
|
921
|
+
], template: "<div class=\"loop-step-row cqa-flex cqa-flex-col\">\n <!-- Inline Edit Mode: Frame 2147224910 - flex row, 1056\u00D736, gap 12px, flex-none -->\n <div *ngIf=\"isEditing\" class=\"cqa-py-2.5 cqa-px-4 cqa-flex cqa-flex-col cqa-gap-3\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-justify-between\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-flex-grow\">\n <!-- Loop Type Label -->\n <div class=\"cqa-font-semibold cqa-text-[#1C398E] cqa-text-[12px] cqa-leading-[15px]\">\n {{ getLoopTypeLabel() }}\n </div>\n\n <!-- FOR LOOP inline edit: cqa-dynamic-select (same usage as default stories) -->\n <ng-container *ngIf=\"loopType === 'for'\">\n <!-- First select: width 216.25px, height 36px, gap 4px, opacity 1 -->\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"forOptionTypeSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"testDataProfileSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"startStepSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"editForm\" [config]=\"endStepSelectConfig\"\n class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n </ng-container>\n\n <!-- WHILE LOOP inline edit: cqa-autocomplete + template variables (same as condition-step) -->\n <ng-container *ngIf=\"isWhileLoop()\">\n <cqa-autocomplete\n *ngIf=\"editForm\"\n [options]=\"whileConditionAutocompleteOptions\"\n [value]=\"editForm.get('condition')?.value ?? ''\"\n (valueChange)=\"onEditFormFieldChange('condition', $event)\"\n (optionSelect)=\"onConditionSelect($event)\"\n placeholder=\"Select condition\"\n [fullWidth]=\"true\"\n class=\"cqa-w-full cqa-max-w-[216px]\">\n </cqa-autocomplete>\n <!-- Template variables (shown when template is selected, same as condition-step) -->\n <div *ngIf=\"selectedTemplate && templateVariables && templateVariables.length > 0\" class=\"cqa-flex cqa-flex-row cqa-flex-wrap cqa-gap-3\">\n <ng-container *ngFor=\"let variable of templateVariables\">\n <ng-container *ngIf=\"variable.type === 'boolean'\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-gray-700\">{{ variable.label }}</label>\n <mat-slide-toggle\n [checked]=\"templateVariablesForm.get(variable.name)?.value || variable.value || false\"\n (change)=\"templateVariablesForm.get(variable.name)?.setValue($event.checked)\"\n color=\"primary\">\n </mat-slide-toggle>\n </div>\n </ng-container>\n <ng-container *ngIf=\"variable.type !== 'boolean' && variable.name !== 'custom_code'\">\n <ng-container *ngIf=\"variable.name === 'type' || variable.name === 'scrollTo' || variable.options\">\n <div class=\"cqa-flex cqa-flex-col\" style=\"min-width: 150px;\">\n <cqa-dynamic-select\n [form]=\"templateVariablesForm\"\n [config]=\"getSelectConfigForVariable(variable)\">\n </cqa-dynamic-select>\n </div>\n </ng-container>\n <ng-container *ngIf=\"variable.name !== 'type' && variable.name !== 'scrollTo' && !variable.options\">\n <div class=\"cqa-flex cqa-flex-col\" style=\"min-width: 150px;\">\n <cqa-custom-input\n [placeholder]=\"'Enter ' + variable.label\"\n [value]=\"templateVariablesForm.get(variable.name)?.value || variable.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"templateVariablesForm.get(variable.name)?.setValue($event)\">\n </cqa-custom-input>\n </div>\n </ng-container>\n </ng-container>\n </ng-container>\n </div>\n </ng-container>\n\n <!-- Edit In depth link -->\n <a href=\"#\" (click)=\"onEditInDepth(); $event.preventDefault()\"\n class=\"cqa-text-[#3F43EE] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-[2px] cqa-no-underline\">\n Edit In depth\n <svg width=\"8\" height=\"8\" viewBox=\"0 0 8 8\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M2.03809 6.74329L2.62809 7.33329L5.96142 3.99996L2.62809 0.666626L2.03809 1.25663L4.78142 3.99996L2.03809 6.74329Z\"\n fill=\"#3F43EE\" />\n </svg>\n </a>\n </div>\n\n <div class=\"cqa-flex cqa-items-center cqa-gap-1.5\">\n <!-- Cancel / Apply -->\n <cqa-button variant=\"outlined\" btnSize=\"lg\" [customClass]=\"'cqa-text-[14px] cqa-py-[9px]'\" [text]=\"'Cancel'\"\n (clicked)=\"onEditCancel()\"></cqa-button>\n <cqa-button variant=\"filled\" btnSize=\"lg\" [customClass]=\"'cqa-text-[14px] cqa-py-[9px]'\" [text]=\"'Apply'\"\n (clicked)=\"onEditApply()\"></cqa-button>\n </div>\n </div>\n </div>\n\n <!-- Loop Header (normal view when not editing) -->\n <div *ngIf=\"!isEditing\"\n [class]=\"'step-row cqa-flex cqa-items-center cqa-gap-3 cqa-py-2 ' + (isInsideLoop ? 'cqa-pl-10 cqa-pr-4' : 'cqa-px-4')\">\n <div class=\"cqa-inline-flex cqa-items-center\">\n <!-- Drag Handle Icon (when isReorder is true) -->\n <div *ngIf=\"isReorder\" class=\"cqa-mr-2 cqa-cursor-move cqa-text-[#6B7280] hover:cqa-text-[#111827]\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"3\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"3\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"3\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"8\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n <circle cx=\"13\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n </svg>\n </div>\n <!-- Checkbox (when isReorder is false) -->\n <label *ngIf=\"!isReorder\" class=\"cqa-flex cqa-items-center cqa-cursor-pointer cqa-relative cqa-mr-2\">\n <input type=\"checkbox\"\n [ngModel]=\"selected\"\n (ngModelChange)=\"onSelectionChange($event)\"\n class=\"cqa-h-4 cqa-w-4 cqa-cursor-pointer cqa-transition-all cqa-appearance-none cqa-rounded shadow hover:cqa-shadow-md cqa-border cqa-border-solid cqa-border-slate-300 cqa-flex-shrink-0\"\n [class.cqa-bg-[#3F43EE]]=\"selected\"\n [class.cqa-border-[#3F43EE]]=\"selected\"\n id=\"check-loop-step\" />\n <span class=\"cqa-absolute cqa-text-white cqa-top-1/2 cqa-left-1/2 cqa--translate-x-1/2 cqa--translate-y-1/2 cqa-pointer-events-none cqa-flex cqa-items-center cqa-justify-center\"\n [class.cqa-opacity-0]=\"!selected\"\n [class.cqa-opacity-100]=\"selected\">\n <svg width=\"12\" height=\"13\" viewBox=\"0 0 12 13\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M10 3.125L4.5 8.625L2 6.125\" stroke=\"white\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n </span>\n </label>\n </div>\n <span class=\"cqa-text-[#6B7280] cqa-text-[14px] cqa-leading-[18px] cqa-min-w-[32px]\">{{ stepNumber }}</span>\n <!-- Expand/Collapse Icon -->\n <button type=\"button\" (click)=\"onToggleExpanded()\" class=\"cqa-p-0\">\n <svg [class.cqa-rotate-180]=\"!expanded\" class=\"cqa-transition-transform\" width=\"16\" height=\"16\"\n viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M12 10L8 6L4 10\" stroke=\"#6B7280\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </button>\n\n <!-- Step Number -->\n <span class=\"cqa-text-[#6B7280] cqa-text-[14px] cqa-leading-[18px] cqa-min-w-[32px] cqa-flex-shrink-0\">{{ stepNumber }}</span>\n\n <!-- Loop Icon -->\n <div *ngIf=\"loopType === 'for'\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M11.334 1.33325L14.0007 3.99992L11.334 6.66659\" stroke=\"#1C398E\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M2 7.33333V6.66667C2 5.95942 2.28095 5.28115 2.78105 4.78105C3.28115 4.28095 3.95942 4 4.66667 4H14\"\n stroke=\"#1C398E\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M4.66667 14.6666L2 11.9999L4.66667 9.33325\" stroke=\"#1C398E\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M14 8.66675V9.33341C14 10.0407 13.719 10.7189 13.219 11.219C12.7189 11.7191 12.0406 12.0001 11.3333 12.0001H2\"\n stroke=\"#1C398E\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </div>\n <div *ngIf=\"loopType === 'while'\">\n <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M11.334 1.33325L14.0007 3.99992L11.334 6.66659\" stroke=\"#59168B\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M2 7.33333V6.66667C2 5.95942 2.28095 5.28115 2.78105 4.78105C3.28115 4.28095 3.95942 4 4.66667 4H14\"\n stroke=\"#59168B\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M4.66667 14.6666L2 11.9999L4.66667 9.33325\" stroke=\"#59168B\" stroke-width=\"1.33333\"\n stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M14 8.66675V9.33341C14 10.0407 13.719 10.7189 13.219 11.219C12.7189 11.7191 12.0406 12.0001 11.3333 12.0001H2\"\n stroke=\"#59168B\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </div>\n\n <!-- Loop Type Label -->\n <span class=\"cqa-font-semibold cqa-text-[#1C398E] cqa-text-[12px] cqa-leading-none\">\n {{ getLoopTypeLabel() }}\n </span>\n\n <!-- For Loop Controls -->\n <div *ngIf=\"loopType === 'for'\" class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-ml-2\">\n <!-- Test Data Profile Button -->\n <button type=\"button\" [class.cqa-bg-primary-100]=\"testDataProfile\" [class.cqa-text-primary]=\"testDataProfile\"\n class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n {{ getTestDataProfileLabel() }}\n </button>\n\n <!-- Start Button -->\n <button type=\"button\" [class.cqa-bg-primary-100]=\"startStep\" [class.cqa-text-primary]=\"startStep\"\n class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n {{ getStartStepLabel() }}\n </button>\n\n <!-- End Button -->\n <button type=\"button\" [class.cqa-bg-primary-100]=\"endStep\" [class.cqa-text-primary]=\"endStep\"\n class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n {{ getEndStepLabel() }}\n </button>\n </div>\n\n <!-- While Loop Condition (only show if condition exists) -->\n <div *ngIf=\"isWhileLoop() && condition\" class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-ml-2 cqa-flex-1\">\n <span class=\"cqa-text-gray-700 cqa-text-sm\">\n While\n </span>\n <span class=\"cqa-text-gray-900 cqa-text-sm\" [innerHTML]=\"getConditionDisplay()\"></span>\n </div>\n\n <!-- Steps Summary -->\n <div\n class=\"cqa-ml-auto cqa-border cqa-border-solid cqa-border-[#E5E5E5] cqa-rounded-lg cqa-py-0.5 cqa-px-2 cqa-text-[#0A0A0A] cqa-text-[12px] cqa-leading-[15px]\">\n {{ getStepsSummary() }}\n </div>\n\n <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-ml-auto\">\n <!-- View Details Link (show only if description is not empty) -->\n <a *ngIf=\"config.description && config.description.trim() !== ''\" #viewDetailsTrigger href=\"#\" (click)=\"onViewDetails($event)\" class=\"cqa-text-[#3F43EE] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-2 cqa-no-underline cqa-ml-auto\">View Details<svg width=\"8\" height=\"8\" viewBox=\"0 0 8 8\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M2.03809 6.74329L2.62809 7.33329L5.96142 3.99996L2.62809 0.666626L2.03809 1.25663L4.78142 3.99996L2.03809 6.74329Z\" fill=\"#3F43EE\"/></svg></a>\n\n <!-- Action Icons: Edit, Link, Duplicate, Delete (show on hover) -->\n <div class=\"step-actions cqa-flex cqa-items-center cqa-gap-3 cqa-px-[7px]\">\n <button type=\"button\" (click)=\"onEdit(); $event.stopPropagation()\" title=\"Edit\"\n class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M7 11.6666H12.25\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path\n d=\"M9.55208 2.1128C9.7843 1.88058 10.0993 1.75012 10.4277 1.75012C10.7561 1.75012 11.071 1.88058 11.3033 2.1128C11.5355 2.34502 11.6659 2.65998 11.6659 2.98838C11.6659 3.31679 11.5355 3.63175 11.3033 3.86397L4.29742 10.8704C4.15864 11.0092 3.9871 11.1107 3.79867 11.1656L2.12333 11.6544C2.07314 11.669 2.01993 11.6699 1.96928 11.6569C1.91863 11.6439 1.8724 11.6176 1.83543 11.5806C1.79846 11.5437 1.7721 11.4974 1.75913 11.4468C1.74615 11.3961 1.74703 11.3429 1.76167 11.2927L2.2505 9.61738C2.30546 9.42916 2.40698 9.25783 2.54567 9.11922L9.55208 2.1128Z\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </button>\n <!-- <button type=\"button\" (click)=\"onLink(); $event.stopPropagation()\" title=\"Link\"\n class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M5.00065 9.91671H3.66732C2.78326 9.91671 1.93542 9.60942 1.3103 9.06244C0.685174 8.51545 0.333984 7.77359 0.333984 7.00004C0.333984 6.22649 0.685174 5.48463 1.3103 4.93765C1.93542 4.39066 2.78326 4.08337 3.66732 4.08337H5.00065\"\n stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M9 4.08337H10.3333C11.2174 4.08337 12.0652 4.39066 12.6904 4.93765C13.3155 5.48463 13.6667 6.22649 13.6667 7.00004C13.6667 7.77359 13.3155 8.51545 12.6904 9.06244C12.0652 9.60942 11.2174 9.91671 10.3333 9.91671H9\"\n stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M4.33398 7H9.66732\" stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </button> -->\n <button type=\"button\" (click)=\"onDuplicate(); $event.stopPropagation()\" title=\"Duplicate\"\n class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path\n d=\"M11.666 4.66663H5.83268C5.18835 4.66663 4.66602 5.18896 4.66602 5.83329V11.6666C4.66602 12.311 5.18835 12.8333 5.83268 12.8333H11.666C12.3103 12.8333 12.8327 12.311 12.8327 11.6666V5.83329C12.8327 5.18896 12.3103 4.66663 11.666 4.66663Z\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M2.33268 9.33329C1.69102 9.33329 1.16602 8.80829 1.16602 8.16663V2.33329C1.16602 1.69163 1.69102 1.16663 2.33268 1.16663H8.16602C8.80768 1.16663 9.33268 1.69163 9.33268 2.33329\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </button>\n <button type=\"button\" (click)=\"onDelete(); $event.stopPropagation()\" title=\"Delete\"\n class=\"cqa-p-0 cqa-text-[#ff6467] hover:cqa-text-[#C63535]\">\n <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M1.75 3.5H12.25\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path\n d=\"M11.0827 3.5V11.6667C11.0827 12.25 10.4993 12.8333 9.91602 12.8333H4.08268C3.49935 12.8333 2.91602 12.25 2.91602 11.6667V3.5\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path\n d=\"M4.66602 3.49996V2.33329C4.66602 1.74996 5.24935 1.16663 5.83268 1.16663H8.16602C8.74935 1.16663 9.33268 1.74996 9.33268 2.33329V3.49996\"\n stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M5.83398 6.41663V9.91663\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n <path d=\"M8.16602 6.41663V9.91663\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" />\n </svg>\n </button>\n </div>\n </div>\n </div>\n\n <!-- Expanded Content with Nested Steps (renderer dispatches by step type, n-level nesting) -->\n <div *ngIf=\"expanded\" class=\"cqa-flex cqa-flex-col\">\n <div *ngIf=\"!isReorder\" class=\"cqa-flex cqa-flex-col\">\n <cqa-test-case-details-renderer *ngFor=\"let step of nestedSteps; let i = index\" [step]=\"step\" [index]=\"i\"\n [selected]=\"$any(step).selected\"\n [isNested]=\"true\" [isInsideLoop]=\"true\" [isReorder]=\"isReorder\"\n [addStepBetween]=\"addStepBetween\"\n [dataProfileOptions]=\"dataProfileOptions\" [hasMoreDataProfiles]=\"hasMoreDataProfiles\" [isLoadingDataProfiles]=\"isLoadingDataProfiles\"\n [naturalTextActionsOptions]=\"naturalTextActionsOptions\"\n (nestedStepChange)=\"onNestedStepChange($event.step, $event.index)\"\n (addStep)=\"onAddStep()\" (deleteStep)=\"onDeleteStep($event)\"\n (toggleExpanded)=\"onNestedToggleExpanded($event, step, i)\"\n (groupNameChange)=\"$any(step).groupName = $event; onNestedStepChange(step, i)\"\n (descriptionChange)=\"$any(step).description = $event; onNestedStepChange(step, i)\"\n (reusableChange)=\"$any(step).reusable = $event; onNestedStepChange(step, i)\"\n (openExternal)=\"onNestedStepChange(step, i)\" (edit)=\"onNestedEdit($event)\"\n (link)=\"onNestedStepChange(step, i)\" (duplicate)=\"onNestedDuplicate($event, step)\" (delete)=\"onDeleteStep($event, i)\"\n (viewDetails)=\"viewDetails.emit($event)\"\n (conditionChange)=\"$any(step).condition = $event; onNestedStepChange(step, i)\"\n (branchStepChange)=\"onNestedStepChange(step, i)\" (addBranch)=\"onNestedConditionAddBranch($any(step), i)\"\n (deleteBranch)=\"onNestedConditionDeleteBranch($any(step), $event, i)\"\n (testDataProfileChange)=\"$any(step).testDataProfile = $event; onNestedStepChange(step, i)\"\n (startStepChange)=\"$any(step).startStep = $event; onNestedStepChange(step, i)\"\n (endStepChange)=\"$any(step).endStep = $event; onNestedStepChange(step, i)\"\n (maxIterationsChange)=\"$any(step).maxIterations = $event; onNestedStepChange(step, i)\"\n (eventTypeChange)=\"$any(step).eventType = $event; onNestedStepChange(step, i)\"\n (parameterChange)=\"onNestedStepChange(step, i)\"\n (selectionChange)=\"$any(step).selected = $event; onNestedStepChange(step, i)\"\n (loadMoreDataProfiles)=\"loadMoreDataProfiles.emit($event)\"\n (searchDataProfiles)=\"searchDataProfiles.emit($event)\"\n (dndDropInZone)=\"dndDropInZone.emit($event)\"\n (clickAction)=\"clickAction.emit($event)\"\n (editInDepth)=\"editInDepth.emit($event)\"\n (addStepBetweenClick)=\"addStepBetweenClick.emit($event)\">\n </cqa-test-case-details-renderer>\n </div>\n <div *ngIf=\"isReorder\" class=\"cqa-flex cqa-flex-col nested-step-drop-list\"\n [dndDropzone]=\"['step']\"\n dndEffectAllowed=\"move\"\n dndDragoverClass=\"dndDragover\"\n (dndDrop)=\"onDndDrop($event)\">\n <div dndPlaceholderRef class=\"step-drag-placeholder-nested cqa-my-1 cqa-min-h-[50px] cqa-border-2 cqa-border-dashed cqa-border-[#3F43EE] cqa-rounded cqa-bg-[rgba(63,67,238,0.08)] cqa-flex cqa-items-center cqa-justify-center cqa-text-[#3F43EE] cqa-text-xs\">Drop here</div>\n <div *ngFor=\"let step of nestedSteps; let i = index\" class=\"nested-step-drag-item\"\n [dndDraggable]=\"step\"\n dndEffectAllowed=\"move\"\n dndType=\"step\">\n <cqa-test-case-details-renderer [step]=\"step\" [index]=\"i\"\n [selected]=\"$any(step).selected\"\n [isNested]=\"true\" [isInsideLoop]=\"true\" [isReorder]=\"isReorder\"\n [addStepBetweenAbove]=\"addStepBetweenAbove\"\n [addStepBetweenBelow]=\"addStepBetweenBelow\"\n [addStepBetween]=\"addStepBetween\"\n [dataProfileOptions]=\"dataProfileOptions\" [hasMoreDataProfiles]=\"hasMoreDataProfiles\" [isLoadingDataProfiles]=\"isLoadingDataProfiles\"\n [naturalTextActionsOptions]=\"naturalTextActionsOptions\"\n (nestedStepChange)=\"onNestedStepChange($event.step, $event.index)\"\n (addStep)=\"onAddStep()\" (deleteStep)=\"onDeleteStep($event)\"\n (toggleExpanded)=\"onNestedToggleExpanded($event, step, i)\"\n (groupNameChange)=\"$any(step).groupName = $event; onNestedStepChange(step, i)\"\n (descriptionChange)=\"$any(step).description = $event; onNestedStepChange(step, i)\"\n (reusableChange)=\"$any(step).reusable = $event; onNestedStepChange(step, i)\"\n (openExternal)=\"onNestedStepChange(step, i)\" (edit)=\"onNestedStepChange(step, i)\"\n (link)=\"onNestedStepChange(step, i)\" (duplicate)=\"onNestedStepChange(step, i)\" (delete)=\"onDeleteStep($event, i)\"\n (viewDetails)=\"viewDetails.emit($event)\"\n (conditionChange)=\"$any(step).condition = $event; onNestedStepChange(step, i)\"\n (branchStepChange)=\"onNestedStepChange(step, i)\" (addBranch)=\"onNestedConditionAddBranch($any(step), i)\"\n (deleteBranch)=\"onNestedConditionDeleteBranch($any(step), $event, i)\"\n (testDataProfileChange)=\"$any(step).testDataProfile = $event; onNestedStepChange(step, i)\"\n (startStepChange)=\"$any(step).startStep = $event; onNestedStepChange(step, i)\"\n (endStepChange)=\"$any(step).endStep = $event; onNestedStepChange(step, i)\"\n (maxIterationsChange)=\"$any(step).maxIterations = $event; onNestedStepChange(step, i)\"\n (eventTypeChange)=\"$any(step).eventType = $event; onNestedStepChange(step, i)\"\n (parameterChange)=\"onNestedStepChange(step, i)\"\n (selectionChange)=\"$any(step).selected = $event; onNestedStepChange(step, i)\"\n (loadMoreDataProfiles)=\"loadMoreDataProfiles.emit($event)\"\n (searchDataProfiles)=\"searchDataProfiles.emit($event)\"\n (dndDropInZone)=\"dndDropInZone.emit($event)\"\n (clickAction)=\"clickAction.emit($event)\"\n (editInDepth)=\"editInDepth.emit($event)\"\n (addStepBetweenClick)=\"addStepBetweenClick.emit($event)\">\n </cqa-test-case-details-renderer>\n </div>\n </div>\n\n <!-- Empty loop: show plus icon on hover when expanded and no steps -->\n <div *ngIf=\"nestedSteps.length === 0\" class=\"loop-step-empty cqa-min-h-[52px] cqa-flex cqa-items-center cqa-justify-center cqa-pl-4 cqa-pr-4\">\n <button type=\"button\" (click)=\"onAddStepEmpty(); $event.stopPropagation()\" title=\"Add step\"\n class=\"loop-step-empty-add\">\n <svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <rect x=\"1\" y=\"1\" width=\"21.1822\" height=\"22\" rx=\"10.5911\" fill=\"white\" />\n <rect x=\"1\" y=\"1\" width=\"21.1822\" height=\"22\" rx=\"10.5911\" stroke=\"#3F43EE\" stroke-width=\"2\" />\n <path d=\"M8.5 12H15.5\" stroke=\"#3F43EE\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n <path d=\"M12 8.5V15.5\" stroke=\"#3F43EE\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n </svg>\n </button>\n </div>\n\n <!-- END Marker -->\n <div\n [class]=\"'cqa-pl-4 cqa-py-1 cqa-bg-[#DBEAFE80] cqa-text-[#1C398E] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium'\"\n style=\"border-top: 1px solid #BEDBFF;\">\n {{ getEndLabel() }}\n </div>\n </div>\n</div>" }]
|
|
922
|
+
}], ctorParameters: function () { return [{ type: i1.FormBuilder }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { config: [{
|
|
463
923
|
type: Input
|
|
464
924
|
}], id: [{
|
|
465
925
|
type: Input
|
|
@@ -497,6 +957,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
|
|
|
497
957
|
type: Input
|
|
498
958
|
}], naturalTextActionsOptions: [{
|
|
499
959
|
type: Input
|
|
960
|
+
}], setConditionTemplateVariables: [{
|
|
961
|
+
type: Input
|
|
500
962
|
}], addStepBetweenAbove: [{
|
|
501
963
|
type: Input
|
|
502
964
|
}], addStepBetweenBelow: [{
|
|
@@ -551,4 +1013,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
|
|
|
551
1013
|
type: ViewChild,
|
|
552
1014
|
args: ['viewDetailsTrigger', { static: false }]
|
|
553
1015
|
}] } });
|
|
554
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"loop-step.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/loop-step/loop-step.component.ts","../../../../../../src/lib/test-case-details/loop-step/loop-step.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, OnChanges, SimpleChanges, ViewChild, ElementRef } from '@angular/core';\nimport { FormBuilder, FormGroup } from '@angular/forms';\nimport { DndDropEvent } from 'ngx-drag-drop';\nimport { DynamicSelectFieldConfig, SelectOption } from '../../dynamic-select/dynamic-select-field.component';\nimport { LoopStepConfig, TestCaseStepConfig, NormalStepConfig, ConditionStepConfig, StepGroupConfig, ConditionBranch, StepTypes } from '../test-case-step.models';\nimport { STEP_ROW_ACTIONS_STYLES } from '../step-row-actions.styles';\nimport { DataProfileOption } from '../../step-builder/step-builder-loop/step-builder-loop.component';\nimport { ViewDetailsPayload } from '../step-details-modal/step-details-modal-data';\n\n@Component({\n  selector: 'cqa-test-case-loop-step',\n  templateUrl: './loop-step.component.html',\n  host: { class: 'cqa-ui-root' },\n  styles: [STEP_ROW_ACTIONS_STYLES],\n})\nexport class TestCaseLoopStepComponent implements OnInit, OnChanges {\n  @Input() config!: LoopStepConfig;\n  @Input() id!: string;\n  @Input() loopType: 'for' | 'while' = 'for';\n  @Input() stepNumber!: number | string;\n  @Input() condition?: string;\n  @Input() maxIterations?: number;\n  @Input() testDataProfile?: string;\n  @Input() startStep?: number;\n  @Input() endStep?: number;\n  @Input() nestedSteps: TestCaseStepConfig[] = [];\n  @Input() expanded: boolean = true;\n  @Input() isNested: boolean = false;\n  @Input() isInsideLoop: boolean = false;\n  @Input() isReorder: boolean = false;\n  @Input() selected: boolean = false;\n  /** Options for the data profile dropdown - accepts DataProfileOption objects */\n  @Input() dataProfileOptions: DataProfileOption[] = [];\n  /** Indicates if more data profiles are available for loading */\n  @Input() hasMoreDataProfiles: boolean = false;\n  /** Loading state for data profiles */\n  @Input() isLoadingDataProfiles: boolean = false;\n  /** Options for natural text actions (for while loop condition) */\n  @Input() naturalTextActionsOptions: SelectOption[] = [];\n  /** Show \"Add Step Between\" button above nested steps */\n  @Input() addStepBetweenAbove: boolean = false;\n  /** Show \"Add Step Between\" button below nested steps */\n  @Input() addStepBetweenBelow: boolean = false;\n  /** @deprecated Use addStepBetweenAbove and addStepBetweenBelow instead. If set, applies to both buttons. */\n  @Input() addStepBetween: boolean = false;\n\n  @Output() toggleExpanded = new EventEmitter<{config: LoopStepConfig, expanded: boolean}>();\n  @Output() testDataProfileChange = new EventEmitter<string>();\n  @Output() startStepChange = new EventEmitter<number>();\n  @Output() endStepChange = new EventEmitter<number>();\n  @Output() conditionChange = new EventEmitter<string>();\n  @Output() maxIterationsChange = new EventEmitter<number>();\n  @Output() nestedStepChange = new EventEmitter<{ step: TestCaseStepConfig; index: number }>();\n  @Output() addStep = new EventEmitter<void>();\n  @Output() deleteStep = new EventEmitter<number>();\n  @Output() duplicate = new EventEmitter<void>();\n  @Output() delete = new EventEmitter<void>();\n  @Output() moreOptions = new EventEmitter<void>();\n  @Output() viewDetails = new EventEmitter<ViewDetailsPayload>();\n  @Output() editInDepth = new EventEmitter<{ step: TestCaseStepConfig; index: number }>();\n  @Output() edit = new EventEmitter<void>();\n  @Output() dndDropInZone = new EventEmitter<{ event: DndDropEvent; targetList: TestCaseStepConfig[] }>();\n  /** Emit when more data profiles need to be loaded */\n  @Output() loadMoreDataProfiles = new EventEmitter<string>();\n  /** Emit when data profile search query changes */\n  @Output() searchDataProfiles = new EventEmitter<string>();\n  /** Emit when step is updated (Apply button clicked in edit mode) */\n  @Output() stepUpdate = new EventEmitter<LoopStepConfig>();\n  /** Re-emit when a nested step (e.g. normal-step) emits clickAction (e.g. element/label click for popup) */\n  @Output() clickAction = new EventEmitter<{ event: MouseEvent; step: TestCaseStepConfig }>();\n  /** Re-emit addStepBetweenClick from nested renderers */\n  @Output() addStepBetweenClick = new EventEmitter<{step: TestCaseStepConfig, index: number, position: 'ABOVE' | 'BELOW'}>();\n  @Output() selectionChange = new EventEmitter<boolean>();\n\n  onDndDrop(event: DndDropEvent): void {\n    this.dndDropInZone.emit({ event, targetList: this.nestedSteps });\n  }\n\n  /** Expose global constructors for template (Angular templates don't have String/Number) */\n  readonly stringFn = String;\n  readonly numberFn = Number;\n\n  /** When true, header shows inline edit form (dropdowns, inputs, Cancel/Apply) */\n  isEditing: boolean = false;\n\n  /** Form for edit mode; bound to cqa-dynamic-select (same pattern as default stories) */\n  editForm!: FormGroup;\n\n  /** Cached select configs (stable refs to avoid change-detection loops) */\n  forOptionTypeSelectConfig!: DynamicSelectFieldConfig;\n  testDataProfileSelectConfig!: DynamicSelectFieldConfig;\n  startStepSelectConfig!: DynamicSelectFieldConfig;\n  endStepSelectConfig!: DynamicSelectFieldConfig;\n  conditionSelectConfig!: DynamicSelectFieldConfig;\n\n  /** Options for Loop Start / Loop End dropdowns (1-10) */\n  loopStepOptions: number[] = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];\n\n  @ViewChild('viewDetailsTrigger', { static: false }) viewDetailsTrigger?: ElementRef<HTMLElement>;\n\n  /** Local copy of values while editing (for Cancel revert) */\n  private editSnapshot: {\n    testDataProfile?: string;\n    startStep?: number;\n    endStep?: number;\n    condition?: string;\n    maxIterations?: number;\n  } = {};\n\n  constructor(\n    private fb: FormBuilder\n  ) {}\n\n  ngOnInit(): void {\n    if (this.config) {\n      this.id = this.config.id;\n      this.loopType = this.config.loopType;\n      this.stepNumber = this.config.stepNumber;\n      this.condition = this.config.condition;\n      this.maxIterations = this.config.maxIterations;\n      this.testDataProfile = this.config.testDataProfile;\n      this.startStep = this.config.startStep;\n      this.endStep = this.config.endStep;\n      this.nestedSteps = this.config.nestedSteps || [];\n      this.expanded = this.config.expanded !== undefined ? this.config.expanded : true;\n      this.selected = (this.config as any).selected || false;\n    }\n    this.buildEditForm();\n    this.buildSelectConfigs();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['config'] && this.config) {\n      this.selected = (this.config as any).selected || false;\n    }\n    if (changes['dataProfileOptions'] || changes['hasMoreDataProfiles'] || changes['isLoadingDataProfiles']) {\n      this.updateTestDataProfileSelectConfig();\n    }\n    if (changes['naturalTextActionsOptions']) {\n      this.updateConditionSelectConfig();\n    }\n  }\n\n  private buildEditForm(): void {\n    this.editForm = this.fb.group({\n      forOptionType: [null as string | null],\n      testDataProfile: [this.testDataProfile ?? ''],\n      startStep: [this.startStep ?? null as number | null],\n      endStep: [this.endStep ?? null as number | null],\n      condition: [this.condition ?? ''],\n      maxIterations: [this.maxIterations ?? 5],\n    });\n  }\n\n  /** Build select configs once (same option shape as default stories: id + name) */\n  private buildSelectConfigs(): void {\n    const stepOptions: SelectOption[] = this.loopStepOptions.map((n) => ({\n      id: n,\n      value: n,\n      name: String(n),\n      label: String(n),\n    }));\n    this.forOptionTypeSelectConfig = {\n      key: 'forOptionType',\n      placeholder: 'Select option',\n      searchable: false,\n      options: [\n        { id: 'test-data', value: 'test-data', name: 'Test Data Profile', label: 'Test Data Profile' },\n        { id: 'fixed', value: 'fixed', name: 'Fixed count', label: 'Fixed count' },\n      ],\n    };\n    this.updateTestDataProfileSelectConfig();\n    this.startStepSelectConfig = {\n      key: 'startStep',\n      placeholder: 'Loop Start',\n      searchable: false,\n      options: stepOptions.map((o) => ({ ...o })),\n    };\n    this.endStepSelectConfig = {\n      key: 'endStep',\n      placeholder: 'Loop End',\n      searchable: false,\n      options: stepOptions.map((o) => ({ ...o })),\n    };\n    this.updateConditionSelectConfig();\n  }\n\n  /** Update condition select config with natural text actions options */\n  private updateConditionSelectConfig(): void {\n    this.conditionSelectConfig = {\n      key: 'condition',\n      placeholder: 'Select condition',\n      searchable: true,\n      options: this.naturalTextActionsOptions || [],\n    };\n  }\n\n  /** Update test data profile select config with current options and callbacks */\n  private updateTestDataProfileSelectConfig(): void {\n    this.testDataProfileSelectConfig = {\n      key: 'testDataProfile',\n      placeholder: 'Testdataprofile',\n      searchable: true,\n      serverSearch: true,\n      hasMore: this.hasMoreDataProfiles,\n      isLoading: this.isLoadingDataProfiles,\n      options: this.convertDataProfileOptionsToSelectOptions(this.dataProfileOptions),\n      onChange: (value: any) => {\n        this.onDataProfileChange(value);\n      },\n      onSearch: (query: string) => {\n        this.onSearchDataProfiles(query);\n      },\n      onLoadMore: (query?: string) => {\n        this.onLoadMoreDataProfiles(query || '');\n      },\n    };\n  }\n\n  /** Convert DataProfileOption[] to SelectOption[] */\n  private convertDataProfileOptionsToSelectOptions(options: DataProfileOption[]): SelectOption[] {\n    return options.map(option => ({\n      id: option.id,\n      value: option.id,\n      name: option.name,\n      label: option.name\n    }));\n  }\n\n  /** Handle data profile selection */\n  private onDataProfileChange(profileId: any): void {\n    this.onTestDataProfileChange(profileId);\n  }\n\n  /** Handle search for data profiles */\n  private onSearchDataProfiles(query: string): void {\n    this.searchDataProfiles.emit(query);\n  }\n\n  /** Handle load more data profiles */\n  private onLoadMoreDataProfiles(query: string): void {\n    this.loadMoreDataProfiles.emit(query);\n  }\n\n  onEditFormFieldChange(controlName: string, value: string | number): void {\n    this.editForm.get(controlName)?.setValue(value);\n  }\n\n  getLoopTypeLabel(): string {\n    return this.loopType === 'for' ? 'FOR LOOP' : 'WHILE LOOP';\n  }\n\n  isWhileLoop(): boolean {\n    return this.loopType === 'while';\n  }\n\n  isForLoop(): boolean {\n    return this.loopType === 'for';\n  }\n\n  getEndLabel(): string {\n    return this.loopType === 'for' ? 'END FOR' : 'END WHILE';\n  }\n\n  getStepsSummary(): string {\n    if (this.nestedSteps.length === 0) {\n      return '0 steps';\n    }\n    const start = this.startStep || this.nestedSteps[0]?.stepNumber || 1;\n    const end = this.endStep || this.nestedSteps[this.nestedSteps.length - 1]?.stepNumber || this.nestedSteps.length;\n    const count = this.nestedSteps.length;\n    return `Steps ${start}-${end} (${count} step${count !== 1 ? 's' : ''})`;\n  }\n\n  getStartStepLabel(): string {\n    if (this.startStep) {\n      return `Start: ${this.startStep}`;\n    }\n    const defaultStart = this.nestedSteps[0]?.stepNumber || 1;\n    return `Start: ${defaultStart}`;\n  }\n\n  getEndStepLabel(): string {\n    if (this.endStep) {\n      return `End: ${this.endStep}`;\n    }\n    const defaultEnd = this.nestedSteps[this.nestedSteps.length - 1]?.stepNumber || this.nestedSteps.length;\n    return `End: ${defaultEnd}`;\n  }\n\n  getTestDataProfileLabel(): string {\n    if (this.testDataProfile) {\n      return `Test Data profile: ${this.testDataProfile}`;\n    }\n    return 'Test Data profile';\n  }\n\n  getConditionDisplay(): string {\n    return this.condition || 'element \".notification-badge\" exists';\n  }\n\n  onToggleExpanded(): void {\n    this.expanded = !this.expanded;\n    this.toggleExpanded.emit({config:this.config, expanded:this.expanded});\n  }\n\n  onTestDataProfileChange(value: string): void {\n    this.testDataProfile = value;\n    this.testDataProfileChange.emit(value);\n  }\n\n  onStartStepChange(value: number): void {\n    this.startStep = value;\n    this.startStepChange.emit(value);\n  }\n\n  onEndStepChange(value: number): void {\n    this.endStep = value;\n    this.endStepChange.emit(value);\n  }\n\n  onConditionChange(value: string): void {\n    this.condition = value;\n    this.conditionChange.emit(value);\n  }\n\n  onMaxIterationsChange(value: number): void {\n    this.maxIterations = value;\n    this.maxIterationsChange.emit(value);\n  }\n\n  onNestedStepChange(step: TestCaseStepConfig, index: number): void {\n    this.nestedStepChange.emit({ step, index });\n  }\n\n  onNestedToggleExpanded(event: boolean | {config: TestCaseStepConfig, expanded: boolean}, step: TestCaseStepConfig, index: number): void {\n    // Handle both boolean and object formats\n    if (typeof event === 'object' && event !== null && 'expanded' in event) {\n      // Object format from step-group: {config, expanded}\n      (step as any).expanded = event.expanded;\n      // Check if the event's config is a step-group (not the current step)\n      // This ensures nested step-group events bubble up correctly even when condition step is inside loop step\n      if (this.isStepGroup(event.config)) {\n        // Re-emit the toggleExpanded event with the object format, preserving the original config\n        // Cast to satisfy TypeScript - the renderer accepts TestCaseStepConfig which includes StepGroupConfig\n        this.toggleExpanded.emit({config: event.config as any as LoopStepConfig, expanded: event.expanded});\n      }\n    } else {\n      // Boolean format from other step types\n      (step as any).expanded = event as boolean;\n    }\n    this.onNestedStepChange(step, index);\n  }\n\n  onAddStep(): void {\n    this.addStep.emit();\n  }\n\n  onDeleteStep(index: number): void {\n    this.deleteStep.emit(index);\n  }\n\n  isNormalStep(step: TestCaseStepConfig): step is NormalStepConfig {\n    return 'eventType' in step;\n  }\n\n  isLoopStep(step: TestCaseStepConfig): step is LoopStepConfig {\n    return 'loopType' in step;\n  }\n\n  isConditionStep(step: TestCaseStepConfig): step is ConditionStepConfig {\n    return 'condition' in step && 'branches' in step;\n  }\n\n  isStepGroup(step: TestCaseStepConfig): step is StepGroupConfig {\n    return 'groupName' in step;\n  }\n\n  onAddBranch(): void {\n    // Not used in loop step, but needed for recursive condition steps\n  }\n\n  onDeleteBranch(branch: any): void {\n    // Not used in loop step, but needed for recursive condition steps\n  }\n\n  onNestedConditionAddBranch(nestedStep: ConditionStepConfig, index: number): void {\n    // Add a new branch to the nested condition step\n    if (!nestedStep.branches) {\n      nestedStep.branches = [];\n    }\n    const newBranch: ConditionBranch = {\n      type: nestedStep.branches.length === 0 ? 'if' : 'else',\n      label: nestedStep.branches.length === 0 ? 'IF TRUE' : 'ELSE', \n      stepType: StepTypes.CONDITION_IF,\n      nestedSteps: []\n    };\n    nestedStep.branches.push(newBranch);\n    this.onNestedStepChange(nestedStep, index);\n  }\n\n  onNestedConditionDeleteBranch(nestedStep: ConditionStepConfig, branch: any, index: number): void {\n    // Delete a branch from the nested condition step\n    if (nestedStep.branches) {\n      const branchIndex = nestedStep.branches.indexOf(branch);\n      if (branchIndex > -1) {\n        nestedStep.branches.splice(branchIndex, 1);\n      }\n    }\n    this.onNestedStepChange(nestedStep, index);\n  }\n\n  onOpenExternal(): void {\n    // Not used in loop step, but needed for recursive step groups\n  }\n\n  onEdit(): void {\n    this.editSnapshot = {\n      testDataProfile: this.testDataProfile,\n      startStep: this.startStep,\n      endStep: this.endStep,\n      condition: this.condition,\n      maxIterations: this.maxIterations,\n    };\n    this.editForm.patchValue({\n      forOptionType: null,\n      testDataProfile: this.testDataProfile ?? '',\n      startStep: this.startStep ?? null,\n      endStep: this.endStep ?? null,\n      condition: this.condition ?? '',\n      maxIterations: this.maxIterations ?? 5,\n    });\n    this.isEditing = true;\n    this.edit.emit();\n  }\n\n  onEditApply(): void {\n    const v = this.editForm.value;\n    const testDataProfileValue = v.testDataProfile ?? '';\n    this.startStep = v.startStep ?? undefined;\n    this.endStep = v.endStep ?? undefined;\n    this.condition = v.condition ?? '';\n    const maxVal = v.maxIterations;\n    this.maxIterations = typeof maxVal === 'number' ? maxVal : (maxVal != null && maxVal !== '' ? Number(maxVal) : undefined);\n    \n    // Find the selected data profile option to get both ID and name\n    let testDataProfileId: string | number | undefined;\n    let testDataProfileName: string = '';\n    \n    if (testDataProfileValue && this.dataProfileOptions.length > 0) {\n      // The form value might be ID (number) or name (string)\n      const selectedProfile = this.dataProfileOptions.find(\n        option => option.id === Number(testDataProfileValue) || option.name === testDataProfileValue || String(option.id) === String(testDataProfileValue)\n      );\n      if (selectedProfile) {\n        testDataProfileId = selectedProfile.id;\n        testDataProfileName = selectedProfile.name;\n        this.testDataProfile = testDataProfileName;\n      } else {\n        // If not found, use the value as-is (might be ID or name)\n        this.testDataProfile = testDataProfileValue;\n        testDataProfileId = Number(testDataProfileValue) || undefined;\n      }\n    } else {\n      this.testDataProfile = testDataProfileValue;\n    }\n    \n    // Update config with new values (include testDataProfileId as additional property)\n    const updatedConfig: LoopStepConfig & { testDataProfileId?: string | number } = {\n      ...this.config,\n      testDataProfile: this.testDataProfile,\n      testDataProfileId: testDataProfileId, // Add ID for API call\n      startStep: this.startStep,\n      endStep: this.endStep,\n      condition: this.condition,\n      maxIterations: this.maxIterations,\n    };\n    \n    // Emit individual change events\n    this.testDataProfileChange.emit(this.testDataProfile);\n    this.startStepChange.emit(this.startStep ?? 1);\n    this.endStepChange.emit(this.endStep ?? 1);\n    if (this.loopType === 'while') {\n      this.conditionChange.emit(this.condition);\n      this.maxIterationsChange.emit(this.maxIterations ?? 5);\n    }\n    \n    // Emit step update event with full config\n    console.log('Loop-step: Emitting stepUpdate event', updatedConfig);\n    this.stepUpdate.emit(updatedConfig);\n    \n    this.isEditing = false;\n  }\n\n  onEditCancel(): void {\n    this.testDataProfile = this.editSnapshot.testDataProfile;\n    this.startStep = this.editSnapshot.startStep;\n    this.endStep = this.editSnapshot.endStep;\n    this.condition = this.editSnapshot.condition;\n    this.maxIterations = this.editSnapshot.maxIterations;\n    this.isEditing = false;\n  }\n\n  onEditInDepth(): void {\n    this.moreOptions.emit();\n    const index = typeof this.stepNumber === 'number' ? this.stepNumber : (typeof this.stepNumber === 'string' ? parseInt(this.stepNumber, 10) : 0);\n    this.editInDepth.emit({ step: this.config as TestCaseStepConfig, index: isNaN(index) ? 0 : index });\n  }\n\n  onLink(): void {\n    // Not used in loop step, but needed for recursive step groups\n  }\n\n  onDuplicate(): void {\n    this.duplicate.emit();\n  }\n\n  onDelete(): void {\n    this.delete.emit();\n  }\n\n  onSelectionChange(checked: boolean): void {\n    this.selected = checked;\n    this.selectionChange.emit(checked);\n  }\n\n  onMoreOptions(): void {\n    this.moreOptions.emit();\n  }\n\n  onViewDetails(event?: MouseEvent): void {\n    if (event) {\n      event.preventDefault();\n      event.stopPropagation();\n    }\n    \n    if (!this.config.description || this.config.description.trim() === '') {\n      return;\n    }\n\n    if (event) {\n      const title = this.loopType === 'for' ? 'FOR LOOP Details' : 'WHILE LOOP Details';\n      this.viewDetails.emit({\n        event,\n        title,\n        description: this.config.description,\n      });\n    }\n  }\n}\n","<div class=\"cqa-flex cqa-flex-col\">\n  <!-- Inline Edit Mode: Frame 2147224910 - flex row, 1056×36, gap 12px, flex-none -->\n  <div *ngIf=\"isEditing\" class=\"cqa-py-2.5 cqa-px-4 cqa-flex cqa-items-center cqa-gap-3 cqa-justify-between\">\n    <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-flex-grow\">\n      <!-- Loop Type Label -->\n      <div class=\"cqa-font-semibold cqa-text-[#1C398E] cqa-text-[12px] cqa-leading-[15px]\">\n        {{ getLoopTypeLabel() }}\n      </div>\n\n      <!-- FOR LOOP inline edit: cqa-dynamic-select (same usage as default stories) -->\n      <ng-container *ngIf=\"loopType === 'for'\">\n        <!-- First select: width 216.25px, height 36px, gap 4px, opacity 1 -->\n        <cqa-dynamic-select [form]=\"editForm\" [config]=\"forOptionTypeSelectConfig\"\n          class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n        <cqa-dynamic-select [form]=\"editForm\" [config]=\"testDataProfileSelectConfig\"\n          class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n        <cqa-dynamic-select [form]=\"editForm\" [config]=\"startStepSelectConfig\"\n          class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n        <cqa-dynamic-select [form]=\"editForm\" [config]=\"endStepSelectConfig\"\n          class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n      </ng-container>\n\n      <!-- WHILE LOOP inline edit -->\n      <ng-container *ngIf=\"isWhileLoop()\">\n        <cqa-dynamic-select [form]=\"editForm\" [config]=\"conditionSelectConfig\"\n          class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n      </ng-container>\n\n      <!-- Edit In depth link -->\n      <a href=\"#\" (click)=\"onEditInDepth(); $event.preventDefault()\"\n        class=\"cqa-text-[#3F43EE] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-[2px] cqa-no-underline\">\n        Edit In depth\n        <svg width=\"8\" height=\"8\" viewBox=\"0 0 8 8\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n          <path\n            d=\"M2.03809 6.74329L2.62809 7.33329L5.96142 3.99996L2.62809 0.666626L2.03809 1.25663L4.78142 3.99996L2.03809 6.74329Z\"\n            fill=\"#3F43EE\" />\n        </svg>\n      </a>\n    </div>\n\n    <div class=\"cqa-flex cqa-items-center cqa-gap-1.5\">\n      <!-- Cancel / Apply -->\n      <cqa-button variant=\"outlined\" btnSize=\"lg\" [customClass]=\"'cqa-text-[14px] cqa-py-[9px]'\" [text]=\"'Cancel'\"\n        (clicked)=\"onEditCancel()\"></cqa-button>\n      <cqa-button variant=\"filled\" btnSize=\"lg\" [customClass]=\"'cqa-text-[14px] cqa-py-[9px]'\" [text]=\"'Apply'\"\n        (clicked)=\"onEditApply()\"></cqa-button>\n    </div>\n  </div>\n\n  <!-- Loop Header (normal view when not editing) -->\n  <div *ngIf=\"!isEditing\"\n    [class]=\"'step-row cqa-flex cqa-items-center cqa-gap-3 cqa-py-2 ' + (isInsideLoop ? 'cqa-pl-10 cqa-pr-4' : 'cqa-px-4')\">\n    <div class=\"cqa-inline-flex cqa-items-center\">\n    <!-- Drag Handle Icon (when isReorder is true) -->\n    <div *ngIf=\"isReorder\" class=\"cqa-mr-2 cqa-cursor-move cqa-text-[#6B7280] hover:cqa-text-[#111827]\">\n      <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n        <circle cx=\"3\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"8\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"13\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"3\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"8\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"13\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"3\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"8\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"13\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n      </svg>\n    </div>\n    <!-- Checkbox (when isReorder is false) -->\n    <label *ngIf=\"!isReorder\" class=\"cqa-flex cqa-items-center cqa-cursor-pointer cqa-relative cqa-mr-2\">\n      <input type=\"checkbox\"\n        [ngModel]=\"selected\"\n        (ngModelChange)=\"onSelectionChange($event)\"\n        class=\"cqa-h-4 cqa-w-4 cqa-cursor-pointer cqa-transition-all cqa-appearance-none cqa-rounded shadow hover:cqa-shadow-md cqa-border cqa-border-solid cqa-border-slate-300 cqa-flex-shrink-0\"\n        [class.cqa-bg-[#3F43EE]]=\"selected\"\n        [class.cqa-border-[#3F43EE]]=\"selected\"\n        id=\"check-loop-step\" />\n      <span class=\"cqa-absolute cqa-text-white cqa-top-1/2 cqa-left-1/2 cqa--translate-x-1/2 cqa--translate-y-1/2 cqa-pointer-events-none cqa-flex cqa-items-center cqa-justify-center\"\n        [class.cqa-opacity-0]=\"!selected\"\n        [class.cqa-opacity-100]=\"selected\">\n        <svg width=\"12\" height=\"13\" viewBox=\"0 0 12 13\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M10 3.125L4.5 8.625L2 6.125\" stroke=\"white\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n      </span>\n    </label>\n    </div>\n    <span class=\"cqa-text-[#6B7280] cqa-text-[14px] cqa-leading-[18px] cqa-min-w-[32px]\">{{ stepNumber }}</span>\n    <!-- Expand/Collapse Icon -->\n    <button type=\"button\" (click)=\"onToggleExpanded()\" class=\"cqa-p-0\">\n      <svg [class.cqa-rotate-180]=\"!expanded\" class=\"cqa-transition-transform\" width=\"16\" height=\"16\"\n        viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n        <path d=\"M12 10L8 6L4 10\" stroke=\"#6B7280\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n          stroke-linejoin=\"round\" />\n      </svg>\n    </button>\n\n    <!-- Loop Icon -->\n    <div *ngIf=\"loopType === 'for'\">\n      <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n        <path d=\"M11.334 1.33325L14.0007 3.99992L11.334 6.66659\" stroke=\"#1C398E\" stroke-width=\"1.33333\"\n          stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        <path d=\"M2 7.33333V6.66667C2 5.95942 2.28095 5.28115 2.78105 4.78105C3.28115 4.28095 3.95942 4 4.66667 4H14\"\n          stroke=\"#1C398E\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        <path d=\"M4.66667 14.6666L2 11.9999L4.66667 9.33325\" stroke=\"#1C398E\" stroke-width=\"1.33333\"\n          stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        <path\n          d=\"M14 8.66675V9.33341C14 10.0407 13.719 10.7189 13.219 11.219C12.7189 11.7191 12.0406 12.0001 11.3333 12.0001H2\"\n          stroke=\"#1C398E\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n      </svg>\n    </div>\n    <div *ngIf=\"loopType === 'while'\">\n      <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n        <path d=\"M11.334 1.33325L14.0007 3.99992L11.334 6.66659\" stroke=\"#59168B\" stroke-width=\"1.33333\"\n          stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        <path d=\"M2 7.33333V6.66667C2 5.95942 2.28095 5.28115 2.78105 4.78105C3.28115 4.28095 3.95942 4 4.66667 4H14\"\n          stroke=\"#59168B\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        <path d=\"M4.66667 14.6666L2 11.9999L4.66667 9.33325\" stroke=\"#59168B\" stroke-width=\"1.33333\"\n          stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        <path\n          d=\"M14 8.66675V9.33341C14 10.0407 13.719 10.7189 13.219 11.219C12.7189 11.7191 12.0406 12.0001 11.3333 12.0001H2\"\n          stroke=\"#59168B\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n      </svg>\n    </div>\n\n    <!-- Loop Type Label -->\n    <span class=\"cqa-font-semibold cqa-text-[#1C398E] cqa-text-[12px] cqa-leading-none\">\n      {{ getLoopTypeLabel() }}\n    </span>\n\n    <!-- For Loop Controls -->\n    <div *ngIf=\"loopType === 'for'\" class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-ml-2\">\n      <!-- Test Data Profile Button -->\n      <button type=\"button\" [class.cqa-bg-primary-100]=\"testDataProfile\" [class.cqa-text-primary]=\"testDataProfile\"\n        class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n        {{ getTestDataProfileLabel() }}\n      </button>\n\n      <!-- Start Button -->\n      <button type=\"button\" [class.cqa-bg-primary-100]=\"startStep\" [class.cqa-text-primary]=\"startStep\"\n        class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n        {{ getStartStepLabel() }}\n      </button>\n\n      <!-- End Button -->\n      <button type=\"button\" [class.cqa-bg-primary-100]=\"endStep\" [class.cqa-text-primary]=\"endStep\"\n        class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n        {{ getEndStepLabel() }}\n      </button>\n    </div>\n\n    <!-- While Loop Condition (only show if condition exists) -->\n    <div *ngIf=\"isWhileLoop() && condition\" class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-ml-2 cqa-flex-1\">\n      <span class=\"cqa-text-gray-700 cqa-text-sm\">\n        While\n      </span>\n      <span class=\"cqa-text-gray-900 cqa-text-sm\" [innerHTML]=\"getConditionDisplay()\"></span>\n    </div>\n\n    <!-- Steps Summary -->\n    <div\n      class=\"cqa-ml-auto cqa-border cqa-border-solid cqa-border-[#E5E5E5] cqa-rounded-lg cqa-py-0.5 cqa-px-2 cqa-text-[#0A0A0A] cqa-text-[12px] cqa-leading-[15px]\">\n      {{ getStepsSummary() }}\n    </div>\n\n    <!-- View Details Link (show only if description is not empty) -->\n    <a *ngIf=\"config.description && config.description.trim() !== ''\" #viewDetailsTrigger href=\"#\" (click)=\"onViewDetails($event)\" class=\"cqa-text-[#3F43EE] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-2 cqa-no-underline cqa-ml-auto\">View Details<svg width=\"8\" height=\"8\" viewBox=\"0 0 8 8\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M2.03809 6.74329L2.62809 7.33329L5.96142 3.99996L2.62809 0.666626L2.03809 1.25663L4.78142 3.99996L2.03809 6.74329Z\" fill=\"#3F43EE\"/></svg></a>\n\n    <!-- Action Icons: Edit, Link, Duplicate, Delete (show on hover) -->\n    <div class=\"step-actions cqa-flex cqa-items-center cqa-gap-3 cqa-px-[7px]\">\n      <button type=\"button\" (click)=\"onEdit(); $event.stopPropagation()\" title=\"Edit\"\n        class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n        <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n          <path d=\"M7 11.6666H12.25\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n            stroke-linejoin=\"round\" />\n          <path\n            d=\"M9.55208 2.1128C9.7843 1.88058 10.0993 1.75012 10.4277 1.75012C10.7561 1.75012 11.071 1.88058 11.3033 2.1128C11.5355 2.34502 11.6659 2.65998 11.6659 2.98838C11.6659 3.31679 11.5355 3.63175 11.3033 3.86397L4.29742 10.8704C4.15864 11.0092 3.9871 11.1107 3.79867 11.1656L2.12333 11.6544C2.07314 11.669 2.01993 11.6699 1.96928 11.6569C1.91863 11.6439 1.8724 11.6176 1.83543 11.5806C1.79846 11.5437 1.7721 11.4974 1.75913 11.4468C1.74615 11.3961 1.74703 11.3429 1.76167 11.2927L2.2505 9.61738C2.30546 9.42916 2.40698 9.25783 2.54567 9.11922L9.55208 2.1128Z\"\n            stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        </svg>\n      </button>\n      <button type=\"button\" (click)=\"onLink(); $event.stopPropagation()\" title=\"Link\"\n        class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n        <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n          <path\n            d=\"M5.00065 9.91671H3.66732C2.78326 9.91671 1.93542 9.60942 1.3103 9.06244C0.685174 8.51545 0.333984 7.77359 0.333984 7.00004C0.333984 6.22649 0.685174 5.48463 1.3103 4.93765C1.93542 4.39066 2.78326 4.08337 3.66732 4.08337H5.00065\"\n            stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n          <path\n            d=\"M9 4.08337H10.3333C11.2174 4.08337 12.0652 4.39066 12.6904 4.93765C13.3155 5.48463 13.6667 6.22649 13.6667 7.00004C13.6667 7.77359 13.3155 8.51545 12.6904 9.06244C12.0652 9.60942 11.2174 9.91671 10.3333 9.91671H9\"\n            stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n          <path d=\"M4.33398 7H9.66732\" stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n            stroke-linejoin=\"round\" />\n        </svg>\n      </button>\n      <button type=\"button\" (click)=\"onDuplicate(); $event.stopPropagation()\" title=\"Duplicate\"\n        class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n        <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n          <path\n            d=\"M11.666 4.66663H5.83268C5.18835 4.66663 4.66602 5.18896 4.66602 5.83329V11.6666C4.66602 12.311 5.18835 12.8333 5.83268 12.8333H11.666C12.3103 12.8333 12.8327 12.311 12.8327 11.6666V5.83329C12.8327 5.18896 12.3103 4.66663 11.666 4.66663Z\"\n            stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n          <path\n            d=\"M2.33268 9.33329C1.69102 9.33329 1.16602 8.80829 1.16602 8.16663V2.33329C1.16602 1.69163 1.69102 1.16663 2.33268 1.16663H8.16602C8.80768 1.16663 9.33268 1.69163 9.33268 2.33329\"\n            stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        </svg>\n      </button>\n      <button type=\"button\" (click)=\"onDelete(); $event.stopPropagation()\" title=\"Delete\"\n        class=\"cqa-p-0 cqa-text-[#ff6467] hover:cqa-text-[#C63535]\">\n        <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n          <path d=\"M1.75 3.5H12.25\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n            stroke-linejoin=\"round\" />\n          <path\n            d=\"M11.0827 3.5V11.6667C11.0827 12.25 10.4993 12.8333 9.91602 12.8333H4.08268C3.49935 12.8333 2.91602 12.25 2.91602 11.6667V3.5\"\n            stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n          <path\n            d=\"M4.66602 3.49996V2.33329C4.66602 1.74996 5.24935 1.16663 5.83268 1.16663H8.16602C8.74935 1.16663 9.33268 1.74996 9.33268 2.33329V3.49996\"\n            stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n          <path d=\"M5.83398 6.41663V9.91663\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n            stroke-linejoin=\"round\" />\n          <path d=\"M8.16602 6.41663V9.91663\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n            stroke-linejoin=\"round\" />\n        </svg>\n      </button>\n    </div>\n  </div>\n\n  <!-- Expanded Content with Nested Steps (renderer dispatches by step type, n-level nesting) -->\n  <div *ngIf=\"expanded\" class=\"cqa-flex cqa-flex-col\">\n    <div *ngIf=\"!isReorder\" class=\"cqa-flex cqa-flex-col\">\n      <cqa-test-case-details-renderer *ngFor=\"let step of nestedSteps; let i = index\" [step]=\"step\" [index]=\"i\"\n        [selected]=\"$any(step).selected\"\n        [isNested]=\"true\" [isInsideLoop]=\"true\" [isReorder]=\"isReorder\"\n        [addStepBetween]=\"addStepBetween\"\n        [dataProfileOptions]=\"dataProfileOptions\" [hasMoreDataProfiles]=\"hasMoreDataProfiles\" [isLoadingDataProfiles]=\"isLoadingDataProfiles\"\n        [naturalTextActionsOptions]=\"naturalTextActionsOptions\"\n        (nestedStepChange)=\"onNestedStepChange($event.step, $event.index)\"\n        (addStep)=\"onAddStep()\" (deleteStep)=\"onDeleteStep($event)\"\n        (toggleExpanded)=\"onNestedToggleExpanded($event, step, i)\"\n        (groupNameChange)=\"$any(step).groupName = $event; onNestedStepChange(step, i)\"\n        (descriptionChange)=\"$any(step).description = $event; onNestedStepChange(step, i)\"\n        (reusableChange)=\"$any(step).reusable = $event; onNestedStepChange(step, i)\"\n        (openExternal)=\"onNestedStepChange(step, i)\" (edit)=\"onNestedStepChange(step, i)\"\n        (link)=\"onNestedStepChange(step, i)\" (duplicate)=\"onNestedStepChange(step, i)\" (delete)=\"onDeleteStep(i)\"\n        (viewDetails)=\"viewDetails.emit($event)\"\n        (conditionChange)=\"$any(step).condition = $event; onNestedStepChange(step, i)\"\n        (branchStepChange)=\"onNestedStepChange(step, i)\" (addBranch)=\"onNestedConditionAddBranch($any(step), i)\"\n        (deleteBranch)=\"onNestedConditionDeleteBranch($any(step), $event, i)\"\n        (testDataProfileChange)=\"$any(step).testDataProfile = $event; onNestedStepChange(step, i)\"\n        (startStepChange)=\"$any(step).startStep = $event; onNestedStepChange(step, i)\"\n        (endStepChange)=\"$any(step).endStep = $event; onNestedStepChange(step, i)\"\n        (maxIterationsChange)=\"$any(step).maxIterations = $event; onNestedStepChange(step, i)\"\n        (eventTypeChange)=\"$any(step).eventType = $event; onNestedStepChange(step, i)\"\n        (parameterChange)=\"onNestedStepChange(step, i)\"\n        (selectionChange)=\"$any(step).selected = $event; onNestedStepChange(step, i)\"\n        (loadMoreDataProfiles)=\"loadMoreDataProfiles.emit($event)\"\n        (searchDataProfiles)=\"searchDataProfiles.emit($event)\"\n        (dndDropInZone)=\"dndDropInZone.emit($event)\"\n        (clickAction)=\"clickAction.emit($event)\"\n        (editInDepth)=\"editInDepth.emit($event)\"\n        (addStepBetweenClick)=\"addStepBetweenClick.emit($event)\">\n      </cqa-test-case-details-renderer>\n    </div>\n    <div *ngIf=\"isReorder\" class=\"cqa-flex cqa-flex-col nested-step-drop-list\"\n         [dndDropzone]=\"['step']\"\n         dndEffectAllowed=\"move\"\n         dndDragoverClass=\"dndDragover\"\n         (dndDrop)=\"onDndDrop($event)\">\n      <div dndPlaceholderRef class=\"step-drag-placeholder-nested cqa-my-1 cqa-min-h-[50px] cqa-border-2 cqa-border-dashed cqa-border-[#3F43EE] cqa-rounded cqa-bg-[rgba(63,67,238,0.08)] cqa-flex cqa-items-center cqa-justify-center cqa-text-[#3F43EE] cqa-text-xs\">Drop here</div>\n      <div *ngFor=\"let step of nestedSteps; let i = index\" class=\"nested-step-drag-item\"\n           [dndDraggable]=\"step\"\n           dndEffectAllowed=\"move\"\n           dndType=\"step\">\n        <cqa-test-case-details-renderer [step]=\"step\" [index]=\"i\"\n          [selected]=\"$any(step).selected\"\n          [isNested]=\"true\" [isInsideLoop]=\"true\" [isReorder]=\"isReorder\"\n          [addStepBetweenAbove]=\"addStepBetweenAbove\"\n          [addStepBetweenBelow]=\"addStepBetweenBelow\"\n          [addStepBetween]=\"addStepBetween\"\n          [dataProfileOptions]=\"dataProfileOptions\" [hasMoreDataProfiles]=\"hasMoreDataProfiles\" [isLoadingDataProfiles]=\"isLoadingDataProfiles\"\n          [naturalTextActionsOptions]=\"naturalTextActionsOptions\"\n          (nestedStepChange)=\"onNestedStepChange($event.step, $event.index)\"\n          (addStep)=\"onAddStep()\" (deleteStep)=\"onDeleteStep($event)\"\n          (toggleExpanded)=\"onNestedToggleExpanded($event, step, i)\"\n          (groupNameChange)=\"$any(step).groupName = $event; onNestedStepChange(step, i)\"\n          (descriptionChange)=\"$any(step).description = $event; onNestedStepChange(step, i)\"\n          (reusableChange)=\"$any(step).reusable = $event; onNestedStepChange(step, i)\"\n          (openExternal)=\"onNestedStepChange(step, i)\" (edit)=\"onNestedStepChange(step, i)\"\n          (link)=\"onNestedStepChange(step, i)\" (duplicate)=\"onNestedStepChange(step, i)\" (delete)=\"onDeleteStep(i)\"\n          (viewDetails)=\"viewDetails.emit($event)\"\n          (conditionChange)=\"$any(step).condition = $event; onNestedStepChange(step, i)\"\n          (branchStepChange)=\"onNestedStepChange(step, i)\" (addBranch)=\"onNestedConditionAddBranch($any(step), i)\"\n          (deleteBranch)=\"onNestedConditionDeleteBranch($any(step), $event, i)\"\n          (testDataProfileChange)=\"$any(step).testDataProfile = $event; onNestedStepChange(step, i)\"\n          (startStepChange)=\"$any(step).startStep = $event; onNestedStepChange(step, i)\"\n          (endStepChange)=\"$any(step).endStep = $event; onNestedStepChange(step, i)\"\n          (maxIterationsChange)=\"$any(step).maxIterations = $event; onNestedStepChange(step, i)\"\n          (eventTypeChange)=\"$any(step).eventType = $event; onNestedStepChange(step, i)\"\n          (parameterChange)=\"onNestedStepChange(step, i)\"\n          (selectionChange)=\"$any(step).selected = $event; onNestedStepChange(step, i)\"\n        (loadMoreDataProfiles)=\"loadMoreDataProfiles.emit($event)\"\n        (searchDataProfiles)=\"searchDataProfiles.emit($event)\"\n        (dndDropInZone)=\"dndDropInZone.emit($event)\"\n        (clickAction)=\"clickAction.emit($event)\"\n        (editInDepth)=\"editInDepth.emit($event)\"\n        (addStepBetweenClick)=\"addStepBetweenClick.emit($event)\">\n        </cqa-test-case-details-renderer>\n      </div>\n    </div>\n\n    <!-- END Marker -->\n    <div\n      [class]=\"'cqa-pl-4 cqa-py-1 cqa-bg-[#DBEAFE80] cqa-text-[#1C398E] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium'\"\n      style=\"border-top: 1px solid #BEDBFF;\">\n      {{ getEndLabel() }}\n    </div>\n  </div>\n</div>"]}
|
|
1016
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"loop-step.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/loop-step/loop-step.component.ts","../../../../../../src/lib/test-case-details/loop-step/loop-step.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, OnChanges, SimpleChanges, ViewChild, ElementRef, ChangeDetectorRef } from '@angular/core';\nimport { FormBuilder, FormGroup, FormControl } from '@angular/forms';\nimport { DndDropEvent } from 'ngx-drag-drop';\nimport { DynamicSelectFieldConfig, SelectOption } from '../../dynamic-select/dynamic-select-field.component';\nimport { LoopStepConfig, TestCaseStepConfig, NormalStepConfig, ConditionStepConfig, StepGroupConfig, ConditionBranch, StepTypes } from '../test-case-step.models';\nimport { STEP_ROW_ACTIONS_STYLES } from '../step-row-actions.styles';\nimport { DataProfileOption } from '../../step-builder/step-builder-loop/step-builder-loop.component';\nimport { ViewDetailsPayload } from '../step-details-modal/step-details-modal-data';\nimport { CqaAutocompleteOption } from '../../autocomplete/autocomplete.model';\n\n@Component({\n  selector: 'cqa-test-case-loop-step',\n  templateUrl: './loop-step.component.html',\n  host: { class: 'cqa-ui-root' },\n  styles: [\n    STEP_ROW_ACTIONS_STYLES,\n    `\n    .loop-step-empty {\n      opacity: 0;\n      transition: opacity 0.1s ease;\n    }\n    .loop-step-row:hover .loop-step-empty {\n      opacity: 1;\n    }\n    .loop-step-empty-add {\n      padding: 0;\n      background: transparent;\n      border: 0;\n      cursor: pointer;\n      color: #3F43EE;\n      transition: opacity 0.15s ease;\n    }\n    .loop-step-empty-add:hover {\n      opacity: 0.9;\n    }\n    `,\n  ],\n})\nexport class TestCaseLoopStepComponent implements OnInit, OnChanges {\n  @Input() config!: LoopStepConfig;\n  @Input() id!: string;\n  @Input() loopType: 'for' | 'while' = 'for';\n  @Input() stepNumber!: number | string;\n  @Input() condition?: string;\n  @Input() maxIterations?: number;\n  @Input() testDataProfile?: string;\n  @Input() startStep?: number;\n  @Input() endStep?: number;\n  @Input() nestedSteps: TestCaseStepConfig[] = [];\n  @Input() expanded: boolean = true;\n  @Input() isNested: boolean = false;\n  @Input() isInsideLoop: boolean = false;\n  @Input() isReorder: boolean = false;\n  @Input() selected: boolean = false;\n  /** Options for the data profile dropdown - accepts DataProfileOption objects */\n  @Input() dataProfileOptions: DataProfileOption[] = [];\n  /** Indicates if more data profiles are available for loading */\n  @Input() hasMoreDataProfiles: boolean = false;\n  /** Loading state for data profiles */\n  @Input() isLoadingDataProfiles: boolean = false;\n  /** Options for natural text actions (for while loop condition) */\n  @Input() naturalTextActionsOptions: SelectOption[] = [];\n  /** Function to process template variables for while condition (same as condition-step) */\n  @Input() setConditionTemplateVariables: (template: any) => any[] = () => [];\n  /** Show \"Add Step Between\" button above nested steps */\n  @Input() addStepBetweenAbove: boolean = false;\n  /** Show \"Add Step Between\" button below nested steps */\n  @Input() addStepBetweenBelow: boolean = false;\n  /** @deprecated Use addStepBetweenAbove and addStepBetweenBelow instead. If set, applies to both buttons. */\n  @Input() addStepBetween: boolean = false;\n\n  @Output() toggleExpanded = new EventEmitter<{config: LoopStepConfig, expanded: boolean}>();\n  @Output() testDataProfileChange = new EventEmitter<string>();\n  @Output() startStepChange = new EventEmitter<number>();\n  @Output() endStepChange = new EventEmitter<number>();\n  @Output() conditionChange = new EventEmitter<string>();\n  @Output() maxIterationsChange = new EventEmitter<number>();\n  @Output() nestedStepChange = new EventEmitter<{ step: TestCaseStepConfig; index: number }>();\n  /** Emits void when add is from nested content; emits { loopStepConfig } when empty-state add is clicked (same behaviour as add-step-between). */\n  @Output() addStep = new EventEmitter<void | { loopStepConfig: LoopStepConfig }>();\n  @Output() deleteStep = new EventEmitter<number>();\n  @Output() duplicate = new EventEmitter<void>();\n  @Output() delete = new EventEmitter<void>();\n  @Output() moreOptions = new EventEmitter<void>();\n  @Output() viewDetails = new EventEmitter<ViewDetailsPayload>();\n  @Output() editInDepth = new EventEmitter<{ step: TestCaseStepConfig; index: number }>();\n  @Output() edit = new EventEmitter<void>();\n  @Output() dndDropInZone = new EventEmitter<{ event: DndDropEvent; targetList: TestCaseStepConfig[] }>();\n  /** Emit when more data profiles need to be loaded */\n  @Output() loadMoreDataProfiles = new EventEmitter<string>();\n  /** Emit when data profile search query changes */\n  @Output() searchDataProfiles = new EventEmitter<string>();\n  /** Emit when step is updated (Apply button clicked in edit mode) */\n  @Output() stepUpdate = new EventEmitter<LoopStepConfig>();\n  /** Re-emit when a nested step (e.g. normal-step) emits clickAction (e.g. element/label click for popup) */\n  @Output() clickAction = new EventEmitter<{ event: MouseEvent; step: TestCaseStepConfig }>();\n  /** Re-emit addStepBetweenClick from nested renderers */\n  @Output() addStepBetweenClick = new EventEmitter<{step: TestCaseStepConfig, index: number, position: 'ABOVE' | 'BELOW'}>();\n  @Output() selectionChange = new EventEmitter<boolean>();\n\n  onDndDrop(event: DndDropEvent): void {\n    this.dndDropInZone.emit({ event, targetList: this.nestedSteps });\n  }\n\n  /** Expose global constructors for template (Angular templates don't have String/Number) */\n  readonly stringFn = String;\n  readonly numberFn = Number;\n\n  /** When true, header shows inline edit form (dropdowns, inputs, Cancel/Apply) */\n  isEditing: boolean = false;\n\n  /** Form for edit mode; bound to cqa-dynamic-select (same pattern as default stories) */\n  editForm!: FormGroup;\n\n  /** Cached select configs (stable refs to avoid change-detection loops) */\n  forOptionTypeSelectConfig!: DynamicSelectFieldConfig;\n  testDataProfileSelectConfig!: DynamicSelectFieldConfig;\n  startStepSelectConfig!: DynamicSelectFieldConfig;\n  endStepSelectConfig!: DynamicSelectFieldConfig;\n  conditionSelectConfig!: DynamicSelectFieldConfig;\n  /** While loop: config for autocomplete options (WHILE_CONDITION only) */\n  whileConditionSelectConfig!: DynamicSelectFieldConfig;\n\n  /** Selected template for while condition (same pattern as condition-step) */\n  selectedTemplate: any = null;\n  /** Template variables for while condition */\n  templateVariables: any[] = [];\n  /** Form group for template variables */\n  templateVariablesForm!: FormGroup;\n\n  /** Options for Loop Start / Loop End dropdowns (1-10) */\n  loopStepOptions: number[] = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];\n\n  @ViewChild('viewDetailsTrigger', { static: false }) viewDetailsTrigger?: ElementRef<HTMLElement>;\n\n  /** Local copy of values while editing (for Cancel revert) */\n  private editSnapshot: {\n    testDataProfile?: string;\n    startStep?: number;\n    endStep?: number;\n    condition?: string;\n    maxIterations?: number;\n    selectedTemplate?: any;\n    templateVariables?: any[];\n    conditionTemplateFormValue?: any;\n  } = {};\n\n  constructor(\n    private fb: FormBuilder,\n    private cdr: ChangeDetectorRef\n  ) {}\n\n  ngOnInit(): void {\n    if (this.config) {\n      this.id = this.config.id;\n      this.loopType = this.config.loopType;\n      this.stepNumber = this.config.stepNumber;\n      this.condition = this.config.condition;\n      this.maxIterations = this.config.maxIterations;\n      this.testDataProfile = this.config.testDataProfile;\n      this.startStep = this.config.startStep;\n      this.endStep = this.config.endStep;\n      this.nestedSteps = this.config.nestedSteps || [];\n      this.expanded = this.config.expanded !== undefined ? this.config.expanded : true;\n      this.selected = (this.config as any).selected || false;\n    }\n    this.buildEditForm();\n    this.buildSelectConfigs();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['config'] && this.config) {\n      this.selected = (this.config as any).selected || false;\n      if (this.loopType === 'while') {\n        this.rebuildWhileConditionFromNaturalTextAction();\n      }\n      // FOR loop: when step config has forLoopTestDataId/forLoopStartIndex/forLoopEndIndex, sync form and refresh select configs so dropdowns show selected\n      if (this.loopType === 'for') {\n        this.updateForLoopSelectConfigsSelectedValues();\n        if (this.editForm) this.syncForLoopEditFormFromConfig();\n        this.cdr.detectChanges();\n      }\n    }\n    if (changes['dataProfileOptions'] || changes['hasMoreDataProfiles'] || changes['isLoadingDataProfiles']) {\n      this.updateTestDataProfileSelectConfig();\n      // When data profiles load in edit mode, sync form value so dropdown shows selected (forLoopTestDataId or testDataProfile)\n      if (this.loopType === 'for' && this.editForm && this.dataProfileOptions?.length) {\n        const cfg = this.config as any;\n        if (cfg?.forLoopTestDataId != null || this.testDataProfile) this.syncForLoopEditFormFromConfig();\n        this.cdr.detectChanges();\n      }\n    }\n    if (changes['naturalTextActionsOptions']) {\n      this.updateConditionSelectConfig();\n      // When options load, populate condition + template from naturalTextActionId (same as condition-step)\n      if (this.loopType === 'while' && this.editForm && this.naturalTextActionsOptions?.length) {\n        this.rebuildWhileConditionFromNaturalTextAction();\n      }\n    }\n  }\n\n  /** Derive forOptionType for FOR loop from config so dropdown shows existing selection */\n  private getForOptionTypeFromConfig(): string | null {\n    if (this.loopType !== 'for') return null;\n    const cfg = this.config as any;\n    const hasTestDataId = cfg?.forLoopTestDataId != null || cfg?.dataMapJson?.forLoopTestDataId != null || cfg?.dataMapBean?.for_loop?.testDataId != null;\n    if (hasTestDataId || this.testDataProfile) return 'test-data';\n    const hasFixed = cfg?.forLoopStartIndex != null || cfg?.forLoopEndIndex != null\n      || cfg?.dataMapJson?.forLoopStartIndex != null || cfg?.dataMapJson?.forLoopEndIndex != null\n      || cfg?.dataMapBean?.for_loop?.startIndex != null || cfg?.dataMapBean?.for_loop?.endIndex != null;\n    if (this.startStep != null || this.endStep != null || hasFixed) return 'fixed';\n    return null;\n  }\n\n  /** Test Data Profile dropdown value: prefer forLoopTestDataId from config (or dataMapJson/dataMapBean). Use option.id when options loaded so type matches select. */\n  private getTestDataProfileFormValue(): string | number {\n    const cfg = this.config as any;\n    const idFromConfig = cfg?.forLoopTestDataId ?? cfg?.dataMapJson?.forLoopTestDataId ?? cfg?.dataMapBean?.for_loop?.testDataId;\n    // When we have options, resolve by id so form value matches option.id (strict equality in dynamic-select)\n    if (this.loopType === 'for' && this.dataProfileOptions?.length) {\n      if (idFromConfig != null) {\n        const match = this.dataProfileOptions.find(\n          o => o.id === idFromConfig || String(o.id) === String(idFromConfig)\n        );\n        if (match) return match.id;\n        return idFromConfig as string | number;\n      }\n      const value = this.testDataProfile ?? '';\n      if (value !== '') {\n        const match = this.dataProfileOptions.find(\n          o => o.name === value || String(o.id) === String(value)\n        );\n        if (match) return match.id;\n      }\n      return value as string | number;\n    }\n    if (idFromConfig != null) return idFromConfig as string | number;\n    return (this.testDataProfile ?? '') as string | number;\n  }\n\n  /** Loop Start dropdown value: prefer forLoopStartIndex from config (or dataMapJson/dataMapBean), else startStep */\n  private getLoopStartFormValue(): number | null {\n    const cfg = this.config as any;\n    const v = cfg?.forLoopStartIndex ?? cfg?.dataMapJson?.forLoopStartIndex ?? cfg?.dataMapBean?.for_loop?.startIndex;\n    if (v != null) return Number(v);\n    return this.startStep ?? null;\n  }\n\n  /** Loop End dropdown value: prefer forLoopEndIndex from config (or dataMapJson/dataMapBean), else endStep */\n  private getLoopEndFormValue(): number | null {\n    const cfg = this.config as any;\n    const v = cfg?.forLoopEndIndex ?? cfg?.dataMapJson?.forLoopEndIndex ?? cfg?.dataMapBean?.for_loop?.endIndex;\n    if (v != null) return Number(v);\n    return this.endStep ?? null;\n  }\n\n  private buildEditForm(): void {\n    // For while loop: if config has naturalTextActionId and options are already loaded, use them for initial condition + template\n    let conditionValue = this.condition ?? '';\n    if (this.loopType === 'while') {\n      const populated = this.populateWhileConditionFromNaturalTextActionId();\n      if (populated?.condition != null) {\n        conditionValue = populated.condition;\n      }\n    }\n    // For FOR loop: initial form values so dropdowns show existing selection (forOptionType, forLoopTestDataId/testDataProfile, forLoopStartIndex/forLoopEndIndex)\n    const forOptionType = this.getForOptionTypeFromConfig();\n    const testDataProfileValue = this.loopType === 'for' ? this.getTestDataProfileFormValue() : (this.testDataProfile ?? '');\n    const loopStartValue = this.loopType === 'for' ? this.getLoopStartFormValue() : (this.startStep ?? null);\n    const loopEndValue = this.loopType === 'for' ? this.getLoopEndFormValue() : (this.endStep ?? null);\n    this.editForm = this.fb.group({\n      forOptionType: [forOptionType as string | null],\n      testDataProfile: [testDataProfileValue],\n      startStep: [loopStartValue as number | null],\n      endStep: [loopEndValue as number | null],\n      condition: [conditionValue],\n      maxIterations: [this.maxIterations ?? 5],\n    });\n    if (!this.templateVariablesForm) {\n      this.templateVariablesForm = this.fb.group({});\n    }\n  }\n\n  /**\n   * Populate condition, selectedTemplate and template variables from config.naturalTextActionId when options are available.\n   * Returns { condition } if a match was found so buildEditForm can set initial form value.\n   */\n  private populateWhileConditionFromNaturalTextActionId(): { condition: string } | null {\n    const configWithId = this.config as any;\n    if (!configWithId?.naturalTextActionId || !this.naturalTextActionsOptions?.length) {\n      return null;\n    }\n    const matchingOption = (this.naturalTextActionsOptions as any[]).find(\n      (option: any) => String(option.id) === String(configWithId.naturalTextActionId)\n    );\n    if (!matchingOption) {\n      return null;\n    }\n    const naturalText = matchingOption.naturalText ?? matchingOption.name ?? matchingOption.value ?? '';\n    if (!naturalText) {\n      return null;\n    }\n    this.condition = naturalText;\n    const fullTemplate = matchingOption.template || matchingOption;\n    this.selectedTemplate = fullTemplate;\n    if (fullTemplate?.variables) {\n      this.templateVariables = this.setConditionTemplateVariables(fullTemplate) || [];\n      this.buildTemplateVariablesForm();\n    } else {\n      this.templateVariables = [];\n    }\n    this.cdr.detectChanges();\n    return { condition: naturalText };\n  }\n\n  /** Rebuild condition + template state when naturalTextActionsOptions load or config changes (same as condition-step). */\n  private rebuildWhileConditionFromNaturalTextAction(): void {\n    if (!this.editForm || this.loopType !== 'while') {\n      return;\n    }\n    const configWithId = this.config as any;\n    if (!configWithId?.naturalTextActionId || !this.naturalTextActionsOptions?.length) {\n      return;\n    }\n    const matchingOption = (this.naturalTextActionsOptions as any[]).find(\n      (option: any) => String(option.id) === String(configWithId.naturalTextActionId)\n    );\n    if (!matchingOption) {\n      return;\n    }\n    const naturalText = matchingOption.naturalText ?? matchingOption.name ?? matchingOption.value ?? '';\n    if (naturalText) {\n      const currentValue = this.editForm.get('condition')?.value ?? '';\n      if (!currentValue || currentValue !== naturalText) {\n        this.editForm.get('condition')?.setValue(naturalText);\n      }\n      this.condition = naturalText;\n    }\n    const fullTemplate = matchingOption.template || matchingOption;\n    this.selectedTemplate = fullTemplate;\n    if (fullTemplate?.variables) {\n      this.templateVariables = this.setConditionTemplateVariables(fullTemplate) || [];\n      this.buildTemplateVariablesForm();\n    } else {\n      this.templateVariables = [];\n    }\n    this.cdr.detectChanges();\n  }\n\n  /** Build select configs once (same option shape as default stories: id + name); FOR loop adds selectedValue from step */\n  private buildSelectConfigs(): void {\n    const stepOptions: SelectOption[] = this.loopStepOptions.map((n) => ({\n      id: n,\n      value: n,\n      name: String(n),\n      label: String(n),\n    }));\n    this.forOptionTypeSelectConfig = {\n      key: 'forOptionType',\n      placeholder: 'Select option',\n      searchable: false,\n      options: [\n        { id: 'test-data', value: 'test-data', name: 'Test Data Profile', label: 'Test Data Profile' },\n        { id: 'fixed', value: 'fixed', name: 'Fixed count', label: 'Fixed count' },\n      ],\n    };\n    this.updateTestDataProfileSelectConfig();\n    const startVal = this.loopType === 'for' ? this.getLoopStartFormValue() : null;\n    const endVal = this.loopType === 'for' ? this.getLoopEndFormValue() : null;\n    this.startStepSelectConfig = {\n      key: 'startStep',\n      placeholder: 'Loop Start',\n      searchable: false,\n      options: stepOptions.map((o) => ({ ...o })),\n      ...(startVal != null && { selectedValue: startVal }),\n    };\n    this.endStepSelectConfig = {\n      key: 'endStep',\n      placeholder: 'Loop End',\n      searchable: false,\n      options: stepOptions.map((o) => ({ ...o })),\n      ...(endVal != null && { selectedValue: endVal }),\n    };\n    this.updateConditionSelectConfig();\n  }\n\n  /** Update condition select config with natural text actions options (legacy mat-select) */\n  private updateConditionSelectConfig(): void {\n    this.conditionSelectConfig = {\n      key: 'condition',\n      placeholder: 'Select condition',\n      searchable: true,\n      options: this.naturalTextActionsOptions || [],\n    };\n    this.updateWhileConditionSelectConfig();\n  }\n\n  /** While loop: only show options with stepActionType === \"WHILE_LOOP\" */\n  private updateWhileConditionSelectConfig(): void {\n    const filteredOptions = (this.naturalTextActionsOptions || []).filter((option: any) => {\n      return option.stepActionType === 'WHILE_LOOP';\n    });\n    this.whileConditionSelectConfig = {\n      key: 'condition',\n      placeholder: 'Select condition',\n      searchable: true,\n      options: filteredOptions,\n    };\n  }\n\n  /** Options for the while condition autocomplete (WHILE_LOOP natural text actions only) */\n  get whileConditionAutocompleteOptions(): (CqaAutocompleteOption & { template?: any })[] {\n    const opts = this.whileConditionSelectConfig?.options ?? [];\n    return opts.map((option: any) => {\n      const naturalText = option.naturalText ?? option.name ?? option.value ?? String(option.id ?? '');\n      return {\n        value: naturalText,\n        label: naturalText,\n        template: option.template || option,\n      } as CqaAutocompleteOption & { template?: any };\n    });\n  }\n\n  onConditionSelect(option: CqaAutocompleteOption & { template?: any }): void {\n    this.editForm?.get('condition')?.setValue(option.value);\n    const matchingTemplate = option.template;\n    let fullTemplate = matchingTemplate;\n    if (!fullTemplate || !fullTemplate.variables) {\n      const matchingOption = (this.naturalTextActionsOptions || []).find(\n        (opt: any) => opt.naturalText === option.value || opt.name === option.value || opt.value === option.value\n      ) as any;\n      fullTemplate = matchingOption?.template || matchingOption;\n    }\n    if (fullTemplate && fullTemplate.variables) {\n      this.selectedTemplate = fullTemplate;\n      this.templateVariables = this.setConditionTemplateVariables(fullTemplate) || [];\n      this.buildTemplateVariablesForm();\n      this.cdr.detectChanges();\n    } else {\n      this.selectedTemplate = null;\n      this.templateVariables = [];\n      if (!this.templateVariablesForm) {\n        this.templateVariablesForm = this.fb.group({});\n      }\n      this.cdr.detectChanges();\n    }\n  }\n\n  private buildTemplateVariablesForm(): void {\n    if (!this.templateVariablesForm) {\n      this.templateVariablesForm = this.fb.group({});\n    }\n    Object.keys(this.templateVariablesForm.controls).forEach(key => {\n      this.templateVariablesForm.removeControl(key);\n    });\n    const configWithData = this.config as any;\n    const existingData = configWithData.templateVariablesData || {};\n    this.templateVariables.forEach(variable => {\n      let defaultValue: any;\n      if (existingData.hasOwnProperty(variable.name)) {\n        defaultValue = existingData[variable.name];\n      } else {\n        defaultValue = variable.type === 'boolean'\n          ? (variable.value === true || variable.value === 'true' || variable.value === 1)\n          : (variable.value || '');\n      }\n      if (variable.type === 'boolean' && typeof defaultValue === 'string') {\n        defaultValue = defaultValue === 'true' || defaultValue === '1';\n      }\n      this.templateVariablesForm.addControl(variable.name, new FormControl(defaultValue));\n    });\n  }\n\n  getSelectConfigForVariable(variable: any): DynamicSelectFieldConfig {\n    const options: SelectOption[] = (variable.options || []).map((opt: string) => ({\n      id: opt,\n      value: opt,\n      name: opt,\n      label: opt\n    }));\n    return {\n      key: variable.name,\n      placeholder: `Select ${variable.label}`,\n      multiple: false,\n      searchable: false,\n      options: options,\n      onChange: (value: any) => {\n        if (this.templateVariablesForm && this.templateVariablesForm.get(variable.name)) {\n          this.templateVariablesForm.get(variable.name)?.setValue(value);\n        }\n      }\n    };\n  }\n\n  /** Update test data profile select config with current options and selectedValue (forLoopTestDataId). Ensure selected id is in options so mat-select can display it. */\n  private updateTestDataProfileSelectConfig(): void {\n    let options = this.convertDataProfileOptionsToSelectOptions(this.dataProfileOptions);\n    const cfg = this.config as any;\n    const forLoopTestDataId = cfg?.forLoopTestDataId ?? cfg?.dataMapJson?.forLoopTestDataId ?? cfg?.dataMapBean?.for_loop?.testDataId;\n    const selectedVal = this.loopType === 'for' ? this.getTestDataProfileFormValue() : undefined;\n    // If we have a preselected id but it's not in the current options list (e.g. options not loaded yet), add a placeholder option so the dropdown can show the selection\n    if (this.loopType === 'for' && forLoopTestDataId != null) {\n      const idStr = String(forLoopTestDataId);\n      const alreadyInList = options.some((o: SelectOption) => String(o.id ?? o.value) === idStr);\n      if (!alreadyInList) {\n        const displayName = this.testDataProfile || 'Selected profile';\n        options = [{ id: forLoopTestDataId, value: forLoopTestDataId, name: displayName, label: displayName }, ...options];\n      }\n    }\n    this.testDataProfileSelectConfig = {\n      key: 'testDataProfile',\n      placeholder: 'Testdataprofile',\n      searchable: true,\n      serverSearch: true,\n      hasMore: this.hasMoreDataProfiles,\n      isLoading: this.isLoadingDataProfiles,\n      options,\n      ...(selectedVal !== undefined && selectedVal !== '' && { selectedValue: selectedVal }),\n      onChange: (value: any) => {\n        this.onDataProfileChange(value);\n      },\n      onSearch: (query: string) => {\n        this.onSearchDataProfiles(query);\n      },\n      onLoadMore: (query?: string) => {\n        this.onLoadMoreDataProfiles(query || '');\n      },\n    };\n  }\n\n  /** Update Loop Start / Loop End select configs with selectedValue (forLoopStartIndex, forLoopEndIndex) so dropdowns show selected */\n  private updateStartEndStepSelectConfigsSelectedValues(): void {\n    if (this.loopType !== 'for') return;\n    const startVal = this.getLoopStartFormValue();\n    const endVal = this.getLoopEndFormValue();\n    const stepOptions: SelectOption[] = this.loopStepOptions.map((n) => ({\n      id: n,\n      value: n,\n      name: String(n),\n      label: String(n),\n    }));\n    this.startStepSelectConfig = {\n      ...this.startStepSelectConfig,\n      options: stepOptions.map((o) => ({ ...o })),\n      ...(startVal != null && { selectedValue: startVal }),\n    };\n    this.endStepSelectConfig = {\n      ...this.endStepSelectConfig,\n      options: stepOptions.map((o) => ({ ...o })),\n      ...(endVal != null && { selectedValue: endVal }),\n    };\n  }\n\n  /** Update all FOR loop select configs (testDataProfile, startStep, endStep) with selectedValue from step so dropdowns show selected */\n  private updateForLoopSelectConfigsSelectedValues(): void {\n    if (this.loopType !== 'for') return;\n    this.updateTestDataProfileSelectConfig();\n    this.updateStartEndStepSelectConfigsSelectedValues();\n  }\n\n  /** Convert DataProfileOption[] to SelectOption[] */\n  private convertDataProfileOptionsToSelectOptions(options: DataProfileOption[]): SelectOption[] {\n    return options.map(option => ({\n      id: option.id,\n      value: option.id,\n      name: option.name,\n      label: option.name\n    }));\n  }\n\n  /** Handle data profile selection */\n  private onDataProfileChange(profileId: any): void {\n    this.onTestDataProfileChange(profileId);\n  }\n\n  /** Handle search for data profiles */\n  private onSearchDataProfiles(query: string): void {\n    this.searchDataProfiles.emit(query);\n  }\n\n  /** Handle load more data profiles */\n  private onLoadMoreDataProfiles(query: string): void {\n    this.loadMoreDataProfiles.emit(query);\n  }\n\n  onEditFormFieldChange(controlName: string, value: string | number): void {\n    this.editForm.get(controlName)?.setValue(value);\n  }\n\n  getLoopTypeLabel(): string {\n    return this.loopType === 'for' ? 'FOR LOOP' : 'WHILE LOOP';\n  }\n\n  isWhileLoop(): boolean {\n    return this.loopType === 'while';\n  }\n\n  isForLoop(): boolean {\n    return this.loopType === 'for';\n  }\n\n  getEndLabel(): string {\n    return this.loopType === 'for' ? 'END FOR' : 'END WHILE';\n  }\n\n  /** Resolve display start index: prefer forLoopStartIndex from step (or dataMapJson/dataMapBean), else startStep, else from nested steps */\n  private getDisplayStartIndex(): number | undefined {\n    const cfg = this.config as any;\n    const v = cfg?.forLoopStartIndex ?? cfg?.dataMapJson?.forLoopStartIndex ?? cfg?.dataMapBean?.for_loop?.startIndex;\n    if (v != null) return Number(v);\n    if (this.startStep != null) return this.startStep;\n    return this.nestedSteps[0]?.stepNumber as number | undefined;\n  }\n\n  /** Resolve display end index: prefer forLoopEndIndex from step (or dataMapJson/dataMapBean), else endStep, else from nested steps */\n  private getDisplayEndIndex(): number | undefined {\n    const cfg = this.config as any;\n    const v = cfg?.forLoopEndIndex ?? cfg?.dataMapJson?.forLoopEndIndex ?? cfg?.dataMapBean?.for_loop?.endIndex;\n    if (v != null) return Number(v);\n    if (this.endStep != null) return this.endStep;\n    const last = this.nestedSteps[this.nestedSteps.length - 1];\n    return last?.stepNumber as number | undefined ?? this.nestedSteps.length;\n  }\n\n  /** Resolve display profile name: map forLoopTestDataId to profile name from dataProfileOptions, else use testDataProfile */\n  private getDisplayTestDataProfileName(): string {\n    const cfg = this.config as any;\n    const id = cfg?.forLoopTestDataId ?? cfg?.dataMapJson?.forLoopTestDataId ?? cfg?.dataMapBean?.for_loop?.testDataId;\n    if (id != null && this.dataProfileOptions?.length) {\n      const match = this.dataProfileOptions.find(\n        o => o.id === id || String(o.id) === String(id)\n      );\n      if (match) return match.name;\n    }\n    return this.testDataProfile ?? '';\n  }\n\n  getStepsSummary(): string {\n    if (this.nestedSteps.length === 0) {\n      return '0 steps';\n    }\n    const start = this.getDisplayStartIndex() ?? 1;\n    const end = this.getDisplayEndIndex() ?? this.nestedSteps.length;\n    const count = this.nestedSteps.length;\n    return `Steps ${start}-${end} (${count} step${count !== 1 ? 's' : ''})`;\n  }\n\n  getStartStepLabel(): string {\n    const start = this.getDisplayStartIndex();\n    if (start != null) return `Start: ${start}`;\n    return `Start: ${this.nestedSteps[0]?.stepNumber ?? 1}`;\n  }\n\n  getEndStepLabel(): string {\n    const end = this.getDisplayEndIndex();\n    if (end != null) return `End: ${end}`;\n    const defaultEnd = this.nestedSteps[this.nestedSteps.length - 1]?.stepNumber ?? this.nestedSteps.length;\n    return `End: ${defaultEnd}`;\n  }\n\n  getTestDataProfileLabel(): string {\n    const name = this.getDisplayTestDataProfileName();\n    if (name) return `Test Data profile: ${name}`;\n    return 'Test Data profile';\n  }\n\n  getConditionDisplay(): string {\n    return this.condition || 'element \".notification-badge\" exists';\n  }\n\n  onToggleExpanded(): void {\n    this.expanded = !this.expanded;\n    this.toggleExpanded.emit({config:this.config, expanded:this.expanded});\n  }\n\n  onTestDataProfileChange(value: string): void {\n    this.testDataProfile = value;\n    this.testDataProfileChange.emit(value);\n  }\n\n  onStartStepChange(value: number): void {\n    this.startStep = value;\n    this.startStepChange.emit(value);\n  }\n\n  onEndStepChange(value: number): void {\n    this.endStep = value;\n    this.endStepChange.emit(value);\n  }\n\n  onConditionChange(value: string): void {\n    this.condition = value;\n    this.conditionChange.emit(value);\n  }\n\n  onMaxIterationsChange(value: number): void {\n    this.maxIterations = value;\n    this.maxIterationsChange.emit(value);\n  }\n\n  onNestedStepChange(step: TestCaseStepConfig, index: number): void {\n    this.nestedStepChange.emit({ step, index });\n  }\n\n  onNestedToggleExpanded(event: boolean | {config: TestCaseStepConfig, expanded: boolean}, step: TestCaseStepConfig, index: number): void {\n    // Handle both boolean and object formats\n    if (typeof event === 'object' && event !== null && 'expanded' in event) {\n      // Object format from step-group: {config, expanded}\n      (step as any).expanded = event.expanded;\n      // Check if the event's config is a step-group (not the current step)\n      // This ensures nested step-group events bubble up correctly even when condition step is inside loop step\n      if (this.isStepGroup(event.config)) {\n        // Re-emit the toggleExpanded event with the object format, preserving the original config\n        // Cast to satisfy TypeScript - the renderer accepts TestCaseStepConfig which includes StepGroupConfig\n        this.toggleExpanded.emit({config: event.config as any as LoopStepConfig, expanded: event.expanded});\n      }\n    } else {\n      // Boolean format from other step types\n      (step as any).expanded = event as boolean;\n    }\n    this.onNestedStepChange(step, index);\n  }\n\n  onAddStep(): void {\n    this.addStep.emit();\n  }\n\n  /** Called from empty-state plus only; same behaviour as add-step-between (parentId = loop step id, position = parent + 1). */\n  onAddStepEmpty(): void {\n    this.addStep.emit({ loopStepConfig: this.config });\n  }\n\n  onDeleteStep(payloadOrIndex: any, index?: number): void {\n    const idx = typeof index === 'number' ? index : payloadOrIndex;\n    const payload = typeof index === 'number' ? payloadOrIndex : undefined;\n    this.deleteStep.emit(idx);\n    if (payload?.step) this.delete.emit(payload);\n  }\n\n  /** Re-emit edit from nested step so the portal can open edit modal (same as root level). */\n  onNestedEdit(event: { step: TestCaseStepConfig; index: number }): void {\n    if (event?.step != null) this.edit.emit(event as any);\n  }\n\n  /** Re-emit duplicate from nested step so the portal can duplicate (same as root level). */\n  onNestedDuplicate(event: any, step: TestCaseStepConfig): void {\n    this.duplicate.emit((event && event.step != null) ? event : { step } as any);\n  }\n\n  isNormalStep(step: TestCaseStepConfig): step is NormalStepConfig {\n    return 'eventType' in step;\n  }\n\n  isLoopStep(step: TestCaseStepConfig): step is LoopStepConfig {\n    return 'loopType' in step;\n  }\n\n  isConditionStep(step: TestCaseStepConfig): step is ConditionStepConfig {\n    return 'condition' in step && 'branches' in step;\n  }\n\n  isStepGroup(step: TestCaseStepConfig): step is StepGroupConfig {\n    return 'groupName' in step;\n  }\n\n  onAddBranch(): void {\n    // Not used in loop step, but needed for recursive condition steps\n  }\n\n  onDeleteBranch(branch: any): void {\n    // Not used in loop step, but needed for recursive condition steps\n  }\n\n  onNestedConditionAddBranch(nestedStep: ConditionStepConfig, index: number): void {\n    // Add a new branch to the nested condition step\n    if (!nestedStep.branches) {\n      nestedStep.branches = [];\n    }\n    const newBranch: ConditionBranch = {\n      type: nestedStep.branches.length === 0 ? 'if' : 'else',\n      label: nestedStep.branches.length === 0 ? 'IF TRUE' : 'ELSE', \n      stepType: StepTypes.CONDITION_IF,\n      nestedSteps: []\n    };\n    nestedStep.branches.push(newBranch);\n    this.onNestedStepChange(nestedStep, index);\n  }\n\n  onNestedConditionDeleteBranch(nestedStep: ConditionStepConfig, branch: any, index: number): void {\n    // Delete a branch from the nested condition step\n    if (nestedStep.branches) {\n      const branchIndex = nestedStep.branches.indexOf(branch);\n      if (branchIndex > -1) {\n        nestedStep.branches.splice(branchIndex, 1);\n      }\n    }\n    this.onNestedStepChange(nestedStep, index);\n  }\n\n  onOpenExternal(): void {\n    // Not used in loop step, but needed for recursive step groups\n  }\n\n  /** Sync FOR loop edit form so dropdowns show existing config (forOptionType, forLoopTestDataId/testDataProfile, forLoopStartIndex/forLoopEndIndex) */\n  private syncForLoopEditFormFromConfig(): void {\n    if (!this.editForm || this.loopType !== 'for') return;\n    const testDataVal = this.getTestDataProfileFormValue();\n    const startVal = this.getLoopStartFormValue();\n    const endVal = this.getLoopEndFormValue();\n    this.editForm.patchValue({\n      forOptionType: this.getForOptionTypeFromConfig() ?? null,\n      testDataProfile: testDataVal,\n      startStep: startVal,\n      endStep: endVal,\n    });\n    // Explicitly set each control so mat-select picks up the value (patchValue can be ignored by child in some timing cases)\n    const testDataCtrl = this.editForm.get('testDataProfile');\n    const startCtrl = this.editForm.get('startStep');\n    const endCtrl = this.editForm.get('endStep');\n    if (testDataCtrl && testDataVal !== undefined && testDataVal !== '') testDataCtrl.setValue(testDataVal, { emitEvent: false });\n    if (startCtrl && startVal != null) startCtrl.setValue(startVal, { emitEvent: false });\n    if (endCtrl && endVal != null) endCtrl.setValue(endVal, { emitEvent: false });\n    this.cdr.detectChanges();\n  }\n\n  onEdit(): void {\n    this.editSnapshot = {\n      testDataProfile: this.testDataProfile,\n      startStep: this.startStep,\n      endStep: this.endStep,\n      condition: this.condition,\n      maxIterations: this.maxIterations,\n      selectedTemplate: this.selectedTemplate,\n      templateVariables: this.templateVariables ? [...this.templateVariables] : [],\n      conditionTemplateFormValue: this.templateVariablesForm ? { ...this.templateVariablesForm.value } : undefined,\n    };\n    // FOR loop: refresh select configs (so Test Data Profile options include selected id if needed) then sync form so dropdowns show selected\n    if (this.loopType === 'for') {\n      this.updateForLoopSelectConfigsSelectedValues();\n      this.syncForLoopEditFormFromConfig();\n      setTimeout(() => {\n        this.syncForLoopEditFormFromConfig();\n        this.cdr.detectChanges();\n      }, 0);\n    } else {\n      this.editForm.patchValue({\n        testDataProfile: this.testDataProfile ?? '',\n        startStep: this.startStep ?? null,\n        endStep: this.endStep ?? null,\n        condition: this.condition ?? '',\n        maxIterations: this.maxIterations ?? 5,\n      });\n    }\n    if (this.loopType === 'while') {\n      this.editForm.patchValue({\n        condition: this.condition ?? '',\n        maxIterations: this.maxIterations ?? 5,\n      });\n    }\n    // While loop: restore selected template and template variables from config when opening edit\n    if (this.loopType === 'while' && (this.condition || (this.config as any).templateVariablesData)) {\n      const configWithData = this.config as any;\n      const conditionStr = this.condition || configWithData.condition || '';\n      if (conditionStr && (this.naturalTextActionsOptions || []).length > 0) {\n        const matchingOption = (this.naturalTextActionsOptions as any[]).find(\n          (opt: any) => (opt.naturalText || opt.name || opt.value) === conditionStr\n        );\n        const fullTemplate = matchingOption?.template || matchingOption;\n        if (fullTemplate && fullTemplate.variables) {\n          this.selectedTemplate = fullTemplate;\n          this.templateVariables = this.setConditionTemplateVariables(fullTemplate) || [];\n          this.buildTemplateVariablesForm();\n        }\n      }\n    }\n    this.isEditing = true;\n    this.edit.emit();\n  }\n\n  onEditApply(): void {\n    const v = this.editForm.value;\n    const testDataProfileValue = v.testDataProfile ?? '';\n    this.startStep = v.startStep ?? undefined;\n    this.endStep = v.endStep ?? undefined;\n    this.condition = v.condition ?? '';\n    const maxVal = v.maxIterations;\n    this.maxIterations = typeof maxVal === 'number' ? maxVal : (maxVal != null && maxVal !== '' ? Number(maxVal) : undefined);\n    \n    // Find the selected data profile option to get both ID and name\n    let testDataProfileId: string | number | undefined;\n    let testDataProfileName: string = '';\n    \n    if (testDataProfileValue && this.dataProfileOptions.length > 0) {\n      // The form value might be ID (number) or name (string)\n      const selectedProfile = this.dataProfileOptions.find(\n        option => option.id === Number(testDataProfileValue) || option.name === testDataProfileValue || String(option.id) === String(testDataProfileValue)\n      );\n      if (selectedProfile) {\n        testDataProfileId = selectedProfile.id;\n        testDataProfileName = selectedProfile.name;\n        this.testDataProfile = testDataProfileName;\n      } else {\n        // If not found, use the value as-is (might be ID or name)\n        this.testDataProfile = testDataProfileValue;\n        testDataProfileId = Number(testDataProfileValue) || undefined;\n      }\n    } else {\n      this.testDataProfile = testDataProfileValue;\n    }\n    \n    // While loop: build templateVariablesData and naturalTextActionId from selected template (same as condition-step)\n    let templateVariablesData: any;\n    let naturalTextActionId: number | undefined;\n    if (this.loopType === 'while' && this.templateVariablesForm && this.templateVariables?.length) {\n      templateVariablesData = {};\n      this.templateVariables.forEach((variable: any) => {\n        const value = this.templateVariablesForm.get(variable.name)?.value;\n        if (value !== undefined && value !== null && value !== '') {\n          templateVariablesData[variable.name] = value;\n        }\n      });\n      if (this.selectedTemplate?.id != null) {\n        naturalTextActionId = this.selectedTemplate.id;\n      }\n    }\n    \n    // Update config with new values (include testDataProfileId, templateVariablesData, naturalTextActionId for while; forLoopTestDataId, forLoopStartIndex/forLoopEndIndex for for)\n    const updatedConfig: LoopStepConfig & { testDataProfileId?: string | number; templateVariablesData?: any; naturalTextActionId?: number } = {\n      ...this.config,\n      testDataProfile: this.testDataProfile,\n      testDataProfileId: testDataProfileId, // Add ID for API call\n      startStep: this.startStep,\n      endStep: this.endStep,\n      condition: this.condition,\n      maxIterations: this.maxIterations,\n      ...(this.loopType === 'while' && Object.keys(templateVariablesData || {}).length > 0 && { templateVariablesData }),\n      ...(this.loopType === 'while' && naturalTextActionId != null && { naturalTextActionId }),\n      ...(this.loopType === 'for' && testDataProfileId != null && { forLoopTestDataId: Number(testDataProfileId) }),\n      ...(this.loopType === 'for' && this.startStep != null && { forLoopStartIndex: this.startStep }),\n      ...(this.loopType === 'for' && this.endStep != null && { forLoopEndIndex: this.endStep }),\n    };\n    \n    // Emit individual change events\n    this.testDataProfileChange.emit(this.testDataProfile);\n    this.startStepChange.emit(this.startStep ?? 1);\n    this.endStepChange.emit(this.endStep ?? 1);\n    if (this.loopType === 'while') {\n      this.conditionChange.emit(this.condition);\n      this.maxIterationsChange.emit(this.maxIterations ?? 5);\n    }\n    \n    // Emit step update event with full config\n    console.log('Loop-step: Emitting stepUpdate event', updatedConfig);\n    this.stepUpdate.emit(updatedConfig);\n    \n    this.isEditing = false;\n  }\n\n  onEditCancel(): void {\n    this.testDataProfile = this.editSnapshot.testDataProfile;\n    this.startStep = this.editSnapshot.startStep;\n    this.endStep = this.editSnapshot.endStep;\n    this.condition = this.editSnapshot.condition;\n    this.maxIterations = this.editSnapshot.maxIterations;\n    this.selectedTemplate = this.editSnapshot.selectedTemplate;\n    this.templateVariables = this.editSnapshot.templateVariables || [];\n    if (this.templateVariablesForm && this.editSnapshot.conditionTemplateFormValue) {\n      this.templateVariablesForm.patchValue(this.editSnapshot.conditionTemplateFormValue);\n    }\n    this.isEditing = false;\n  }\n\n  onEditInDepth(): void {\n    this.moreOptions.emit();\n    const index = typeof this.stepNumber === 'number' ? this.stepNumber : (typeof this.stepNumber === 'string' ? parseInt(this.stepNumber, 10) : 0);\n    this.editInDepth.emit({ step: this.config as TestCaseStepConfig, index: isNaN(index) ? 0 : index });\n  }\n\n  onLink(): void {\n    // Not used in loop step, but needed for recursive step groups\n  }\n\n  onDuplicate(): void {\n    this.duplicate.emit();\n  }\n\n  onDelete(): void {\n    this.delete.emit();\n  }\n\n  onSelectionChange(checked: boolean): void {\n    this.selected = checked;\n    this.selectionChange.emit(checked);\n  }\n\n  onMoreOptions(): void {\n    this.moreOptions.emit();\n  }\n\n  onViewDetails(event?: MouseEvent): void {\n    if (event) {\n      event.preventDefault();\n      event.stopPropagation();\n    }\n    \n    if (!this.config.description || this.config.description.trim() === '') {\n      return;\n    }\n\n    if (event) {\n      const title = this.loopType === 'for' ? 'FOR LOOP Details' : 'WHILE LOOP Details';\n      this.viewDetails.emit({\n        event,\n        title,\n        description: this.config.description,\n      });\n    }\n  }\n}\n","<div class=\"loop-step-row cqa-flex cqa-flex-col\">\n  <!-- Inline Edit Mode: Frame 2147224910 - flex row, 1056×36, gap 12px, flex-none -->\n  <div *ngIf=\"isEditing\" class=\"cqa-py-2.5 cqa-px-4 cqa-flex cqa-flex-col cqa-gap-3\">\n    <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-justify-between\">\n      <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-flex-grow\">\n        <!-- Loop Type Label -->\n        <div class=\"cqa-font-semibold cqa-text-[#1C398E] cqa-text-[12px] cqa-leading-[15px]\">\n          {{ getLoopTypeLabel() }}\n        </div>\n\n        <!-- FOR LOOP inline edit: cqa-dynamic-select (same usage as default stories) -->\n        <ng-container *ngIf=\"loopType === 'for'\">\n          <!-- First select: width 216.25px, height 36px, gap 4px, opacity 1 -->\n          <cqa-dynamic-select [form]=\"editForm\" [config]=\"forOptionTypeSelectConfig\"\n            class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n          <cqa-dynamic-select [form]=\"editForm\" [config]=\"testDataProfileSelectConfig\"\n            class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n          <cqa-dynamic-select [form]=\"editForm\" [config]=\"startStepSelectConfig\"\n            class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n          <cqa-dynamic-select [form]=\"editForm\" [config]=\"endStepSelectConfig\"\n            class=\"cqa-w-full cqa-max-w-[216px]\"></cqa-dynamic-select>\n        </ng-container>\n\n        <!-- WHILE LOOP inline edit: cqa-autocomplete + template variables (same as condition-step) -->\n        <ng-container *ngIf=\"isWhileLoop()\">\n          <cqa-autocomplete\n            *ngIf=\"editForm\"\n            [options]=\"whileConditionAutocompleteOptions\"\n            [value]=\"editForm.get('condition')?.value ?? ''\"\n            (valueChange)=\"onEditFormFieldChange('condition', $event)\"\n            (optionSelect)=\"onConditionSelect($event)\"\n            placeholder=\"Select condition\"\n            [fullWidth]=\"true\"\n            class=\"cqa-w-full cqa-max-w-[216px]\">\n          </cqa-autocomplete>\n          <!-- Template variables (shown when template is selected, same as condition-step) -->\n          <div *ngIf=\"selectedTemplate && templateVariables && templateVariables.length > 0\" class=\"cqa-flex cqa-flex-row cqa-flex-wrap cqa-gap-3\">\n            <ng-container *ngFor=\"let variable of templateVariables\">\n              <ng-container *ngIf=\"variable.type === 'boolean'\">\n                <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n                  <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-gray-700\">{{ variable.label }}</label>\n                  <mat-slide-toggle\n                    [checked]=\"templateVariablesForm.get(variable.name)?.value || variable.value || false\"\n                    (change)=\"templateVariablesForm.get(variable.name)?.setValue($event.checked)\"\n                    color=\"primary\">\n                  </mat-slide-toggle>\n                </div>\n              </ng-container>\n              <ng-container *ngIf=\"variable.type !== 'boolean' && variable.name !== 'custom_code'\">\n                <ng-container *ngIf=\"variable.name === 'type' || variable.name === 'scrollTo' || variable.options\">\n                  <div class=\"cqa-flex cqa-flex-col\" style=\"min-width: 150px;\">\n                    <cqa-dynamic-select\n                      [form]=\"templateVariablesForm\"\n                      [config]=\"getSelectConfigForVariable(variable)\">\n                    </cqa-dynamic-select>\n                  </div>\n                </ng-container>\n                <ng-container *ngIf=\"variable.name !== 'type' && variable.name !== 'scrollTo' && !variable.options\">\n                  <div class=\"cqa-flex cqa-flex-col\" style=\"min-width: 150px;\">\n                    <cqa-custom-input\n                      [placeholder]=\"'Enter ' + variable.label\"\n                      [value]=\"templateVariablesForm.get(variable.name)?.value || variable.value || ''\"\n                      [fullWidth]=\"true\"\n                      (valueChange)=\"templateVariablesForm.get(variable.name)?.setValue($event)\">\n                    </cqa-custom-input>\n                  </div>\n                </ng-container>\n              </ng-container>\n            </ng-container>\n          </div>\n        </ng-container>\n\n        <!-- Edit In depth link -->\n      <a href=\"#\" (click)=\"onEditInDepth(); $event.preventDefault()\"\n        class=\"cqa-text-[#3F43EE] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-[2px] cqa-no-underline\">\n        Edit In depth\n        <svg width=\"8\" height=\"8\" viewBox=\"0 0 8 8\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n          <path\n            d=\"M2.03809 6.74329L2.62809 7.33329L5.96142 3.99996L2.62809 0.666626L2.03809 1.25663L4.78142 3.99996L2.03809 6.74329Z\"\n            fill=\"#3F43EE\" />\n        </svg>\n      </a>\n      </div>\n\n      <div class=\"cqa-flex cqa-items-center cqa-gap-1.5\">\n        <!-- Cancel / Apply -->\n      <cqa-button variant=\"outlined\" btnSize=\"lg\" [customClass]=\"'cqa-text-[14px] cqa-py-[9px]'\" [text]=\"'Cancel'\"\n        (clicked)=\"onEditCancel()\"></cqa-button>\n      <cqa-button variant=\"filled\" btnSize=\"lg\" [customClass]=\"'cqa-text-[14px] cqa-py-[9px]'\" [text]=\"'Apply'\"\n        (clicked)=\"onEditApply()\"></cqa-button>\n      </div>\n    </div>\n  </div>\n\n  <!-- Loop Header (normal view when not editing) -->\n  <div *ngIf=\"!isEditing\"\n    [class]=\"'step-row cqa-flex cqa-items-center cqa-gap-3 cqa-py-2 ' + (isInsideLoop ? 'cqa-pl-10 cqa-pr-4' : 'cqa-px-4')\">\n    <div class=\"cqa-inline-flex cqa-items-center\">\n    <!-- Drag Handle Icon (when isReorder is true) -->\n    <div *ngIf=\"isReorder\" class=\"cqa-mr-2 cqa-cursor-move cqa-text-[#6B7280] hover:cqa-text-[#111827]\">\n      <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n        <circle cx=\"3\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"8\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"13\" cy=\"3\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"3\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"8\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"13\" cy=\"8\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"3\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"8\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n        <circle cx=\"13\" cy=\"13\" r=\"1.5\" fill=\"currentColor\"/>\n      </svg>\n    </div>\n    <!-- Checkbox (when isReorder is false) -->\n    <label *ngIf=\"!isReorder\" class=\"cqa-flex cqa-items-center cqa-cursor-pointer cqa-relative cqa-mr-2\">\n      <input type=\"checkbox\"\n        [ngModel]=\"selected\"\n        (ngModelChange)=\"onSelectionChange($event)\"\n        class=\"cqa-h-4 cqa-w-4 cqa-cursor-pointer cqa-transition-all cqa-appearance-none cqa-rounded shadow hover:cqa-shadow-md cqa-border cqa-border-solid cqa-border-slate-300 cqa-flex-shrink-0\"\n        [class.cqa-bg-[#3F43EE]]=\"selected\"\n        [class.cqa-border-[#3F43EE]]=\"selected\"\n        id=\"check-loop-step\" />\n      <span class=\"cqa-absolute cqa-text-white cqa-top-1/2 cqa-left-1/2 cqa--translate-x-1/2 cqa--translate-y-1/2 cqa-pointer-events-none cqa-flex cqa-items-center cqa-justify-center\"\n        [class.cqa-opacity-0]=\"!selected\"\n        [class.cqa-opacity-100]=\"selected\">\n        <svg width=\"12\" height=\"13\" viewBox=\"0 0 12 13\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M10 3.125L4.5 8.625L2 6.125\" stroke=\"white\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/></svg>\n      </span>\n    </label>\n    </div>\n    <span class=\"cqa-text-[#6B7280] cqa-text-[14px] cqa-leading-[18px] cqa-min-w-[32px]\">{{ stepNumber }}</span>\n    <!-- Expand/Collapse Icon -->\n    <button type=\"button\" (click)=\"onToggleExpanded()\" class=\"cqa-p-0\">\n      <svg [class.cqa-rotate-180]=\"!expanded\" class=\"cqa-transition-transform\" width=\"16\" height=\"16\"\n        viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n        <path d=\"M12 10L8 6L4 10\" stroke=\"#6B7280\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n          stroke-linejoin=\"round\" />\n      </svg>\n    </button>\n\n    <!-- Step Number -->\n    <span class=\"cqa-text-[#6B7280] cqa-text-[14px] cqa-leading-[18px] cqa-min-w-[32px] cqa-flex-shrink-0\">{{ stepNumber }}</span>\n\n    <!-- Loop Icon -->\n    <div *ngIf=\"loopType === 'for'\">\n      <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n        <path d=\"M11.334 1.33325L14.0007 3.99992L11.334 6.66659\" stroke=\"#1C398E\" stroke-width=\"1.33333\"\n          stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        <path d=\"M2 7.33333V6.66667C2 5.95942 2.28095 5.28115 2.78105 4.78105C3.28115 4.28095 3.95942 4 4.66667 4H14\"\n          stroke=\"#1C398E\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        <path d=\"M4.66667 14.6666L2 11.9999L4.66667 9.33325\" stroke=\"#1C398E\" stroke-width=\"1.33333\"\n          stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        <path\n          d=\"M14 8.66675V9.33341C14 10.0407 13.719 10.7189 13.219 11.219C12.7189 11.7191 12.0406 12.0001 11.3333 12.0001H2\"\n          stroke=\"#1C398E\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n      </svg>\n    </div>\n    <div *ngIf=\"loopType === 'while'\">\n      <svg width=\"16\" height=\"16\" viewBox=\"0 0 16 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n        <path d=\"M11.334 1.33325L14.0007 3.99992L11.334 6.66659\" stroke=\"#59168B\" stroke-width=\"1.33333\"\n          stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        <path d=\"M2 7.33333V6.66667C2 5.95942 2.28095 5.28115 2.78105 4.78105C3.28115 4.28095 3.95942 4 4.66667 4H14\"\n          stroke=\"#59168B\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        <path d=\"M4.66667 14.6666L2 11.9999L4.66667 9.33325\" stroke=\"#59168B\" stroke-width=\"1.33333\"\n          stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        <path\n          d=\"M14 8.66675V9.33341C14 10.0407 13.719 10.7189 13.219 11.219C12.7189 11.7191 12.0406 12.0001 11.3333 12.0001H2\"\n          stroke=\"#59168B\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n      </svg>\n    </div>\n\n    <!-- Loop Type Label -->\n    <span class=\"cqa-font-semibold cqa-text-[#1C398E] cqa-text-[12px] cqa-leading-none\">\n      {{ getLoopTypeLabel() }}\n    </span>\n\n    <!-- For Loop Controls -->\n    <div *ngIf=\"loopType === 'for'\" class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-ml-2\">\n      <!-- Test Data Profile Button -->\n      <button type=\"button\" [class.cqa-bg-primary-100]=\"testDataProfile\" [class.cqa-text-primary]=\"testDataProfile\"\n        class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n        {{ getTestDataProfileLabel() }}\n      </button>\n\n      <!-- Start Button -->\n      <button type=\"button\" [class.cqa-bg-primary-100]=\"startStep\" [class.cqa-text-primary]=\"startStep\"\n        class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n        {{ getStartStepLabel() }}\n      </button>\n\n      <!-- End Button -->\n      <button type=\"button\" [class.cqa-bg-primary-100]=\"endStep\" [class.cqa-text-primary]=\"endStep\"\n        class=\"cqa-py-0.5 cqa-px-2 cqa-text-[#3F43EE] cqa-text-[14px] cqa-leading-[17px] cqa-font-semibold cqa-border cqa-border-solid cqa-border-[#8A8CF4] cqa-rounded cqa-bg-[#D8D9FC]\">\n        {{ getEndStepLabel() }}\n      </button>\n    </div>\n\n    <!-- While Loop Condition (only show if condition exists) -->\n    <div *ngIf=\"isWhileLoop() && condition\" class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-ml-2 cqa-flex-1\">\n      <span class=\"cqa-text-gray-700 cqa-text-sm\">\n        While\n      </span>\n      <span class=\"cqa-text-gray-900 cqa-text-sm\" [innerHTML]=\"getConditionDisplay()\"></span>\n    </div>\n\n    <!-- Steps Summary -->\n    <div\n      class=\"cqa-ml-auto cqa-border cqa-border-solid cqa-border-[#E5E5E5] cqa-rounded-lg cqa-py-0.5 cqa-px-2 cqa-text-[#0A0A0A] cqa-text-[12px] cqa-leading-[15px]\">\n      {{ getStepsSummary() }}\n    </div>\n\n    <div class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-ml-auto\">\n      <!-- View Details Link (show only if description is not empty) -->\n      <a *ngIf=\"config.description && config.description.trim() !== ''\" #viewDetailsTrigger href=\"#\" (click)=\"onViewDetails($event)\" class=\"cqa-text-[#3F43EE] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-2 cqa-no-underline cqa-ml-auto\">View Details<svg width=\"8\" height=\"8\" viewBox=\"0 0 8 8\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M2.03809 6.74329L2.62809 7.33329L5.96142 3.99996L2.62809 0.666626L2.03809 1.25663L4.78142 3.99996L2.03809 6.74329Z\" fill=\"#3F43EE\"/></svg></a>\n\n      <!-- Action Icons: Edit, Link, Duplicate, Delete (show on hover) -->\n      <div class=\"step-actions cqa-flex cqa-items-center cqa-gap-3 cqa-px-[7px]\">\n        <button type=\"button\" (click)=\"onEdit(); $event.stopPropagation()\" title=\"Edit\"\n          class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n          <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n            <path d=\"M7 11.6666H12.25\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n              stroke-linejoin=\"round\" />\n            <path\n              d=\"M9.55208 2.1128C9.7843 1.88058 10.0993 1.75012 10.4277 1.75012C10.7561 1.75012 11.071 1.88058 11.3033 2.1128C11.5355 2.34502 11.6659 2.65998 11.6659 2.98838C11.6659 3.31679 11.5355 3.63175 11.3033 3.86397L4.29742 10.8704C4.15864 11.0092 3.9871 11.1107 3.79867 11.1656L2.12333 11.6544C2.07314 11.669 2.01993 11.6699 1.96928 11.6569C1.91863 11.6439 1.8724 11.6176 1.83543 11.5806C1.79846 11.5437 1.7721 11.4974 1.75913 11.4468C1.74615 11.3961 1.74703 11.3429 1.76167 11.2927L2.2505 9.61738C2.30546 9.42916 2.40698 9.25783 2.54567 9.11922L9.55208 2.1128Z\"\n              stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n          </svg>\n        </button>\n        <!-- <button type=\"button\" (click)=\"onLink(); $event.stopPropagation()\" title=\"Link\"\n          class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n          <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n            <path\n              d=\"M5.00065 9.91671H3.66732C2.78326 9.91671 1.93542 9.60942 1.3103 9.06244C0.685174 8.51545 0.333984 7.77359 0.333984 7.00004C0.333984 6.22649 0.685174 5.48463 1.3103 4.93765C1.93542 4.39066 2.78326 4.08337 3.66732 4.08337H5.00065\"\n              stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n            <path\n              d=\"M9 4.08337H10.3333C11.2174 4.08337 12.0652 4.39066 12.6904 4.93765C13.3155 5.48463 13.6667 6.22649 13.6667 7.00004C13.6667 7.77359 13.3155 8.51545 12.6904 9.06244C12.0652 9.60942 11.2174 9.91671 10.3333 9.91671H9\"\n              stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n            <path d=\"M4.33398 7H9.66732\" stroke=\"currentColor\" stroke-width=\"1.33333\" stroke-linecap=\"round\"\n              stroke-linejoin=\"round\" />\n          </svg>\n        </button> -->\n        <button type=\"button\" (click)=\"onDuplicate(); $event.stopPropagation()\" title=\"Duplicate\"\n          class=\"cqa-p-0 cqa-text-[#99A1Af] hover:cqa-text-[#1447E6]\">\n          <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n            <path\n              d=\"M11.666 4.66663H5.83268C5.18835 4.66663 4.66602 5.18896 4.66602 5.83329V11.6666C4.66602 12.311 5.18835 12.8333 5.83268 12.8333H11.666C12.3103 12.8333 12.8327 12.311 12.8327 11.6666V5.83329C12.8327 5.18896 12.3103 4.66663 11.666 4.66663Z\"\n              stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n            <path\n              d=\"M2.33268 9.33329C1.69102 9.33329 1.16602 8.80829 1.16602 8.16663V2.33329C1.16602 1.69163 1.69102 1.16663 2.33268 1.16663H8.16602C8.80768 1.16663 9.33268 1.69163 9.33268 2.33329\"\n              stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n          </svg>\n        </button>\n        <button type=\"button\" (click)=\"onDelete(); $event.stopPropagation()\" title=\"Delete\"\n          class=\"cqa-p-0 cqa-text-[#ff6467] hover:cqa-text-[#C63535]\">\n          <svg width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n            <path d=\"M1.75 3.5H12.25\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n              stroke-linejoin=\"round\" />\n            <path\n              d=\"M11.0827 3.5V11.6667C11.0827 12.25 10.4993 12.8333 9.91602 12.8333H4.08268C3.49935 12.8333 2.91602 12.25 2.91602 11.6667V3.5\"\n              stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n            <path\n              d=\"M4.66602 3.49996V2.33329C4.66602 1.74996 5.24935 1.16663 5.83268 1.16663H8.16602C8.74935 1.16663 9.33268 1.74996 9.33268 2.33329V3.49996\"\n              stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n            <path d=\"M5.83398 6.41663V9.91663\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n              stroke-linejoin=\"round\" />\n            <path d=\"M8.16602 6.41663V9.91663\" stroke=\"currentColor\" stroke-width=\"1.16667\" stroke-linecap=\"round\"\n              stroke-linejoin=\"round\" />\n          </svg>\n        </button>\n      </div>\n    </div>\n  </div>\n\n  <!-- Expanded Content with Nested Steps (renderer dispatches by step type, n-level nesting) -->\n  <div *ngIf=\"expanded\" class=\"cqa-flex cqa-flex-col\">\n    <div *ngIf=\"!isReorder\" class=\"cqa-flex cqa-flex-col\">\n      <cqa-test-case-details-renderer *ngFor=\"let step of nestedSteps; let i = index\" [step]=\"step\" [index]=\"i\"\n        [selected]=\"$any(step).selected\"\n        [isNested]=\"true\" [isInsideLoop]=\"true\" [isReorder]=\"isReorder\"\n        [addStepBetween]=\"addStepBetween\"\n        [dataProfileOptions]=\"dataProfileOptions\" [hasMoreDataProfiles]=\"hasMoreDataProfiles\" [isLoadingDataProfiles]=\"isLoadingDataProfiles\"\n        [naturalTextActionsOptions]=\"naturalTextActionsOptions\"\n        (nestedStepChange)=\"onNestedStepChange($event.step, $event.index)\"\n        (addStep)=\"onAddStep()\" (deleteStep)=\"onDeleteStep($event)\"\n        (toggleExpanded)=\"onNestedToggleExpanded($event, step, i)\"\n        (groupNameChange)=\"$any(step).groupName = $event; onNestedStepChange(step, i)\"\n        (descriptionChange)=\"$any(step).description = $event; onNestedStepChange(step, i)\"\n        (reusableChange)=\"$any(step).reusable = $event; onNestedStepChange(step, i)\"\n        (openExternal)=\"onNestedStepChange(step, i)\" (edit)=\"onNestedEdit($event)\"\n        (link)=\"onNestedStepChange(step, i)\" (duplicate)=\"onNestedDuplicate($event, step)\" (delete)=\"onDeleteStep($event, i)\"\n        (viewDetails)=\"viewDetails.emit($event)\"\n        (conditionChange)=\"$any(step).condition = $event; onNestedStepChange(step, i)\"\n        (branchStepChange)=\"onNestedStepChange(step, i)\" (addBranch)=\"onNestedConditionAddBranch($any(step), i)\"\n        (deleteBranch)=\"onNestedConditionDeleteBranch($any(step), $event, i)\"\n        (testDataProfileChange)=\"$any(step).testDataProfile = $event; onNestedStepChange(step, i)\"\n        (startStepChange)=\"$any(step).startStep = $event; onNestedStepChange(step, i)\"\n        (endStepChange)=\"$any(step).endStep = $event; onNestedStepChange(step, i)\"\n        (maxIterationsChange)=\"$any(step).maxIterations = $event; onNestedStepChange(step, i)\"\n        (eventTypeChange)=\"$any(step).eventType = $event; onNestedStepChange(step, i)\"\n        (parameterChange)=\"onNestedStepChange(step, i)\"\n        (selectionChange)=\"$any(step).selected = $event; onNestedStepChange(step, i)\"\n        (loadMoreDataProfiles)=\"loadMoreDataProfiles.emit($event)\"\n        (searchDataProfiles)=\"searchDataProfiles.emit($event)\"\n        (dndDropInZone)=\"dndDropInZone.emit($event)\"\n        (clickAction)=\"clickAction.emit($event)\"\n        (editInDepth)=\"editInDepth.emit($event)\"\n        (addStepBetweenClick)=\"addStepBetweenClick.emit($event)\">\n      </cqa-test-case-details-renderer>\n    </div>\n    <div *ngIf=\"isReorder\" class=\"cqa-flex cqa-flex-col nested-step-drop-list\"\n         [dndDropzone]=\"['step']\"\n         dndEffectAllowed=\"move\"\n         dndDragoverClass=\"dndDragover\"\n         (dndDrop)=\"onDndDrop($event)\">\n      <div dndPlaceholderRef class=\"step-drag-placeholder-nested cqa-my-1 cqa-min-h-[50px] cqa-border-2 cqa-border-dashed cqa-border-[#3F43EE] cqa-rounded cqa-bg-[rgba(63,67,238,0.08)] cqa-flex cqa-items-center cqa-justify-center cqa-text-[#3F43EE] cqa-text-xs\">Drop here</div>\n      <div *ngFor=\"let step of nestedSteps; let i = index\" class=\"nested-step-drag-item\"\n           [dndDraggable]=\"step\"\n           dndEffectAllowed=\"move\"\n           dndType=\"step\">\n        <cqa-test-case-details-renderer [step]=\"step\" [index]=\"i\"\n          [selected]=\"$any(step).selected\"\n          [isNested]=\"true\" [isInsideLoop]=\"true\" [isReorder]=\"isReorder\"\n          [addStepBetweenAbove]=\"addStepBetweenAbove\"\n          [addStepBetweenBelow]=\"addStepBetweenBelow\"\n          [addStepBetween]=\"addStepBetween\"\n          [dataProfileOptions]=\"dataProfileOptions\" [hasMoreDataProfiles]=\"hasMoreDataProfiles\" [isLoadingDataProfiles]=\"isLoadingDataProfiles\"\n          [naturalTextActionsOptions]=\"naturalTextActionsOptions\"\n          (nestedStepChange)=\"onNestedStepChange($event.step, $event.index)\"\n          (addStep)=\"onAddStep()\" (deleteStep)=\"onDeleteStep($event)\"\n          (toggleExpanded)=\"onNestedToggleExpanded($event, step, i)\"\n          (groupNameChange)=\"$any(step).groupName = $event; onNestedStepChange(step, i)\"\n          (descriptionChange)=\"$any(step).description = $event; onNestedStepChange(step, i)\"\n          (reusableChange)=\"$any(step).reusable = $event; onNestedStepChange(step, i)\"\n          (openExternal)=\"onNestedStepChange(step, i)\" (edit)=\"onNestedStepChange(step, i)\"\n          (link)=\"onNestedStepChange(step, i)\" (duplicate)=\"onNestedStepChange(step, i)\" (delete)=\"onDeleteStep($event, i)\"\n          (viewDetails)=\"viewDetails.emit($event)\"\n          (conditionChange)=\"$any(step).condition = $event; onNestedStepChange(step, i)\"\n          (branchStepChange)=\"onNestedStepChange(step, i)\" (addBranch)=\"onNestedConditionAddBranch($any(step), i)\"\n          (deleteBranch)=\"onNestedConditionDeleteBranch($any(step), $event, i)\"\n          (testDataProfileChange)=\"$any(step).testDataProfile = $event; onNestedStepChange(step, i)\"\n          (startStepChange)=\"$any(step).startStep = $event; onNestedStepChange(step, i)\"\n          (endStepChange)=\"$any(step).endStep = $event; onNestedStepChange(step, i)\"\n          (maxIterationsChange)=\"$any(step).maxIterations = $event; onNestedStepChange(step, i)\"\n          (eventTypeChange)=\"$any(step).eventType = $event; onNestedStepChange(step, i)\"\n          (parameterChange)=\"onNestedStepChange(step, i)\"\n          (selectionChange)=\"$any(step).selected = $event; onNestedStepChange(step, i)\"\n        (loadMoreDataProfiles)=\"loadMoreDataProfiles.emit($event)\"\n        (searchDataProfiles)=\"searchDataProfiles.emit($event)\"\n        (dndDropInZone)=\"dndDropInZone.emit($event)\"\n        (clickAction)=\"clickAction.emit($event)\"\n        (editInDepth)=\"editInDepth.emit($event)\"\n        (addStepBetweenClick)=\"addStepBetweenClick.emit($event)\">\n        </cqa-test-case-details-renderer>\n      </div>\n    </div>\n\n    <!-- Empty loop: show plus icon on hover when expanded and no steps -->\n    <div *ngIf=\"nestedSteps.length === 0\" class=\"loop-step-empty cqa-min-h-[52px] cqa-flex cqa-items-center cqa-justify-center cqa-pl-4 cqa-pr-4\">\n      <button type=\"button\" (click)=\"onAddStepEmpty(); $event.stopPropagation()\" title=\"Add step\"\n        class=\"loop-step-empty-add\">\n        <svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n          <rect x=\"1\" y=\"1\" width=\"21.1822\" height=\"22\" rx=\"10.5911\" fill=\"white\" />\n          <rect x=\"1\" y=\"1\" width=\"21.1822\" height=\"22\" rx=\"10.5911\" stroke=\"#3F43EE\" stroke-width=\"2\" />\n          <path d=\"M8.5 12H15.5\" stroke=\"#3F43EE\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n          <path d=\"M12 8.5V15.5\" stroke=\"#3F43EE\" stroke-linecap=\"round\" stroke-linejoin=\"round\" />\n        </svg>\n      </button>\n    </div>\n\n    <!-- END Marker -->\n    <div\n      [class]=\"'cqa-pl-4 cqa-py-1 cqa-bg-[#DBEAFE80] cqa-text-[#1C398E] cqa-text-[10px] cqa-leading-[15px] cqa-font-medium'\"\n      style=\"border-top: 1px solid #BEDBFF;\">\n      {{ getEndLabel() }}\n    </div>\n  </div>\n</div>"]}
|