@cqa-lib/cqa-ui 1.1.158 → 1.1.160
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/step-builder/step-builder-action/step-builder-action.component.mjs +119 -0
- package/esm2020/lib/table-action-toolbar/table-action-toolbar.component.mjs +27 -3
- package/esm2020/lib/templates/table-template.component.mjs +1 -1
- package/esm2020/lib/test-case-details/ai-agent-step/ai-agent-step.component.mjs +73 -0
- package/esm2020/lib/test-case-details/api-step/api-step.component.mjs +107 -0
- package/esm2020/lib/test-case-details/condition-step/condition-step.component.mjs +292 -0
- package/esm2020/lib/test-case-details/custom-code-step/custom-code-step.component.mjs +84 -0
- package/esm2020/lib/test-case-details/database-step/database-step.component.mjs +95 -0
- package/esm2020/lib/test-case-details/loop-step/loop-step.component.mjs +328 -0
- package/esm2020/lib/test-case-details/normal-step/normal-step.component.mjs +202 -0
- package/esm2020/lib/test-case-details/screenshot-step/screenshot-step.component.mjs +87 -0
- package/esm2020/lib/test-case-details/step-group/step-group.component.mjs +247 -0
- package/esm2020/lib/test-case-details/step-row-actions.styles.mjs +15 -0
- package/esm2020/lib/test-case-details/test-case-details-renderer/test-case-details-renderer.component.mjs +224 -0
- package/esm2020/lib/test-case-details/test-case-step-components.token.mjs +3 -0
- package/esm2020/lib/test-case-details/test-case-step.models.mjs +51 -0
- package/esm2020/lib/test-case-details/upload-step/upload-step.component.mjs +85 -0
- package/esm2020/lib/ui-kit.module.mjs +96 -5
- package/esm2020/public-api.mjs +14 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +2265 -208
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +2252 -208
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/step-builder/step-builder-action/step-builder-action.component.d.ts +59 -0
- package/lib/table-action-toolbar/table-action-toolbar.component.d.ts +11 -1
- package/lib/test-case-details/ai-agent-step/ai-agent-step.component.d.ts +30 -0
- package/lib/test-case-details/api-step/api-step.component.d.ts +42 -0
- package/lib/test-case-details/condition-step/condition-step.component.d.ts +87 -0
- package/lib/test-case-details/custom-code-step/custom-code-step.component.d.ts +33 -0
- package/lib/test-case-details/database-step/database-step.component.d.ts +41 -0
- package/lib/test-case-details/loop-step/loop-step.component.d.ts +96 -0
- package/lib/test-case-details/normal-step/normal-step.component.d.ts +44 -0
- package/lib/test-case-details/screenshot-step/screenshot-step.component.d.ts +33 -0
- package/lib/test-case-details/step-group/step-group.component.d.ts +70 -0
- package/lib/test-case-details/step-row-actions.styles.d.ts +6 -0
- package/lib/test-case-details/test-case-details-renderer/test-case-details-renderer.component.d.ts +67 -0
- package/lib/test-case-details/test-case-step-components.token.d.ts +15 -0
- package/lib/test-case-details/test-case-step.models.d.ts +133 -0
- package/lib/test-case-details/upload-step/upload-step.component.d.ts +33 -0
- package/lib/ui-kit.module.d.ts +31 -19
- package/package.json +1 -1
- package/public-api.d.ts +13 -0
- package/styles.css +1 -1
|
@@ -0,0 +1,328 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
2
|
+
import { StepTypes } from '../test-case-step.models';
|
|
3
|
+
import { STEP_ROW_ACTIONS_STYLES } from '../step-row-actions.styles';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@angular/forms";
|
|
6
|
+
import * as i2 from "../../dynamic-select/dynamic-select-field.component";
|
|
7
|
+
import * as i3 from "../../custom-input/custom-input.component";
|
|
8
|
+
import * as i4 from "../../button/button.component";
|
|
9
|
+
import * as i5 from "../test-case-details-renderer/test-case-details-renderer.component";
|
|
10
|
+
import * as i6 from "@angular/common";
|
|
11
|
+
export class TestCaseLoopStepComponent {
|
|
12
|
+
constructor(fb) {
|
|
13
|
+
this.fb = fb;
|
|
14
|
+
this.nestedSteps = [];
|
|
15
|
+
this.expanded = true;
|
|
16
|
+
this.isNested = false;
|
|
17
|
+
this.isInsideLoop = false;
|
|
18
|
+
this.toggleExpanded = new EventEmitter();
|
|
19
|
+
this.testDataProfileChange = new EventEmitter();
|
|
20
|
+
this.startStepChange = new EventEmitter();
|
|
21
|
+
this.endStepChange = new EventEmitter();
|
|
22
|
+
this.conditionChange = new EventEmitter();
|
|
23
|
+
this.maxIterationsChange = new EventEmitter();
|
|
24
|
+
this.nestedStepChange = new EventEmitter();
|
|
25
|
+
this.addStep = new EventEmitter();
|
|
26
|
+
this.deleteStep = new EventEmitter();
|
|
27
|
+
this.duplicate = new EventEmitter();
|
|
28
|
+
this.delete = new EventEmitter();
|
|
29
|
+
this.moreOptions = new EventEmitter();
|
|
30
|
+
this.edit = new EventEmitter();
|
|
31
|
+
/** Expose global constructors for template (Angular templates don't have String/Number) */
|
|
32
|
+
this.stringFn = String;
|
|
33
|
+
this.numberFn = Number;
|
|
34
|
+
/** When true, header shows inline edit form (dropdowns, inputs, Cancel/Apply) */
|
|
35
|
+
this.isEditing = false;
|
|
36
|
+
/** Options for Loop Start / Loop End dropdowns (1-10) */
|
|
37
|
+
this.loopStepOptions = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
|
|
38
|
+
/** Local copy of values while editing (for Cancel revert) */
|
|
39
|
+
this.editSnapshot = {};
|
|
40
|
+
}
|
|
41
|
+
ngOnInit() {
|
|
42
|
+
if (this.config) {
|
|
43
|
+
this.id = this.config.id;
|
|
44
|
+
this.loopType = this.config.loopType;
|
|
45
|
+
this.stepNumber = this.config.stepNumber;
|
|
46
|
+
this.condition = this.config.condition;
|
|
47
|
+
this.maxIterations = this.config.maxIterations;
|
|
48
|
+
this.testDataProfile = this.config.testDataProfile;
|
|
49
|
+
this.startStep = this.config.startStep;
|
|
50
|
+
this.endStep = this.config.endStep;
|
|
51
|
+
this.nestedSteps = this.config.nestedSteps || [];
|
|
52
|
+
this.expanded = this.config.expanded !== undefined ? this.config.expanded : true;
|
|
53
|
+
}
|
|
54
|
+
this.buildEditForm();
|
|
55
|
+
this.buildSelectConfigs();
|
|
56
|
+
}
|
|
57
|
+
buildEditForm() {
|
|
58
|
+
this.editForm = this.fb.group({
|
|
59
|
+
forOptionType: [null],
|
|
60
|
+
testDataProfile: [this.testDataProfile ?? ''],
|
|
61
|
+
startStep: [this.startStep ?? null],
|
|
62
|
+
endStep: [this.endStep ?? null],
|
|
63
|
+
condition: [this.condition ?? ''],
|
|
64
|
+
maxIterations: [this.maxIterations ?? 5],
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
/** Build select configs once (same option shape as default stories: id + name) */
|
|
68
|
+
buildSelectConfigs() {
|
|
69
|
+
const stepOptions = this.loopStepOptions.map((n) => ({
|
|
70
|
+
id: n,
|
|
71
|
+
value: n,
|
|
72
|
+
name: String(n),
|
|
73
|
+
label: String(n),
|
|
74
|
+
}));
|
|
75
|
+
this.forOptionTypeSelectConfig = {
|
|
76
|
+
key: 'forOptionType',
|
|
77
|
+
placeholder: 'Select option',
|
|
78
|
+
searchable: false,
|
|
79
|
+
options: [
|
|
80
|
+
{ id: 'test-data', value: 'test-data', name: 'Test Data Profile', label: 'Test Data Profile' },
|
|
81
|
+
{ id: 'fixed', value: 'fixed', name: 'Fixed count', label: 'Fixed count' },
|
|
82
|
+
],
|
|
83
|
+
};
|
|
84
|
+
this.startStepSelectConfig = {
|
|
85
|
+
key: 'startStep',
|
|
86
|
+
placeholder: 'Loop Start',
|
|
87
|
+
searchable: false,
|
|
88
|
+
options: stepOptions.map((o) => ({ ...o })),
|
|
89
|
+
};
|
|
90
|
+
this.endStepSelectConfig = {
|
|
91
|
+
key: 'endStep',
|
|
92
|
+
placeholder: 'Loop End',
|
|
93
|
+
searchable: false,
|
|
94
|
+
options: stepOptions.map((o) => ({ ...o })),
|
|
95
|
+
};
|
|
96
|
+
}
|
|
97
|
+
onEditFormFieldChange(controlName, value) {
|
|
98
|
+
this.editForm.get(controlName)?.setValue(value);
|
|
99
|
+
}
|
|
100
|
+
getLoopTypeLabel() {
|
|
101
|
+
return this.loopType === 'for' ? 'FOR LOOP' : 'WHILE LOOP';
|
|
102
|
+
}
|
|
103
|
+
getEndLabel() {
|
|
104
|
+
return this.loopType === 'for' ? 'END FOR' : 'END WHILE';
|
|
105
|
+
}
|
|
106
|
+
getStepsSummary() {
|
|
107
|
+
if (this.nestedSteps.length === 0) {
|
|
108
|
+
return '0 steps';
|
|
109
|
+
}
|
|
110
|
+
const start = this.startStep || this.nestedSteps[0]?.stepNumber || 1;
|
|
111
|
+
const end = this.endStep || this.nestedSteps[this.nestedSteps.length - 1]?.stepNumber || this.nestedSteps.length;
|
|
112
|
+
const count = this.nestedSteps.length;
|
|
113
|
+
return `Steps ${start}-${end} (${count} step${count !== 1 ? 's' : ''})`;
|
|
114
|
+
}
|
|
115
|
+
onToggleExpanded() {
|
|
116
|
+
this.expanded = !this.expanded;
|
|
117
|
+
this.toggleExpanded.emit({ config: this.config, expanded: this.expanded });
|
|
118
|
+
}
|
|
119
|
+
onTestDataProfileChange(value) {
|
|
120
|
+
this.testDataProfile = value;
|
|
121
|
+
this.testDataProfileChange.emit(value);
|
|
122
|
+
}
|
|
123
|
+
onStartStepChange(value) {
|
|
124
|
+
this.startStep = value;
|
|
125
|
+
this.startStepChange.emit(value);
|
|
126
|
+
}
|
|
127
|
+
onEndStepChange(value) {
|
|
128
|
+
this.endStep = value;
|
|
129
|
+
this.endStepChange.emit(value);
|
|
130
|
+
}
|
|
131
|
+
onConditionChange(value) {
|
|
132
|
+
this.condition = value;
|
|
133
|
+
this.conditionChange.emit(value);
|
|
134
|
+
}
|
|
135
|
+
onMaxIterationsChange(value) {
|
|
136
|
+
this.maxIterations = value;
|
|
137
|
+
this.maxIterationsChange.emit(value);
|
|
138
|
+
}
|
|
139
|
+
onNestedStepChange(step, index) {
|
|
140
|
+
this.nestedStepChange.emit({ step, index });
|
|
141
|
+
}
|
|
142
|
+
onNestedToggleExpanded(event, step, index) {
|
|
143
|
+
// Handle both boolean and object formats
|
|
144
|
+
if (typeof event === 'object' && event !== null && 'expanded' in event) {
|
|
145
|
+
// Object format from step-group: {config, expanded}
|
|
146
|
+
step.expanded = event.expanded;
|
|
147
|
+
// If it's a step-group, we need to emit the event up the chain
|
|
148
|
+
if (this.isStepGroup(step)) {
|
|
149
|
+
// Re-emit the toggleExpanded event with the object format
|
|
150
|
+
this.toggleExpanded.emit(event);
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
else {
|
|
154
|
+
// Boolean format from other step types
|
|
155
|
+
step.expanded = event;
|
|
156
|
+
}
|
|
157
|
+
this.onNestedStepChange(step, index);
|
|
158
|
+
}
|
|
159
|
+
onAddStep() {
|
|
160
|
+
this.addStep.emit();
|
|
161
|
+
}
|
|
162
|
+
onDeleteStep(index) {
|
|
163
|
+
this.deleteStep.emit(index);
|
|
164
|
+
}
|
|
165
|
+
isNormalStep(step) {
|
|
166
|
+
return 'eventType' in step;
|
|
167
|
+
}
|
|
168
|
+
isLoopStep(step) {
|
|
169
|
+
return 'loopType' in step;
|
|
170
|
+
}
|
|
171
|
+
isConditionStep(step) {
|
|
172
|
+
return 'condition' in step && 'branches' in step;
|
|
173
|
+
}
|
|
174
|
+
isStepGroup(step) {
|
|
175
|
+
return 'groupName' in step;
|
|
176
|
+
}
|
|
177
|
+
onAddBranch() {
|
|
178
|
+
// Not used in loop step, but needed for recursive condition steps
|
|
179
|
+
}
|
|
180
|
+
onDeleteBranch(branch) {
|
|
181
|
+
// Not used in loop step, but needed for recursive condition steps
|
|
182
|
+
}
|
|
183
|
+
onNestedConditionAddBranch(nestedStep, index) {
|
|
184
|
+
// Add a new branch to the nested condition step
|
|
185
|
+
if (!nestedStep.branches) {
|
|
186
|
+
nestedStep.branches = [];
|
|
187
|
+
}
|
|
188
|
+
const newBranch = {
|
|
189
|
+
type: nestedStep.branches.length === 0 ? 'if' : 'else',
|
|
190
|
+
label: nestedStep.branches.length === 0 ? 'IF TRUE' : 'ELSE',
|
|
191
|
+
stepType: StepTypes.CONDITION_IF,
|
|
192
|
+
nestedSteps: []
|
|
193
|
+
};
|
|
194
|
+
nestedStep.branches.push(newBranch);
|
|
195
|
+
this.onNestedStepChange(nestedStep, index);
|
|
196
|
+
}
|
|
197
|
+
onNestedConditionDeleteBranch(nestedStep, branch, index) {
|
|
198
|
+
// Delete a branch from the nested condition step
|
|
199
|
+
if (nestedStep.branches) {
|
|
200
|
+
const branchIndex = nestedStep.branches.indexOf(branch);
|
|
201
|
+
if (branchIndex > -1) {
|
|
202
|
+
nestedStep.branches.splice(branchIndex, 1);
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
this.onNestedStepChange(nestedStep, index);
|
|
206
|
+
}
|
|
207
|
+
onOpenExternal() {
|
|
208
|
+
// Not used in loop step, but needed for recursive step groups
|
|
209
|
+
}
|
|
210
|
+
onEdit() {
|
|
211
|
+
this.editSnapshot = {
|
|
212
|
+
testDataProfile: this.testDataProfile,
|
|
213
|
+
startStep: this.startStep,
|
|
214
|
+
endStep: this.endStep,
|
|
215
|
+
condition: this.condition,
|
|
216
|
+
maxIterations: this.maxIterations,
|
|
217
|
+
};
|
|
218
|
+
this.editForm.patchValue({
|
|
219
|
+
forOptionType: null,
|
|
220
|
+
testDataProfile: this.testDataProfile ?? '',
|
|
221
|
+
startStep: this.startStep ?? null,
|
|
222
|
+
endStep: this.endStep ?? null,
|
|
223
|
+
condition: this.condition ?? '',
|
|
224
|
+
maxIterations: this.maxIterations ?? 5,
|
|
225
|
+
});
|
|
226
|
+
this.isEditing = true;
|
|
227
|
+
this.edit.emit();
|
|
228
|
+
}
|
|
229
|
+
onEditApply() {
|
|
230
|
+
const v = this.editForm.value;
|
|
231
|
+
this.testDataProfile = v.testDataProfile ?? '';
|
|
232
|
+
this.startStep = v.startStep ?? undefined;
|
|
233
|
+
this.endStep = v.endStep ?? undefined;
|
|
234
|
+
this.condition = v.condition ?? '';
|
|
235
|
+
const maxVal = v.maxIterations;
|
|
236
|
+
this.maxIterations = typeof maxVal === 'number' ? maxVal : (maxVal != null && maxVal !== '' ? Number(maxVal) : undefined);
|
|
237
|
+
this.testDataProfileChange.emit(this.testDataProfile);
|
|
238
|
+
this.startStepChange.emit(this.startStep ?? 1);
|
|
239
|
+
this.endStepChange.emit(this.endStep ?? 1);
|
|
240
|
+
if (this.loopType === 'while') {
|
|
241
|
+
this.conditionChange.emit(this.condition);
|
|
242
|
+
this.maxIterationsChange.emit(this.maxIterations ?? 5);
|
|
243
|
+
}
|
|
244
|
+
this.isEditing = false;
|
|
245
|
+
}
|
|
246
|
+
onEditCancel() {
|
|
247
|
+
this.testDataProfile = this.editSnapshot.testDataProfile;
|
|
248
|
+
this.startStep = this.editSnapshot.startStep;
|
|
249
|
+
this.endStep = this.editSnapshot.endStep;
|
|
250
|
+
this.condition = this.editSnapshot.condition;
|
|
251
|
+
this.maxIterations = this.editSnapshot.maxIterations;
|
|
252
|
+
this.isEditing = false;
|
|
253
|
+
}
|
|
254
|
+
onEditInDepth() {
|
|
255
|
+
this.moreOptions.emit();
|
|
256
|
+
}
|
|
257
|
+
onLink() {
|
|
258
|
+
// Not used in loop step, but needed for recursive step groups
|
|
259
|
+
}
|
|
260
|
+
onDuplicate() {
|
|
261
|
+
this.duplicate.emit();
|
|
262
|
+
}
|
|
263
|
+
onDelete() {
|
|
264
|
+
this.delete.emit();
|
|
265
|
+
}
|
|
266
|
+
onMoreOptions() {
|
|
267
|
+
this.moreOptions.emit();
|
|
268
|
+
}
|
|
269
|
+
}
|
|
270
|
+
TestCaseLoopStepComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TestCaseLoopStepComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component });
|
|
271
|
+
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" }, 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", edit: "edit" }, host: { classAttribute: "cqa-ui-root" }, 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-custom-input [value]=\"editForm.get('testDataProfile')?.value ?? ''\"\n (valueChange)=\"onEditFormFieldChange('testDataProfile', $event)\" placeholder=\"Testdataprofile\"\n class=\"cqa-w-full cqa-max-w-[216px]\" size=\"sm\">\n </cqa-custom-input>\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=\"loopType === 'while'\">\n <cqa-custom-input [value]=\"editForm.get('condition')?.value ?? ''\" class=\"cqa-w-full cqa-max-w-[216px]\"\n (valueChange)=\"onEditFormFieldChange('condition', $event)\" placeholder='element \".selector\" exists' size=\"sm\">\n </cqa-custom-input>\n <span class=\"cqa-text-[#6B7280] cqa-text-sm\">(max</span>\n <cqa-custom-input type=\"number\" class=\"cqa-w-full cqa-max-w-[216px]\"\n [value]=\"editForm.get('maxIterations')?.value != null ? stringFn(editForm.get('maxIterations')?.value) : ''\"\n (valueChange)=\"onEditFormFieldChange('maxIterations', ($event !== '' && $event != null) ? numberFn($event) : 5)\"\n placeholder=\"5\" size=\"sm\">\n </cqa-custom-input>\n <span class=\"cqa-text-[#6B7280] cqa-text-sm\">iterations)</span>\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 <!-- 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 Test Data profile\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 Start\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 End\n </button>\n </div>\n\n <!-- While Loop Condition -->\n <div *ngIf=\"loopType === 'while'\" 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\">{{ condition || 'element \".notification-badge\" exists' }}</span>\n <span class=\"cqa-text-gray-700 cqa-text-sm\">(max {{ maxIterations ?? 5 }} iterations)</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 <!-- 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 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 [isNested]=\"true\" [isInsideLoop]=\"true\" (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 (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 </cqa-test-case-details-renderer>\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"] }, { type: i3.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: i4.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }, { type: i5.TestCaseDetailsRendererComponent, selector: "cqa-test-case-details-renderer", inputs: ["step", "index", "isNested", "isInsideLoop", "branch"], outputs: ["nestedStepChange", "addStep", "deleteStep", "toggleExpanded", "groupNameChange", "descriptionChange", "reusableChange", "openExternal", "edit", "link", "duplicate", "delete", "viewDetails", "selectionChange", "conditionChange", "branchStepChange", "addStepForBranch", "deleteStepWithBranch", "addBranch", "deleteBranch", "testDataProfileChange", "startStepChange", "endStepChange", "maxIterationsChange", "eventTypeChange", "parameterChange"] }], directives: [{ type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
|
|
272
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TestCaseLoopStepComponent, decorators: [{
|
|
273
|
+
type: Component,
|
|
274
|
+
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-custom-input [value]=\"editForm.get('testDataProfile')?.value ?? ''\"\n (valueChange)=\"onEditFormFieldChange('testDataProfile', $event)\" placeholder=\"Testdataprofile\"\n class=\"cqa-w-full cqa-max-w-[216px]\" size=\"sm\">\n </cqa-custom-input>\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=\"loopType === 'while'\">\n <cqa-custom-input [value]=\"editForm.get('condition')?.value ?? ''\" class=\"cqa-w-full cqa-max-w-[216px]\"\n (valueChange)=\"onEditFormFieldChange('condition', $event)\" placeholder='element \".selector\" exists' size=\"sm\">\n </cqa-custom-input>\n <span class=\"cqa-text-[#6B7280] cqa-text-sm\">(max</span>\n <cqa-custom-input type=\"number\" class=\"cqa-w-full cqa-max-w-[216px]\"\n [value]=\"editForm.get('maxIterations')?.value != null ? stringFn(editForm.get('maxIterations')?.value) : ''\"\n (valueChange)=\"onEditFormFieldChange('maxIterations', ($event !== '' && $event != null) ? numberFn($event) : 5)\"\n placeholder=\"5\" size=\"sm\">\n </cqa-custom-input>\n <span class=\"cqa-text-[#6B7280] cqa-text-sm\">iterations)</span>\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 <!-- 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 Test Data profile\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 Start\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 End\n </button>\n </div>\n\n <!-- While Loop Condition -->\n <div *ngIf=\"loopType === 'while'\" 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\">{{ condition || 'element \".notification-badge\" exists' }}</span>\n <span class=\"cqa-text-gray-700 cqa-text-sm\">(max {{ maxIterations ?? 5 }} iterations)</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 <!-- 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 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 [isNested]=\"true\" [isInsideLoop]=\"true\" (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 (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 </cqa-test-case-details-renderer>\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>" }]
|
|
275
|
+
}], ctorParameters: function () { return [{ type: i1.FormBuilder }]; }, propDecorators: { config: [{
|
|
276
|
+
type: Input
|
|
277
|
+
}], id: [{
|
|
278
|
+
type: Input
|
|
279
|
+
}], loopType: [{
|
|
280
|
+
type: Input
|
|
281
|
+
}], stepNumber: [{
|
|
282
|
+
type: Input
|
|
283
|
+
}], condition: [{
|
|
284
|
+
type: Input
|
|
285
|
+
}], maxIterations: [{
|
|
286
|
+
type: Input
|
|
287
|
+
}], testDataProfile: [{
|
|
288
|
+
type: Input
|
|
289
|
+
}], startStep: [{
|
|
290
|
+
type: Input
|
|
291
|
+
}], endStep: [{
|
|
292
|
+
type: Input
|
|
293
|
+
}], nestedSteps: [{
|
|
294
|
+
type: Input
|
|
295
|
+
}], expanded: [{
|
|
296
|
+
type: Input
|
|
297
|
+
}], isNested: [{
|
|
298
|
+
type: Input
|
|
299
|
+
}], isInsideLoop: [{
|
|
300
|
+
type: Input
|
|
301
|
+
}], toggleExpanded: [{
|
|
302
|
+
type: Output
|
|
303
|
+
}], testDataProfileChange: [{
|
|
304
|
+
type: Output
|
|
305
|
+
}], startStepChange: [{
|
|
306
|
+
type: Output
|
|
307
|
+
}], endStepChange: [{
|
|
308
|
+
type: Output
|
|
309
|
+
}], conditionChange: [{
|
|
310
|
+
type: Output
|
|
311
|
+
}], maxIterationsChange: [{
|
|
312
|
+
type: Output
|
|
313
|
+
}], nestedStepChange: [{
|
|
314
|
+
type: Output
|
|
315
|
+
}], addStep: [{
|
|
316
|
+
type: Output
|
|
317
|
+
}], deleteStep: [{
|
|
318
|
+
type: Output
|
|
319
|
+
}], duplicate: [{
|
|
320
|
+
type: Output
|
|
321
|
+
}], delete: [{
|
|
322
|
+
type: Output
|
|
323
|
+
}], moreOptions: [{
|
|
324
|
+
type: Output
|
|
325
|
+
}], edit: [{
|
|
326
|
+
type: Output
|
|
327
|
+
}] } });
|
|
328
|
+
//# 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 } from '@angular/core';\nimport { FormBuilder, FormGroup } from '@angular/forms';\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';\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 {\n  @Input() config!: LoopStepConfig;\n  @Input() id!: string;\n  @Input() loopType!: 'for' | 'while';\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\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() edit = new EventEmitter<void>();\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  startStepSelectConfig!: DynamicSelectFieldConfig;\n  endStepSelectConfig!: 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  /** 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(private fb: FormBuilder) {}\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    }\n    this.buildEditForm();\n    this.buildSelectConfigs();\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.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  }\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  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  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      // If it's a step-group, we need to emit the event up the chain\n      if (this.isStepGroup(step)) {\n        // Re-emit the toggleExpanded event with the object format\n        this.toggleExpanded.emit(event as {config: LoopStepConfig, expanded: boolean});\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    this.testDataProfile = 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    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    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  }\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  onMoreOptions(): void {\n    this.moreOptions.emit();\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-custom-input [value]=\"editForm.get('testDataProfile')?.value ?? ''\"\n          (valueChange)=\"onEditFormFieldChange('testDataProfile', $event)\" placeholder=\"Testdataprofile\"\n          class=\"cqa-w-full cqa-max-w-[216px]\" size=\"sm\">\n        </cqa-custom-input>\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=\"loopType === 'while'\">\n        <cqa-custom-input [value]=\"editForm.get('condition')?.value ?? ''\" class=\"cqa-w-full cqa-max-w-[216px]\"\n          (valueChange)=\"onEditFormFieldChange('condition', $event)\" placeholder='element \".selector\" exists' size=\"sm\">\n        </cqa-custom-input>\n        <span class=\"cqa-text-[#6B7280] cqa-text-sm\">(max</span>\n        <cqa-custom-input type=\"number\" class=\"cqa-w-full cqa-max-w-[216px]\"\n          [value]=\"editForm.get('maxIterations')?.value != null ? stringFn(editForm.get('maxIterations')?.value) : ''\"\n          (valueChange)=\"onEditFormFieldChange('maxIterations', ($event !== '' && $event != null) ? numberFn($event) : 5)\"\n          placeholder=\"5\" size=\"sm\">\n        </cqa-custom-input>\n        <span class=\"cqa-text-[#6B7280] cqa-text-sm\">iterations)</span>\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    <!-- 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        Test Data profile\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        Start\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        End\n      </button>\n    </div>\n\n    <!-- While Loop Condition -->\n    <div *ngIf=\"loopType === 'while'\" 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\">{{ condition || 'element \".notification-badge\" exists' }}</span>\n      <span class=\"cqa-text-gray-700 cqa-text-sm\">(max {{ maxIterations ?? 5 }} iterations)</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    <!-- 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 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        [isNested]=\"true\" [isInsideLoop]=\"true\" (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        (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      </cqa-test-case-details-renderer>\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>"]}
|