@cqa-lib/cqa-ui 1.1.201 → 1.1.202
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 +75 -13
- package/esm2020/lib/step-builder/step-builder-ai-agent/step-builder-ai-agent.component.mjs +3 -3
- package/esm2020/lib/step-builder/step-builder-document/step-builder-document.component.mjs +5 -6
- package/esm2020/lib/step-builder/step-builder-group/step-builder-group.component.mjs +148 -0
- package/esm2020/lib/test-case-details/condition-step/condition-step.component.mjs +7 -5
- package/esm2020/lib/test-case-details/loop-step/loop-step.component.mjs +7 -5
- package/esm2020/lib/test-case-details/step-group/step-group.component.mjs +6 -5
- package/esm2020/lib/ui-kit.module.mjs +6 -6
- package/esm2020/public-api.mjs +2 -2
- package/fesm2015/cqa-lib-cqa-ui.mjs +192 -617
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +195 -612
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/step-builder/step-builder-action/step-builder-action.component.d.ts +10 -2
- package/lib/step-builder/step-builder-group/step-builder-group.component.d.ts +48 -0
- package/lib/ui-kit.module.d.ts +2 -2
- package/package.json +1 -1
- package/public-api.d.ts +1 -1
- package/styles.css +1 -1
- package/esm2020/lib/step-builder/step-builder-api/step-builder-api.component.mjs +0 -636
- package/lib/step-builder/step-builder-api/step-builder-api.component.d.ts +0 -121
|
@@ -1,636 +0,0 @@
|
|
|
1
|
-
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
2
|
-
import { FormArray, FormGroup, Validators } from '@angular/forms';
|
|
3
|
-
import * as i0 from "@angular/core";
|
|
4
|
-
import * as i1 from "@angular/forms";
|
|
5
|
-
import * as i2 from "../../dynamic-select/dynamic-select-field.component";
|
|
6
|
-
import * as i3 from "../../custom-input/custom-input.component";
|
|
7
|
-
import * as i4 from "../../button/button.component";
|
|
8
|
-
import * as i5 from "@angular/material/icon";
|
|
9
|
-
import * as i6 from "../../custom-textarea/custom-textarea.component";
|
|
10
|
-
import * as i7 from "@angular/material/checkbox";
|
|
11
|
-
import * as i8 from "@angular/common";
|
|
12
|
-
export class StepBuilderApiComponent {
|
|
13
|
-
constructor(fb) {
|
|
14
|
-
this.fb = fb;
|
|
15
|
-
/** Options for HTTP method dropdown */
|
|
16
|
-
this.httpMethodOptions = [
|
|
17
|
-
{ id: 'GET', value: 'GET', name: 'GET', label: 'GET' },
|
|
18
|
-
{ id: 'POST', value: 'POST', name: 'POST', label: 'POST' },
|
|
19
|
-
{ id: 'PUT', value: 'PUT', name: 'PUT', label: 'PUT' },
|
|
20
|
-
{ id: 'PATCH', value: 'PATCH', name: 'PATCH', label: 'PATCH' },
|
|
21
|
-
{ id: 'DELETE', value: 'DELETE', name: 'DELETE', label: 'DELETE' },
|
|
22
|
-
{ id: 'HEAD', value: 'HEAD', name: 'HEAD', label: 'HEAD' },
|
|
23
|
-
{ id: 'OPTIONS', value: 'OPTIONS', name: 'OPTIONS', label: 'OPTIONS' }
|
|
24
|
-
];
|
|
25
|
-
/** Options for header name dropdown */
|
|
26
|
-
this.headerNameOptions = [];
|
|
27
|
-
/** Current progress step */
|
|
28
|
-
this.currentStep = 'request-details';
|
|
29
|
-
/** Response preview data */
|
|
30
|
-
this.responsePreview = null;
|
|
31
|
-
/** Loading state */
|
|
32
|
-
this.isLoading = false;
|
|
33
|
-
/** Emit when step is created */
|
|
34
|
-
this.createStep = new EventEmitter();
|
|
35
|
-
/** Emit when cancelled */
|
|
36
|
-
this.cancelled = new EventEmitter();
|
|
37
|
-
/** Emit when request is sent */
|
|
38
|
-
this.sendRequest = new EventEmitter();
|
|
39
|
-
/** Emit when cURL is imported */
|
|
40
|
-
this.importCurl = new EventEmitter();
|
|
41
|
-
this.selectedTab = 'headers';
|
|
42
|
-
this.selectedProgressStep = 'request-details';
|
|
43
|
-
this.formSubmitted = false;
|
|
44
|
-
// progressSteps: ProgressStep[] = ['request-details', 'store-response', 'validation'];
|
|
45
|
-
// Cache config objects to prevent infinite change detection loops
|
|
46
|
-
this.verificationTypeConfigCache = null;
|
|
47
|
-
this.expectedTypeConfigCache = null;
|
|
48
|
-
this.httpMethodConfigCache = null;
|
|
49
|
-
this.headerNameConfigCache = null;
|
|
50
|
-
this.paramNameConfigCache = null;
|
|
51
|
-
this.hasLoadedInitialData = false;
|
|
52
|
-
// Get first HTTP method option (default to 'GET' if options not available yet)
|
|
53
|
-
const firstMethod = this.httpMethodOptions.length > 0
|
|
54
|
-
? this.httpMethodOptions[0].value
|
|
55
|
-
: 'GET';
|
|
56
|
-
this.apiForm = this.fb.group({
|
|
57
|
-
method: [firstMethod, Validators.required],
|
|
58
|
-
url: ['', [Validators.required, this.urlValidator()]],
|
|
59
|
-
headers: this.fb.array([]),
|
|
60
|
-
body: [''],
|
|
61
|
-
params: this.fb.array([]),
|
|
62
|
-
scripts: [''],
|
|
63
|
-
variableName: [''],
|
|
64
|
-
validation: this.fb.array([])
|
|
65
|
-
});
|
|
66
|
-
}
|
|
67
|
-
urlValidator() {
|
|
68
|
-
return (control) => {
|
|
69
|
-
const url = control.value;
|
|
70
|
-
if (!url) {
|
|
71
|
-
return null; // Let required validator handle empty case
|
|
72
|
-
}
|
|
73
|
-
// Pattern for plain URLs only (no parameters)
|
|
74
|
-
const urlPattern = /^(https?:\/\/)?([\w-]+(\.[\w-]+)+)([\w.,@?^=%&:/~+#-]*[\w@?^=%&/~+#-])?$/;
|
|
75
|
-
// Pattern to detect any parameter patterns: *|param|, $|param|, @|param|, or ${param}
|
|
76
|
-
const parameterPattern = /(\*\|[\w-]+\|(\*)?|\$\|[\w-]+\|(\*)?|@\|[\w-]+\|(\*)?|\$\{[^}]+\})/;
|
|
77
|
-
// Pattern for parameter-only values: *|param|, $|param|, @|param|, or ${param}
|
|
78
|
-
const envPattern = /^\*\|[\w-]+\|(\*)?$|^\$\|[\w-]+\|(\*)?$|^@\|[\w-]+\|(\*)?$|^\$\{[^}]+\}$/;
|
|
79
|
-
// Pattern to allow URLs with embedded parameters: *|param|, $|param|, @|param|, or ${param}
|
|
80
|
-
const flexibleUrlPattern = /^[@]?(https?:\/\/)?([\w-]+(\.[\w-]+)+)([\w.,@?^=%&:/~+#-]*(\*\|[\w-]+\|(\*)?|\$\|[\w-]+\|(\*)?|@\|[\w-]+\|(\*)?|\$\{[^}]+\})[\w.,@?^=%&:/~+#-]*)*$/;
|
|
81
|
-
// Pattern that allows parameters anywhere after http:// or https://
|
|
82
|
-
const protocolWithParamsPattern = /^https?:\/\/(\*\|[\w-]+\|(\*)?|\$\|[\w-]+\|(\*)?|@\|[\w-]+\|(\*)?|\$\{[^}]+\})[^\s]*$|^https?:\/\/[^\s]*(\*\|[\w-]+\|(\*)?|\$\|[\w-]+\|(\*)?|@\|[\w-]+\|(\*)?|\$\{[^}]+\})[^\s]*$/;
|
|
83
|
-
// Check if URL contains parameters
|
|
84
|
-
if (parameterPattern.test(url)) {
|
|
85
|
-
// If it contains parameters, check if it matches allowed patterns
|
|
86
|
-
const envCombined = new RegExp(`${urlPattern.source}|${flexibleUrlPattern.source}|${envPattern.source}|${protocolWithParamsPattern.source}`);
|
|
87
|
-
if (!envCombined.test(url)) {
|
|
88
|
-
return { hasParameter: true };
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
else {
|
|
92
|
-
// If no parameters, check if it's a valid plain URL
|
|
93
|
-
if (!urlPattern.test(url)) {
|
|
94
|
-
return { pattern: true };
|
|
95
|
-
}
|
|
96
|
-
}
|
|
97
|
-
return null;
|
|
98
|
-
};
|
|
99
|
-
}
|
|
100
|
-
ngOnInit() {
|
|
101
|
-
// Sync currentStep input with selectedProgressStep
|
|
102
|
-
if (this.currentStep) {
|
|
103
|
-
this.selectedProgressStep = this.currentStep;
|
|
104
|
-
}
|
|
105
|
-
// Set method to first option if not already set
|
|
106
|
-
if (!this.apiForm.get('method')?.value && this.httpMethodOptions.length > 0) {
|
|
107
|
-
this.apiForm.get('method')?.setValue(this.httpMethodOptions[0].value);
|
|
108
|
-
}
|
|
109
|
-
// Load initial data if provided (edit mode)
|
|
110
|
-
if (this.initialData) {
|
|
111
|
-
this.loadInitialData(this.initialData);
|
|
112
|
-
this.hasLoadedInitialData = true;
|
|
113
|
-
}
|
|
114
|
-
else {
|
|
115
|
-
// Add initial header row for new step
|
|
116
|
-
this.addHeader();
|
|
117
|
-
}
|
|
118
|
-
// Initialize validation form array if starting on validation step
|
|
119
|
-
if (this.selectedProgressStep === 'validation' && this.validationFormArray.length === 0) {
|
|
120
|
-
this.addValidationRule();
|
|
121
|
-
}
|
|
122
|
-
}
|
|
123
|
-
loadInitialData(data) {
|
|
124
|
-
console.log('loadInitialData: Loading data', data);
|
|
125
|
-
console.log('loadInitialData: Headers count', data.headers?.length || 0);
|
|
126
|
-
console.log('loadInitialData: Headers data', data.headers);
|
|
127
|
-
// Set basic fields
|
|
128
|
-
this.apiForm.patchValue({
|
|
129
|
-
method: data.method || 'GET',
|
|
130
|
-
url: data.url || '',
|
|
131
|
-
body: data.body || '',
|
|
132
|
-
scripts: data.scripts || '',
|
|
133
|
-
variableName: data.variableName || ''
|
|
134
|
-
});
|
|
135
|
-
// Load headers
|
|
136
|
-
// Clear existing headers first
|
|
137
|
-
while (this.headersFormArray.length !== 0) {
|
|
138
|
-
this.headersFormArray.removeAt(0);
|
|
139
|
-
}
|
|
140
|
-
if (data.headers && data.headers.length > 0) {
|
|
141
|
-
// Add headers from initial data (include all headers, even if empty)
|
|
142
|
-
data.headers.forEach(header => {
|
|
143
|
-
const headerGroup = this.fb.group({
|
|
144
|
-
name: [header.name || '', Validators.required],
|
|
145
|
-
value: [header.value || '', Validators.required]
|
|
146
|
-
});
|
|
147
|
-
this.headersFormArray.push(headerGroup);
|
|
148
|
-
});
|
|
149
|
-
console.log('loadInitialData: Loaded headers into form array, count:', this.headersFormArray.length);
|
|
150
|
-
}
|
|
151
|
-
// Ensure at least one header row exists
|
|
152
|
-
if (this.headersFormArray.length === 0) {
|
|
153
|
-
console.log('loadInitialData: No headers found, adding empty header row');
|
|
154
|
-
this.addHeader();
|
|
155
|
-
}
|
|
156
|
-
// Load params
|
|
157
|
-
// Clear existing params first
|
|
158
|
-
while (this.paramsFormArray.length !== 0) {
|
|
159
|
-
this.paramsFormArray.removeAt(0);
|
|
160
|
-
}
|
|
161
|
-
if (data.params && data.params.length > 0) {
|
|
162
|
-
// Add params from initial data (include all params, even if empty)
|
|
163
|
-
data.params.forEach(param => {
|
|
164
|
-
const paramGroup = this.fb.group({
|
|
165
|
-
name: [param.name || '', Validators.required],
|
|
166
|
-
value: [param.value || '', Validators.required]
|
|
167
|
-
});
|
|
168
|
-
this.paramsFormArray.push(paramGroup);
|
|
169
|
-
});
|
|
170
|
-
}
|
|
171
|
-
// Load validation rules
|
|
172
|
-
if (data.validation && data.validation.length > 0) {
|
|
173
|
-
// Clear existing validation rules
|
|
174
|
-
while (this.validationFormArray.length !== 0) {
|
|
175
|
-
this.validationFormArray.removeAt(0);
|
|
176
|
-
}
|
|
177
|
-
// Add validation rules from initial data
|
|
178
|
-
data.validation.forEach(rule => {
|
|
179
|
-
const validationGroup = this.fb.group({
|
|
180
|
-
jsonPath: [rule.jsonPath || '', Validators.required],
|
|
181
|
-
verificationType: [rule.verificationType || 'equals', Validators.required],
|
|
182
|
-
expectedType: [rule.expectedType || 'string', Validators.required],
|
|
183
|
-
expectedValue: [rule.expectedValue !== undefined ? rule.expectedValue : '', Validators.required],
|
|
184
|
-
result: [rule.result || 'Not run'],
|
|
185
|
-
checked: [rule.checked || false]
|
|
186
|
-
});
|
|
187
|
-
// Ensure all controls are enabled
|
|
188
|
-
validationGroup.enable({ emitEvent: false });
|
|
189
|
-
this.validationFormArray.push(validationGroup);
|
|
190
|
-
});
|
|
191
|
-
}
|
|
192
|
-
}
|
|
193
|
-
get headersFormArray() {
|
|
194
|
-
return this.apiForm.get('headers');
|
|
195
|
-
}
|
|
196
|
-
get paramsFormArray() {
|
|
197
|
-
return this.apiForm.get('params');
|
|
198
|
-
}
|
|
199
|
-
get validationFormArray() {
|
|
200
|
-
return this.apiForm.get('validation');
|
|
201
|
-
}
|
|
202
|
-
addHeader() {
|
|
203
|
-
// Get first header name option
|
|
204
|
-
const headerOptions = this.headerNameOptions.length > 0
|
|
205
|
-
? this.headerNameOptions
|
|
206
|
-
: [
|
|
207
|
-
{ id: 'string', value: 'string', name: 'string', label: 'string' },
|
|
208
|
-
{ id: 'Content-Type', value: 'Content-Type', name: 'Content-Type', label: 'Content-Type' },
|
|
209
|
-
{ id: 'Authorization', value: 'Authorization', name: 'Authorization', label: 'Authorization' },
|
|
210
|
-
{ id: 'Accept', value: 'Accept', name: 'Accept', label: 'Accept' }
|
|
211
|
-
];
|
|
212
|
-
const firstHeaderName = headerOptions.length > 0 ? headerOptions[0].value : '';
|
|
213
|
-
const headerGroup = this.fb.group({
|
|
214
|
-
name: [firstHeaderName, Validators.required],
|
|
215
|
-
value: ['', Validators.required]
|
|
216
|
-
});
|
|
217
|
-
this.headersFormArray.push(headerGroup);
|
|
218
|
-
}
|
|
219
|
-
removeHeader(index) {
|
|
220
|
-
this.headersFormArray.removeAt(index);
|
|
221
|
-
}
|
|
222
|
-
addParam() {
|
|
223
|
-
const paramGroup = this.fb.group({
|
|
224
|
-
name: ['', Validators.required],
|
|
225
|
-
value: ['', Validators.required]
|
|
226
|
-
});
|
|
227
|
-
this.paramsFormArray.push(paramGroup);
|
|
228
|
-
}
|
|
229
|
-
removeParam(index) {
|
|
230
|
-
this.paramsFormArray.removeAt(index);
|
|
231
|
-
}
|
|
232
|
-
ngOnChanges(changes) {
|
|
233
|
-
// Reset config caches when inputs change
|
|
234
|
-
if (changes['httpMethodOptions']) {
|
|
235
|
-
this.httpMethodConfigCache = null;
|
|
236
|
-
// Set method to first option if not already set or if we're not in edit mode
|
|
237
|
-
if (!this.hasLoadedInitialData && this.httpMethodOptions.length > 0) {
|
|
238
|
-
const methodControl = this.apiForm.get('method');
|
|
239
|
-
if (methodControl && !methodControl.value) {
|
|
240
|
-
methodControl.setValue(this.httpMethodOptions[0].value);
|
|
241
|
-
}
|
|
242
|
-
}
|
|
243
|
-
}
|
|
244
|
-
if (changes['headerNameOptions']) {
|
|
245
|
-
this.headerNameConfigCache = null;
|
|
246
|
-
// Update header names to first option for empty headers (only if not in edit mode)
|
|
247
|
-
if (!this.hasLoadedInitialData && this.headerNameOptions.length > 0) {
|
|
248
|
-
this.headersFormArray.controls.forEach((headerGroup) => {
|
|
249
|
-
const nameControl = headerGroup.get('name');
|
|
250
|
-
if (nameControl && !nameControl.value) {
|
|
251
|
-
nameControl.setValue(this.headerNameOptions[0].value);
|
|
252
|
-
}
|
|
253
|
-
});
|
|
254
|
-
}
|
|
255
|
-
}
|
|
256
|
-
// Load initial data if it's set after component initialization
|
|
257
|
-
if (changes['initialData'] && !changes['initialData'].firstChange && this.initialData && !this.hasLoadedInitialData) {
|
|
258
|
-
this.loadInitialData(this.initialData);
|
|
259
|
-
this.hasLoadedInitialData = true;
|
|
260
|
-
}
|
|
261
|
-
}
|
|
262
|
-
getHttpMethodConfig() {
|
|
263
|
-
if (!this.httpMethodConfigCache) {
|
|
264
|
-
this.httpMethodConfigCache = {
|
|
265
|
-
key: 'method',
|
|
266
|
-
placeholder: 'Select method',
|
|
267
|
-
multiple: false,
|
|
268
|
-
searchable: false,
|
|
269
|
-
options: this.httpMethodOptions
|
|
270
|
-
};
|
|
271
|
-
}
|
|
272
|
-
return this.httpMethodConfigCache;
|
|
273
|
-
}
|
|
274
|
-
getHeaderNameConfig(index) {
|
|
275
|
-
if (!this.headerNameConfigCache) {
|
|
276
|
-
const defaultOptions = [
|
|
277
|
-
{ id: 'string', value: 'string', name: 'string', label: 'string' },
|
|
278
|
-
{ id: 'Content-Type', value: 'Content-Type', name: 'Content-Type', label: 'Content-Type' },
|
|
279
|
-
{ id: 'Authorization', value: 'Authorization', name: 'Authorization', label: 'Authorization' },
|
|
280
|
-
{ id: 'Accept', value: 'Accept', name: 'Accept', label: 'Accept' }
|
|
281
|
-
];
|
|
282
|
-
this.headerNameConfigCache = {
|
|
283
|
-
key: 'name',
|
|
284
|
-
placeholder: 'Select header',
|
|
285
|
-
multiple: false,
|
|
286
|
-
searchable: true,
|
|
287
|
-
options: this.headerNameOptions.length > 0 ? this.headerNameOptions : defaultOptions
|
|
288
|
-
};
|
|
289
|
-
}
|
|
290
|
-
return this.headerNameConfigCache;
|
|
291
|
-
}
|
|
292
|
-
getParamNameConfig(index) {
|
|
293
|
-
if (!this.paramNameConfigCache) {
|
|
294
|
-
this.paramNameConfigCache = {
|
|
295
|
-
key: 'name',
|
|
296
|
-
placeholder: 'Parameter name',
|
|
297
|
-
multiple: false,
|
|
298
|
-
searchable: false,
|
|
299
|
-
options: []
|
|
300
|
-
};
|
|
301
|
-
}
|
|
302
|
-
return this.paramNameConfigCache;
|
|
303
|
-
}
|
|
304
|
-
getHeaderFormGroup(index) {
|
|
305
|
-
return this.headersFormArray.at(index);
|
|
306
|
-
}
|
|
307
|
-
getParamFormGroup(index) {
|
|
308
|
-
return this.paramsFormArray.at(index);
|
|
309
|
-
}
|
|
310
|
-
onTabChange(tab) {
|
|
311
|
-
this.selectedTab = tab;
|
|
312
|
-
}
|
|
313
|
-
onSendRequest() {
|
|
314
|
-
this.formSubmitted = true;
|
|
315
|
-
this.markFormGroupTouched(this.apiForm);
|
|
316
|
-
console.log('onSendRequest: apiForm', this.apiForm.value);
|
|
317
|
-
console.log('onSendRequest: apiForm valid', this.apiForm);
|
|
318
|
-
if (this.apiForm.valid) {
|
|
319
|
-
const formValue = this.apiForm.value;
|
|
320
|
-
const headers = formValue.headers.map((h) => ({
|
|
321
|
-
name: h.name,
|
|
322
|
-
value: h.value
|
|
323
|
-
}));
|
|
324
|
-
const params = formValue.params ? formValue.params.map((p) => ({
|
|
325
|
-
name: p.name,
|
|
326
|
-
value: p.value
|
|
327
|
-
})) : [];
|
|
328
|
-
this.sendRequest.emit({
|
|
329
|
-
method: formValue.method,
|
|
330
|
-
url: formValue.url,
|
|
331
|
-
headers,
|
|
332
|
-
body: formValue.body,
|
|
333
|
-
params
|
|
334
|
-
});
|
|
335
|
-
}
|
|
336
|
-
}
|
|
337
|
-
onImportCurl() {
|
|
338
|
-
this.importCurl.emit();
|
|
339
|
-
}
|
|
340
|
-
onCancel() {
|
|
341
|
-
this.cancelled.emit();
|
|
342
|
-
}
|
|
343
|
-
onCreateStep() {
|
|
344
|
-
this.formSubmitted = true;
|
|
345
|
-
this.markFormGroupTouched(this.apiForm);
|
|
346
|
-
if (this.apiForm.valid) {
|
|
347
|
-
const formValue = this.apiForm.value;
|
|
348
|
-
const stepData = {
|
|
349
|
-
method: formValue.method,
|
|
350
|
-
url: formValue.url,
|
|
351
|
-
headers: formValue.headers.map((h) => ({
|
|
352
|
-
name: h.name,
|
|
353
|
-
value: h.value
|
|
354
|
-
})),
|
|
355
|
-
body: formValue.body,
|
|
356
|
-
params: formValue.params ? formValue.params.map((p) => ({
|
|
357
|
-
name: p.name,
|
|
358
|
-
value: p.value
|
|
359
|
-
})) : [],
|
|
360
|
-
scripts: formValue.scripts,
|
|
361
|
-
variableName: formValue.variableName || '',
|
|
362
|
-
validation: formValue.validation ? formValue.validation.map((v) => ({
|
|
363
|
-
jsonPath: v.jsonPath,
|
|
364
|
-
verificationType: v.verificationType,
|
|
365
|
-
expectedType: v.expectedType,
|
|
366
|
-
expectedValue: v.expectedValue,
|
|
367
|
-
result: v.result || 'Not run'
|
|
368
|
-
})) : []
|
|
369
|
-
};
|
|
370
|
-
console.log('9999999', stepData);
|
|
371
|
-
this.createStep.emit(stepData);
|
|
372
|
-
}
|
|
373
|
-
}
|
|
374
|
-
onNext() {
|
|
375
|
-
// Move to next step or create step if on last step
|
|
376
|
-
const stepOrder = ['request-details', 'store-response', 'validation'];
|
|
377
|
-
const currentIndex = stepOrder.indexOf(this.selectedProgressStep);
|
|
378
|
-
if (currentIndex < stepOrder.length - 1) {
|
|
379
|
-
this.onProgressStepChange(stepOrder[currentIndex + 1]);
|
|
380
|
-
}
|
|
381
|
-
else {
|
|
382
|
-
this.onCreateStep();
|
|
383
|
-
}
|
|
384
|
-
}
|
|
385
|
-
onBack() {
|
|
386
|
-
// Move to previous step
|
|
387
|
-
const stepOrder = ['request-details', 'store-response', 'validation'];
|
|
388
|
-
const currentIndex = stepOrder.indexOf(this.selectedProgressStep);
|
|
389
|
-
if (currentIndex > 0) {
|
|
390
|
-
this.onProgressStepChange(stepOrder[currentIndex - 1]);
|
|
391
|
-
}
|
|
392
|
-
}
|
|
393
|
-
canGoBack() {
|
|
394
|
-
const stepOrder = ['request-details', 'store-response', 'validation'];
|
|
395
|
-
const currentIndex = stepOrder.indexOf(this.selectedProgressStep);
|
|
396
|
-
return currentIndex > 0;
|
|
397
|
-
}
|
|
398
|
-
isLastStep() {
|
|
399
|
-
const stepOrder = ['request-details', 'store-response', 'validation'];
|
|
400
|
-
const currentIndex = stepOrder.indexOf(this.selectedProgressStep);
|
|
401
|
-
return currentIndex === stepOrder.length - 1;
|
|
402
|
-
}
|
|
403
|
-
getProgressStepClass(step) {
|
|
404
|
-
const stepOrder = ['request-details', 'store-response', 'validation'];
|
|
405
|
-
const currentIndex = stepOrder.indexOf(this.currentStep);
|
|
406
|
-
const stepIndex = stepOrder.indexOf(step);
|
|
407
|
-
if (stepIndex < currentIndex) {
|
|
408
|
-
return 'cqa-text-gray-400'; // Completed
|
|
409
|
-
}
|
|
410
|
-
else if (stepIndex === currentIndex) {
|
|
411
|
-
return 'cqa-text-purple-600 cqa-font-semibold'; // Active
|
|
412
|
-
}
|
|
413
|
-
else {
|
|
414
|
-
return 'cqa-text-gray-400'; // Pending
|
|
415
|
-
}
|
|
416
|
-
}
|
|
417
|
-
getProgressStepNumber(step) {
|
|
418
|
-
const stepOrder = ['request-details', 'store-response', 'validation'];
|
|
419
|
-
return stepOrder.indexOf(step) + 1;
|
|
420
|
-
}
|
|
421
|
-
getProgressStepLabel(step) {
|
|
422
|
-
const labels = {
|
|
423
|
-
'request-details': 'Request Details',
|
|
424
|
-
'store-response': 'Store Response',
|
|
425
|
-
'validation': 'Validation'
|
|
426
|
-
};
|
|
427
|
-
return labels[step];
|
|
428
|
-
}
|
|
429
|
-
formatJsonResponse(data) {
|
|
430
|
-
if (!data)
|
|
431
|
-
return '';
|
|
432
|
-
try {
|
|
433
|
-
return JSON.stringify(data, null, 2);
|
|
434
|
-
}
|
|
435
|
-
catch {
|
|
436
|
-
return String(data);
|
|
437
|
-
}
|
|
438
|
-
}
|
|
439
|
-
onProgressStepChange(step) {
|
|
440
|
-
this.selectedProgressStep = step;
|
|
441
|
-
// Initialize validation form array if switching to validation step
|
|
442
|
-
if (step === 'validation' && this.validationFormArray.length === 0) {
|
|
443
|
-
this.addValidationRule();
|
|
444
|
-
}
|
|
445
|
-
}
|
|
446
|
-
addValidationRule() {
|
|
447
|
-
const validationGroup = this.fb.group({
|
|
448
|
-
checked: [false],
|
|
449
|
-
jsonPath: ['', Validators.required],
|
|
450
|
-
verificationType: ['equals'],
|
|
451
|
-
expectedType: ['string'],
|
|
452
|
-
expectedValue: ['', Validators.required],
|
|
453
|
-
result: ['Not run']
|
|
454
|
-
});
|
|
455
|
-
// Ensure all controls are enabled to avoid disabled attribute warnings
|
|
456
|
-
validationGroup.enable({ emitEvent: false });
|
|
457
|
-
this.validationFormArray.push(validationGroup);
|
|
458
|
-
}
|
|
459
|
-
removeValidationRule(index) {
|
|
460
|
-
this.validationFormArray.removeAt(index);
|
|
461
|
-
}
|
|
462
|
-
getValidationFormGroup(index) {
|
|
463
|
-
return this.validationFormArray.at(index);
|
|
464
|
-
}
|
|
465
|
-
trackByValidationRuleIndex(index) {
|
|
466
|
-
return index;
|
|
467
|
-
}
|
|
468
|
-
trackByHeaderIndex(index) {
|
|
469
|
-
return index;
|
|
470
|
-
}
|
|
471
|
-
trackByParamIndex(index) {
|
|
472
|
-
return index;
|
|
473
|
-
}
|
|
474
|
-
getVerificationTypeOptions() {
|
|
475
|
-
return [
|
|
476
|
-
{ id: 'equals', value: 'equals', name: 'Equals', label: 'Equals' },
|
|
477
|
-
{ id: 'not_equals', value: 'not_equals', name: 'Not Equals', label: 'Not Equals' },
|
|
478
|
-
{ id: 'contains', value: 'contains', name: 'Contains', label: 'Contains' },
|
|
479
|
-
{ id: 'not_contains', value: 'not_contains', name: 'Not Contains', label: 'Not Contains' },
|
|
480
|
-
{ id: 'greater_than', value: 'greater_than', name: 'Greater Than', label: 'Greater Than' },
|
|
481
|
-
{ id: 'less_than', value: 'less_than', name: 'Less Than', label: 'Less Than' },
|
|
482
|
-
{ id: 'greater_than_or_equals', value: 'greater_than_or_equals', name: 'Greater Than Or Equals', label: 'Greater Than Or Equals' },
|
|
483
|
-
{ id: 'less_than_or_equals', value: 'less_than_or_equals', name: 'Less Than Or Equals', label: 'Less Than Or Equals' },
|
|
484
|
-
{ id: 'exists', value: 'exists', name: 'Exists', label: 'Exists' },
|
|
485
|
-
{ id: 'not_exists', value: 'not_exists', name: 'Not Exists', label: 'Not Exists' }
|
|
486
|
-
];
|
|
487
|
-
}
|
|
488
|
-
getExpectedTypeOptions() {
|
|
489
|
-
return [
|
|
490
|
-
{ id: 'string', value: 'string', name: 'String', label: 'String' },
|
|
491
|
-
{ id: 'number', value: 'number', name: 'Number', label: 'Number' },
|
|
492
|
-
{ id: 'boolean', value: 'boolean', name: 'Boolean', label: 'Boolean' },
|
|
493
|
-
{ id: 'object', value: 'object', name: 'Object', label: 'Object' },
|
|
494
|
-
{ id: 'array', value: 'array', name: 'Array', label: 'Array' },
|
|
495
|
-
{ id: 'null', value: 'null', name: 'Null', label: 'Null' }
|
|
496
|
-
];
|
|
497
|
-
}
|
|
498
|
-
getVerificationTypeConfig(index) {
|
|
499
|
-
// Cache the config to prevent infinite change detection loops
|
|
500
|
-
if (!this.verificationTypeConfigCache) {
|
|
501
|
-
this.verificationTypeConfigCache = {
|
|
502
|
-
key: 'verificationType',
|
|
503
|
-
placeholder: 'Select verification type',
|
|
504
|
-
multiple: false,
|
|
505
|
-
searchable: false,
|
|
506
|
-
options: this.getVerificationTypeOptions()
|
|
507
|
-
};
|
|
508
|
-
}
|
|
509
|
-
return this.verificationTypeConfigCache;
|
|
510
|
-
}
|
|
511
|
-
getExpectedTypeConfig(index) {
|
|
512
|
-
// Cache the config to prevent infinite change detection loops
|
|
513
|
-
if (!this.expectedTypeConfigCache) {
|
|
514
|
-
this.expectedTypeConfigCache = {
|
|
515
|
-
key: 'expectedType',
|
|
516
|
-
placeholder: 'Select expected type',
|
|
517
|
-
multiple: false,
|
|
518
|
-
searchable: false,
|
|
519
|
-
options: this.getExpectedTypeOptions()
|
|
520
|
-
};
|
|
521
|
-
}
|
|
522
|
-
return this.expectedTypeConfigCache;
|
|
523
|
-
}
|
|
524
|
-
getResultClass(result) {
|
|
525
|
-
if (result === 'Pass') {
|
|
526
|
-
return 'cqa-bg-green-500 cqa-text-white';
|
|
527
|
-
}
|
|
528
|
-
else if (result === 'Fail') {
|
|
529
|
-
return 'cqa-bg-red-500 cqa-text-white';
|
|
530
|
-
}
|
|
531
|
-
return 'cqa-bg-gray-300 cqa-text-gray-700';
|
|
532
|
-
}
|
|
533
|
-
get allValidationRulesSelected() {
|
|
534
|
-
return this.validationFormArray.length > 0 &&
|
|
535
|
-
this.validationFormArray.controls.every(control => control.get('checked')?.value);
|
|
536
|
-
}
|
|
537
|
-
get someValidationRulesSelected() {
|
|
538
|
-
return this.validationFormArray.controls.some(control => control.get('checked')?.value) &&
|
|
539
|
-
!this.allValidationRulesSelected;
|
|
540
|
-
}
|
|
541
|
-
onSelectAllValidationRules(checked) {
|
|
542
|
-
this.validationFormArray.controls.forEach(control => {
|
|
543
|
-
control.get('checked')?.setValue(checked);
|
|
544
|
-
});
|
|
545
|
-
}
|
|
546
|
-
onDeleteSelectedValidationRules() {
|
|
547
|
-
const indicesToRemove = [];
|
|
548
|
-
this.validationFormArray.controls.forEach((control, index) => {
|
|
549
|
-
if (control.get('checked')?.value) {
|
|
550
|
-
indicesToRemove.push(index);
|
|
551
|
-
}
|
|
552
|
-
});
|
|
553
|
-
// Remove in reverse order to maintain correct indices
|
|
554
|
-
indicesToRemove.reverse().forEach(index => {
|
|
555
|
-
this.removeValidationRule(index);
|
|
556
|
-
});
|
|
557
|
-
}
|
|
558
|
-
markFormGroupTouched(formGroup) {
|
|
559
|
-
Object.keys(formGroup.controls).forEach(key => {
|
|
560
|
-
const control = formGroup.get(key);
|
|
561
|
-
if (control) {
|
|
562
|
-
control.markAsTouched();
|
|
563
|
-
if (control instanceof FormGroup) {
|
|
564
|
-
this.markFormGroupTouched(control);
|
|
565
|
-
}
|
|
566
|
-
else if (control instanceof FormArray) {
|
|
567
|
-
control.controls.forEach((ctrl) => {
|
|
568
|
-
if (ctrl instanceof FormGroup) {
|
|
569
|
-
this.markFormGroupTouched(ctrl);
|
|
570
|
-
}
|
|
571
|
-
else {
|
|
572
|
-
ctrl.markAsTouched();
|
|
573
|
-
}
|
|
574
|
-
});
|
|
575
|
-
}
|
|
576
|
-
}
|
|
577
|
-
});
|
|
578
|
-
}
|
|
579
|
-
onUrlChange(value) {
|
|
580
|
-
const urlControl = this.apiForm.get('url');
|
|
581
|
-
if (urlControl) {
|
|
582
|
-
urlControl.setValue(value);
|
|
583
|
-
urlControl.updateValueAndValidity();
|
|
584
|
-
}
|
|
585
|
-
}
|
|
586
|
-
onUrlBlur() {
|
|
587
|
-
const urlControl = this.apiForm.get('url');
|
|
588
|
-
if (urlControl) {
|
|
589
|
-
urlControl.markAsTouched();
|
|
590
|
-
}
|
|
591
|
-
}
|
|
592
|
-
getUrlErrors() {
|
|
593
|
-
const urlControl = this.apiForm.get('url');
|
|
594
|
-
if (!urlControl || (!this.formSubmitted && !urlControl.touched)) {
|
|
595
|
-
return [];
|
|
596
|
-
}
|
|
597
|
-
const errors = [];
|
|
598
|
-
if (urlControl.errors?.['required']) {
|
|
599
|
-
errors.push('URL is required');
|
|
600
|
-
}
|
|
601
|
-
else if (urlControl.errors?.['pattern']) {
|
|
602
|
-
errors.push('URL is not valid');
|
|
603
|
-
}
|
|
604
|
-
else if (urlControl.errors?.['hasParameter']) {
|
|
605
|
-
errors.push('URL format is invalid. Please check the parameter format.');
|
|
606
|
-
}
|
|
607
|
-
return errors;
|
|
608
|
-
}
|
|
609
|
-
}
|
|
610
|
-
StepBuilderApiComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepBuilderApiComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component });
|
|
611
|
-
StepBuilderApiComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: StepBuilderApiComponent, selector: "cqa-step-builder-api", inputs: { httpMethodOptions: "httpMethodOptions", headerNameOptions: "headerNameOptions", currentStep: "currentStep", responsePreview: "responsePreview", isLoading: "isLoading", initialData: "initialData" }, outputs: { createStep: "createStep", cancelled: "cancelled", sendRequest: "sendRequest", importCurl: "importCurl" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n Create API Test Step\n </h2>\n\n <!-- Progress Tracker -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-4 cqa-mb-6 cqa-pb-4 cqa-border-b cqa-border-gray-200\">\n <!-- <div *ngFor=\"let step of progressSteps\"\n class=\"cqa-flex cqa-items-center cqa-gap-2\"\n [ngClass]=\"getProgressStepClass(step)\">\n <span class=\"cqa-text-sm cqa-font-medium\">{{ getProgressStepNumber(step) }} {{ getProgressStepLabel(step) }}</span>\n <span *ngIf=\"step !== 'validation'\" class=\"cqa-text-gray-300\">\u2022</span>\n </div> -->\n <div class=\"cqa-flex cqa-items-center cqa-w-full cqa-justify-between\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\" [ngClass]=\"{'cqa-text-purple-600 cqa-font-semibold': selectedProgressStep === 'request-details'}\" (click)=\"onProgressStepChange('request-details')\">\n <div class=\"cqa-w-8 cqa-h-8 cqa-flex cqa-items-center cqa-justify-center cqa-rounded-full cqa-bg-purple-600 cqa-text-white cqa-text-xs cqa-font-semibold\">1</div>\n <div>Request Details</div>\n <div></div>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\" [ngClass]=\"{'cqa-text-purple-600 cqa-font-semibold': selectedProgressStep === 'store-response'}\" (click)=\"onProgressStepChange('store-response')\">\n <div class=\"cqa-w-8 cqa-h-8 cqa-flex cqa-items-center cqa-justify-center cqa-rounded-full cqa-bg-purple-600 cqa-text-white cqa-text-xs cqa-font-semibold\">2</div>\n <div>Store Response</div>\n <div></div>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\" [ngClass]=\"{'cqa-text-purple-600 cqa-font-semibold': selectedProgressStep === 'validation'}\" (click)=\"onProgressStepChange('validation')\">\n <div class=\"cqa-w-8 cqa-h-8 cqa-flex cqa-items-center cqa-justify-center cqa-rounded-full cqa-bg-purple-600 cqa-text-white cqa-text-xs cqa-font-semibold\">3</div>\n <div>Validation</div>\n <div></div>\n </div>\n </div>\n </div>\n <ng-container *ngIf=\"selectedProgressStep === 'request-details'\">\n <!-- API Request Configuration -->\n <div class=\"cqa-flex cqa-items-start cqa-gap-2 cqa-mb-3\">\n <div class=\"cqa-w-32 cqa-flex-shrink-0\">\n <cqa-dynamic-select\n [form]=\"apiForm\"\n [config]=\"getHttpMethodConfig()\">\n </cqa-dynamic-select>\n </div>\n <div class=\"cqa-flex-1\">\n <cqa-custom-input\n [placeholder]=\"'Enter API endpoint URL'\"\n [value]=\"apiForm.get('url')?.value || ''\"\n [fullWidth]=\"true\"\n [errors]=\"getUrlErrors()\"\n (valueChange)=\"onUrlChange($event)\"\n (blurred)=\"onUrlBlur()\">\n </cqa-custom-input>\n </div>\n <cqa-button\n variant=\"outlined\"\n text=\"Import API cURL\"\n (clicked)=\"onImportCurl()\">\n </cqa-button>\n <cqa-button\n variant=\"filled\"\n text=\"Send Request\"\n (clicked)=\"onSendRequest()\"\n [disabled]=\"!apiForm.get('url')?.value || isLoading\">\n </cqa-button>\n </div>\n\n <!-- Request Details Tabs -->\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n <!-- Tab Navigation -->\n <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n <button\n type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'headers'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'headers'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'headers'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'headers'\"\n (click)=\"onTabChange('headers')\">\n Headers\n </button>\n <button\n type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'body'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'body'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'body'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'body'\"\n (click)=\"onTabChange('body')\">\n Body\n </button>\n <button\n type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'params'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'params'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'params'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'params'\"\n (click)=\"onTabChange('params')\">\n Params\n </button>\n <button\n type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'scripts'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'scripts'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'scripts'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'scripts'\"\n (click)=\"onTabChange('scripts')\">\n Scripts\n </button>\n </div>\n\n <!-- Tab Content -->\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-4 cqa-border cqa-border-gray-200 cqa-border-t-0 cqa-rounded-b-lg\">\n <!-- Headers Tab -->\n <div *ngIf=\"selectedTab === 'headers'\" class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4 cqa-items-start\">\n <div class=\"cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Header Name*</div>\n <div class=\"cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Header Value*</div>\n </div>\n <div *ngFor=\"let header of headersFormArray.controls; let i = index; trackBy: trackByHeaderIndex\" \n class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4 cqa-items-center\">\n <div>\n <cqa-dynamic-select\n [form]=\"getHeaderFormGroup(i)\"\n [config]=\"getHeaderNameConfig(i)\">\n </cqa-dynamic-select>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div class=\"cqa-flex-1\">\n <cqa-custom-input\n [placeholder]=\"'Header value'\"\n [value]=\"getHeaderFormGroup(i).get('value')?.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"getHeaderFormGroup(i).get('value')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n <button\n type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-8 cqa-h-8 cqa-rounded cqa-text-red-500 hover:cqa-text-red-700 hover:cqa-bg-red-50 cqa-transition-colors\"\n (click)=\"removeHeader(i)\"\n [attr.aria-label]=\"'Remove header'\">\n <mat-icon class=\"cqa-text-lg\">delete</mat-icon>\n </button>\n </div>\n </div>\n <button\n type=\"button\"\n class=\"cqa-text-blue-600 cqa-text-sm cqa-font-medium cqa-self-start cqa-flex cqa-items-center cqa-gap-1 hover:cqa-text-blue-700 cqa-transition-colors\"\n (click)=\"addHeader()\">\n <mat-icon class=\"cqa-text-base\">add</mat-icon>\n <span>Add Header</span>\n </button>\n </div>\n\n <!-- Body Tab -->\n <div *ngIf=\"selectedTab === 'body'\" class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <cqa-custom-textarea\n [placeholder]=\"'Enter request body (JSON, XML, etc.)'\"\n [value]=\"apiForm.get('body')?.value || ''\"\n [fullWidth]=\"true\"\n [rows]=\"10\"\n (valueChange)=\"apiForm.get('body')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n\n <!-- Params Tab -->\n <div *ngIf=\"selectedTab === 'params'\" class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4 cqa-items-start\">\n <div class=\"cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Parameter Name</div>\n <div class=\"cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Parameter Value</div>\n </div>\n <div *ngFor=\"let param of paramsFormArray.controls; let i = index; trackBy: trackByParamIndex\" \n class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4 cqa-items-center\">\n <div>\n <cqa-custom-input\n [placeholder]=\"'Parameter name'\"\n [value]=\"getParamFormGroup(i).get('name')?.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"getParamFormGroup(i).get('name')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div class=\"cqa-flex-1\">\n <cqa-custom-input\n [placeholder]=\"'Parameter value'\"\n [value]=\"getParamFormGroup(i).get('value')?.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"getParamFormGroup(i).get('value')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n <button\n type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-8 cqa-h-8 cqa-rounded cqa-text-red-500 hover:cqa-text-red-700 hover:cqa-bg-red-50 cqa-transition-colors\"\n (click)=\"removeParam(i)\"\n [attr.aria-label]=\"'Remove parameter'\">\n <mat-icon class=\"cqa-text-lg\">delete</mat-icon>\n </button>\n </div>\n </div>\n <button\n type=\"button\"\n class=\"cqa-text-blue-600 cqa-text-sm cqa-font-medium cqa-self-start cqa-flex cqa-items-center cqa-gap-1 hover:cqa-text-blue-700 cqa-transition-colors\"\n (click)=\"addParam()\">\n <mat-icon class=\"cqa-text-base\">add</mat-icon>\n <span>Add Parameter</span>\n </button>\n </div>\n\n <!-- Scripts Tab -->\n <div *ngIf=\"selectedTab === 'scripts'\" class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <cqa-custom-textarea\n [placeholder]=\"'Enter scripts (JavaScript, etc.)'\"\n [value]=\"apiForm.get('scripts')?.value || ''\"\n [fullWidth]=\"true\"\n [rows]=\"10\"\n (valueChange)=\"apiForm.get('scripts')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n </div>\n </div>\n\n\n</ng-container>\n<ng-container *ngIf=\"selectedProgressStep === 'store-response'\">\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-max-h-[500px] cqa-overflow-y-auto\">\n <!-- Store Response Title -->\n <h3 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n Store Response\n </h3>\n\n <!-- Variable Name Input -->\n <div class=\"cqa-mb-6\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-gray-700 cqa-mb-1 cqa-block\">\n Variable Name\n </label>\n <cqa-custom-input\n [placeholder]=\"'Variable Name'\"\n [value]=\"apiForm.get('variableName')?.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"apiForm.get('variableName')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n </div>\n</ng-container>\n<ng-container *ngIf=\"selectedProgressStep === 'validation'\">\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-max-h-[500px] cqa-overflow-y-auto\">\n <!-- Validation Rules Title -->\n <h3 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n Validation Rules\n </h3>\n\n <!-- Validation Rules Table -->\n <div class=\"cqa-bg-white cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden cqa-mb-4\">\n <!-- Table Header -->\n <div class=\"cqa-grid cqa-grid-cols-6 cqa-gap-2 cqa-p-3 cqa-bg-gray-50 cqa-border-b cqa-border-gray-200 cqa-items-center\">\n <div class=\"cqa-flex cqa-items-center\">\n <mat-checkbox\n [checked]=\"allValidationRulesSelected\"\n [indeterminate]=\"someValidationRulesSelected\"\n (change)=\"onSelectAllValidationRules($event.checked)\">\n </mat-checkbox>\n </div>\n <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">jsonPath</div>\n <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">verificationType</div>\n <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">expectedType</div>\n <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">expectedValue</div>\n <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Result</div>\n </div>\n\n <!-- Table Body -->\n <div class=\"cqa-flex cqa-flex-col\">\n <div *ngFor=\"let validationRule of validationFormArray.controls; let i = index; trackBy: trackByValidationRuleIndex\" \n [formGroup]=\"getValidationFormGroup(i)\"\n class=\"cqa-grid cqa-grid-cols-6 cqa-gap-2 cqa-p-3 cqa-border-b cqa-border-gray-200 cqa-items-center cqa-last:border-b-0\">\n <!-- Checkbox -->\n <div class=\"cqa-flex cqa-items-center\">\n <mat-checkbox formControlName=\"checked\">\n </mat-checkbox>\n </div>\n\n <!-- jsonPath -->\n <div>\n <cqa-custom-input\n [placeholder]=\"'jsonPath'\"\n [value]=\"getValidationFormGroup(i).get('jsonPath')?.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"getValidationFormGroup(i).get('jsonPath')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- verificationType -->\n <div>\n <cqa-dynamic-select\n [form]=\"getValidationFormGroup(i)\"\n [config]=\"getVerificationTypeConfig(i)\">\n </cqa-dynamic-select>\n </div>\n\n <!-- expectedType -->\n <div>\n <cqa-dynamic-select\n [form]=\"getValidationFormGroup(i)\"\n [config]=\"getExpectedTypeConfig(i)\">\n </cqa-dynamic-select>\n </div>\n\n <!-- expectedValue -->\n <div>\n <cqa-custom-input\n [placeholder]=\"'expectedValue'\"\n [value]=\"getValidationFormGroup(i).get('expectedValue')?.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"getValidationFormGroup(i).get('expectedValue')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- Result and Delete -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <button\n type=\"button\"\n class=\"cqa-px-3 cqa-py-1 cqa-rounded cqa-text-[10px] cqa-font-medium\"\n [ngClass]=\"getResultClass(getValidationFormGroup(i).get('result')?.value || 'Not run')\">\n {{ getValidationFormGroup(i).get('result')?.value || 'Not run' }}\n </button>\n <button\n type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-6 cqa-h-6 cqa-rounded cqa-text-red-500 hover:cqa-text-red-700 hover:cqa-bg-red-50 cqa-transition-colors\"\n (click)=\"removeValidationRule(i)\"\n [attr.aria-label]=\"'Delete validation rule'\">\n <mat-icon class=\"cqa-text-base\">delete</mat-icon>\n </button>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Add Another Rule and Delete Selected -->\n <div class=\"cqa-flex cqa-justify-between cqa-items-center cqa-mb-4\">\n <button\n *ngIf=\"someValidationRulesSelected || allValidationRulesSelected\"\n type=\"button\"\n class=\"cqa-text-red-600 cqa-text-sm cqa-font-medium cqa-flex cqa-items-center cqa-gap-1 hover:cqa-text-red-700 cqa-transition-colors\"\n (click)=\"onDeleteSelectedValidationRules()\">\n <mat-icon class=\"cqa-text-base\">delete</mat-icon>\n <span>Delete Selected</span>\n </button>\n <button\n type=\"button\"\n class=\"cqa-text-blue-600 cqa-text-sm cqa-font-medium cqa-flex cqa-items-center cqa-gap-1 hover:cqa-text-blue-700 cqa-transition-colors\"\n (click)=\"addValidationRule()\">\n <mat-icon class=\"cqa-text-base\">add</mat-icon>\n <span>Add Another</span>\n </button>\n </div>\n </div>\n</ng-container>\n\n\n <!-- Response Preview Section -->\n <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden cqa-mb-6\">\n <div class=\"cqa-p-3 cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Response Preview</h3>\n </div>\n <div class=\"cqa-p-4 cqa-bg-gray-50 cqa-overflow-auto\" style=\"max-height: 300px;\">\n <pre *ngIf=\"responsePreview\" class=\"cqa-text-xs cqa-font-mono cqa-text-gray-800 cqa-whitespace-pre-wrap\">{{ formatJsonResponse(responsePreview) }}</pre>\n <p *ngIf=\"!responsePreview\" class=\"cqa-text-sm cqa-text-gray-400 cqa-text-center cqa-py-8\">\n No response yet. Send a request to see the response preview.\n </p>\n </div>\n </div>\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n <!-- Cancel button (only on first step) -->\n <cqa-button\n *ngIf=\"selectedProgressStep === 'request-details'\"\n class=\"cqa-w-1/2\"\n variant=\"outlined\"\n text=\"Cancel\"\n [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <!-- Back button (on all steps except first) -->\n <cqa-button\n *ngIf=\"selectedProgressStep !== 'request-details'\"\n class=\"cqa-w-1/2\"\n variant=\"outlined\"\n text=\"Back\"\n [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onBack()\">\n </cqa-button>\n <!-- Next/Create Step button -->\n <cqa-button\n class=\"cqa-w-1/2\"\n variant=\"filled\"\n [text]=\"isLastStep() ? 'Create Step' : 'Next'\"\n [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"isLastStep() ? onCreateStep() : onNext()\">\n </cqa-button>\n </div>\n</div>\n\n", components: [{ type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }, { 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.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i6.CustomTextareaComponent, selector: "cqa-custom-textarea", inputs: ["label", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "rows", "cols", "resize", "textareaInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused"] }, { type: i7.MatCheckbox, selector: "mat-checkbox", inputs: ["disableRipple", "color", "tabIndex", "aria-label", "aria-labelledby", "aria-describedby", "id", "required", "labelPosition", "name", "value", "checked", "disabled", "indeterminate"], outputs: ["change", "indeterminateChange"], exportAs: ["matCheckbox"] }], directives: [{ type: i8.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i8.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i8.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }] });
|
|
612
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepBuilderApiComponent, decorators: [{
|
|
613
|
-
type: Component,
|
|
614
|
-
args: [{ selector: 'cqa-step-builder-api', host: { class: 'cqa-ui-root' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n Create API Test Step\n </h2>\n\n <!-- Progress Tracker -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-4 cqa-mb-6 cqa-pb-4 cqa-border-b cqa-border-gray-200\">\n <!-- <div *ngFor=\"let step of progressSteps\"\n class=\"cqa-flex cqa-items-center cqa-gap-2\"\n [ngClass]=\"getProgressStepClass(step)\">\n <span class=\"cqa-text-sm cqa-font-medium\">{{ getProgressStepNumber(step) }} {{ getProgressStepLabel(step) }}</span>\n <span *ngIf=\"step !== 'validation'\" class=\"cqa-text-gray-300\">\u2022</span>\n </div> -->\n <div class=\"cqa-flex cqa-items-center cqa-w-full cqa-justify-between\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\" [ngClass]=\"{'cqa-text-purple-600 cqa-font-semibold': selectedProgressStep === 'request-details'}\" (click)=\"onProgressStepChange('request-details')\">\n <div class=\"cqa-w-8 cqa-h-8 cqa-flex cqa-items-center cqa-justify-center cqa-rounded-full cqa-bg-purple-600 cqa-text-white cqa-text-xs cqa-font-semibold\">1</div>\n <div>Request Details</div>\n <div></div>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\" [ngClass]=\"{'cqa-text-purple-600 cqa-font-semibold': selectedProgressStep === 'store-response'}\" (click)=\"onProgressStepChange('store-response')\">\n <div class=\"cqa-w-8 cqa-h-8 cqa-flex cqa-items-center cqa-justify-center cqa-rounded-full cqa-bg-purple-600 cqa-text-white cqa-text-xs cqa-font-semibold\">2</div>\n <div>Store Response</div>\n <div></div>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\" [ngClass]=\"{'cqa-text-purple-600 cqa-font-semibold': selectedProgressStep === 'validation'}\" (click)=\"onProgressStepChange('validation')\">\n <div class=\"cqa-w-8 cqa-h-8 cqa-flex cqa-items-center cqa-justify-center cqa-rounded-full cqa-bg-purple-600 cqa-text-white cqa-text-xs cqa-font-semibold\">3</div>\n <div>Validation</div>\n <div></div>\n </div>\n </div>\n </div>\n <ng-container *ngIf=\"selectedProgressStep === 'request-details'\">\n <!-- API Request Configuration -->\n <div class=\"cqa-flex cqa-items-start cqa-gap-2 cqa-mb-3\">\n <div class=\"cqa-w-32 cqa-flex-shrink-0\">\n <cqa-dynamic-select\n [form]=\"apiForm\"\n [config]=\"getHttpMethodConfig()\">\n </cqa-dynamic-select>\n </div>\n <div class=\"cqa-flex-1\">\n <cqa-custom-input\n [placeholder]=\"'Enter API endpoint URL'\"\n [value]=\"apiForm.get('url')?.value || ''\"\n [fullWidth]=\"true\"\n [errors]=\"getUrlErrors()\"\n (valueChange)=\"onUrlChange($event)\"\n (blurred)=\"onUrlBlur()\">\n </cqa-custom-input>\n </div>\n <cqa-button\n variant=\"outlined\"\n text=\"Import API cURL\"\n (clicked)=\"onImportCurl()\">\n </cqa-button>\n <cqa-button\n variant=\"filled\"\n text=\"Send Request\"\n (clicked)=\"onSendRequest()\"\n [disabled]=\"!apiForm.get('url')?.value || isLoading\">\n </cqa-button>\n </div>\n\n <!-- Request Details Tabs -->\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n <!-- Tab Navigation -->\n <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n <button\n type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'headers'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'headers'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'headers'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'headers'\"\n (click)=\"onTabChange('headers')\">\n Headers\n </button>\n <button\n type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'body'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'body'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'body'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'body'\"\n (click)=\"onTabChange('body')\">\n Body\n </button>\n <button\n type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'params'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'params'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'params'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'params'\"\n (click)=\"onTabChange('params')\">\n Params\n </button>\n <button\n type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'scripts'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'scripts'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'scripts'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'scripts'\"\n (click)=\"onTabChange('scripts')\">\n Scripts\n </button>\n </div>\n\n <!-- Tab Content -->\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-4 cqa-border cqa-border-gray-200 cqa-border-t-0 cqa-rounded-b-lg\">\n <!-- Headers Tab -->\n <div *ngIf=\"selectedTab === 'headers'\" class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4 cqa-items-start\">\n <div class=\"cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Header Name*</div>\n <div class=\"cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Header Value*</div>\n </div>\n <div *ngFor=\"let header of headersFormArray.controls; let i = index; trackBy: trackByHeaderIndex\" \n class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4 cqa-items-center\">\n <div>\n <cqa-dynamic-select\n [form]=\"getHeaderFormGroup(i)\"\n [config]=\"getHeaderNameConfig(i)\">\n </cqa-dynamic-select>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div class=\"cqa-flex-1\">\n <cqa-custom-input\n [placeholder]=\"'Header value'\"\n [value]=\"getHeaderFormGroup(i).get('value')?.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"getHeaderFormGroup(i).get('value')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n <button\n type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-8 cqa-h-8 cqa-rounded cqa-text-red-500 hover:cqa-text-red-700 hover:cqa-bg-red-50 cqa-transition-colors\"\n (click)=\"removeHeader(i)\"\n [attr.aria-label]=\"'Remove header'\">\n <mat-icon class=\"cqa-text-lg\">delete</mat-icon>\n </button>\n </div>\n </div>\n <button\n type=\"button\"\n class=\"cqa-text-blue-600 cqa-text-sm cqa-font-medium cqa-self-start cqa-flex cqa-items-center cqa-gap-1 hover:cqa-text-blue-700 cqa-transition-colors\"\n (click)=\"addHeader()\">\n <mat-icon class=\"cqa-text-base\">add</mat-icon>\n <span>Add Header</span>\n </button>\n </div>\n\n <!-- Body Tab -->\n <div *ngIf=\"selectedTab === 'body'\" class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <cqa-custom-textarea\n [placeholder]=\"'Enter request body (JSON, XML, etc.)'\"\n [value]=\"apiForm.get('body')?.value || ''\"\n [fullWidth]=\"true\"\n [rows]=\"10\"\n (valueChange)=\"apiForm.get('body')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n\n <!-- Params Tab -->\n <div *ngIf=\"selectedTab === 'params'\" class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4 cqa-items-start\">\n <div class=\"cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Parameter Name</div>\n <div class=\"cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Parameter Value</div>\n </div>\n <div *ngFor=\"let param of paramsFormArray.controls; let i = index; trackBy: trackByParamIndex\" \n class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4 cqa-items-center\">\n <div>\n <cqa-custom-input\n [placeholder]=\"'Parameter name'\"\n [value]=\"getParamFormGroup(i).get('name')?.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"getParamFormGroup(i).get('name')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <div class=\"cqa-flex-1\">\n <cqa-custom-input\n [placeholder]=\"'Parameter value'\"\n [value]=\"getParamFormGroup(i).get('value')?.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"getParamFormGroup(i).get('value')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n <button\n type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-8 cqa-h-8 cqa-rounded cqa-text-red-500 hover:cqa-text-red-700 hover:cqa-bg-red-50 cqa-transition-colors\"\n (click)=\"removeParam(i)\"\n [attr.aria-label]=\"'Remove parameter'\">\n <mat-icon class=\"cqa-text-lg\">delete</mat-icon>\n </button>\n </div>\n </div>\n <button\n type=\"button\"\n class=\"cqa-text-blue-600 cqa-text-sm cqa-font-medium cqa-self-start cqa-flex cqa-items-center cqa-gap-1 hover:cqa-text-blue-700 cqa-transition-colors\"\n (click)=\"addParam()\">\n <mat-icon class=\"cqa-text-base\">add</mat-icon>\n <span>Add Parameter</span>\n </button>\n </div>\n\n <!-- Scripts Tab -->\n <div *ngIf=\"selectedTab === 'scripts'\" class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <cqa-custom-textarea\n [placeholder]=\"'Enter scripts (JavaScript, etc.)'\"\n [value]=\"apiForm.get('scripts')?.value || ''\"\n [fullWidth]=\"true\"\n [rows]=\"10\"\n (valueChange)=\"apiForm.get('scripts')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n </div>\n </div>\n\n\n</ng-container>\n<ng-container *ngIf=\"selectedProgressStep === 'store-response'\">\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-max-h-[500px] cqa-overflow-y-auto\">\n <!-- Store Response Title -->\n <h3 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n Store Response\n </h3>\n\n <!-- Variable Name Input -->\n <div class=\"cqa-mb-6\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-gray-700 cqa-mb-1 cqa-block\">\n Variable Name\n </label>\n <cqa-custom-input\n [placeholder]=\"'Variable Name'\"\n [value]=\"apiForm.get('variableName')?.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"apiForm.get('variableName')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n </div>\n</ng-container>\n<ng-container *ngIf=\"selectedProgressStep === 'validation'\">\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-max-h-[500px] cqa-overflow-y-auto\">\n <!-- Validation Rules Title -->\n <h3 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n Validation Rules\n </h3>\n\n <!-- Validation Rules Table -->\n <div class=\"cqa-bg-white cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden cqa-mb-4\">\n <!-- Table Header -->\n <div class=\"cqa-grid cqa-grid-cols-6 cqa-gap-2 cqa-p-3 cqa-bg-gray-50 cqa-border-b cqa-border-gray-200 cqa-items-center\">\n <div class=\"cqa-flex cqa-items-center\">\n <mat-checkbox\n [checked]=\"allValidationRulesSelected\"\n [indeterminate]=\"someValidationRulesSelected\"\n (change)=\"onSelectAllValidationRules($event.checked)\">\n </mat-checkbox>\n </div>\n <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">jsonPath</div>\n <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">verificationType</div>\n <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">expectedType</div>\n <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">expectedValue</div>\n <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Result</div>\n </div>\n\n <!-- Table Body -->\n <div class=\"cqa-flex cqa-flex-col\">\n <div *ngFor=\"let validationRule of validationFormArray.controls; let i = index; trackBy: trackByValidationRuleIndex\" \n [formGroup]=\"getValidationFormGroup(i)\"\n class=\"cqa-grid cqa-grid-cols-6 cqa-gap-2 cqa-p-3 cqa-border-b cqa-border-gray-200 cqa-items-center cqa-last:border-b-0\">\n <!-- Checkbox -->\n <div class=\"cqa-flex cqa-items-center\">\n <mat-checkbox formControlName=\"checked\">\n </mat-checkbox>\n </div>\n\n <!-- jsonPath -->\n <div>\n <cqa-custom-input\n [placeholder]=\"'jsonPath'\"\n [value]=\"getValidationFormGroup(i).get('jsonPath')?.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"getValidationFormGroup(i).get('jsonPath')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- verificationType -->\n <div>\n <cqa-dynamic-select\n [form]=\"getValidationFormGroup(i)\"\n [config]=\"getVerificationTypeConfig(i)\">\n </cqa-dynamic-select>\n </div>\n\n <!-- expectedType -->\n <div>\n <cqa-dynamic-select\n [form]=\"getValidationFormGroup(i)\"\n [config]=\"getExpectedTypeConfig(i)\">\n </cqa-dynamic-select>\n </div>\n\n <!-- expectedValue -->\n <div>\n <cqa-custom-input\n [placeholder]=\"'expectedValue'\"\n [value]=\"getValidationFormGroup(i).get('expectedValue')?.value || ''\"\n [fullWidth]=\"true\"\n (valueChange)=\"getValidationFormGroup(i).get('expectedValue')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- Result and Delete -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <button\n type=\"button\"\n class=\"cqa-px-3 cqa-py-1 cqa-rounded cqa-text-[10px] cqa-font-medium\"\n [ngClass]=\"getResultClass(getValidationFormGroup(i).get('result')?.value || 'Not run')\">\n {{ getValidationFormGroup(i).get('result')?.value || 'Not run' }}\n </button>\n <button\n type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-6 cqa-h-6 cqa-rounded cqa-text-red-500 hover:cqa-text-red-700 hover:cqa-bg-red-50 cqa-transition-colors\"\n (click)=\"removeValidationRule(i)\"\n [attr.aria-label]=\"'Delete validation rule'\">\n <mat-icon class=\"cqa-text-base\">delete</mat-icon>\n </button>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Add Another Rule and Delete Selected -->\n <div class=\"cqa-flex cqa-justify-between cqa-items-center cqa-mb-4\">\n <button\n *ngIf=\"someValidationRulesSelected || allValidationRulesSelected\"\n type=\"button\"\n class=\"cqa-text-red-600 cqa-text-sm cqa-font-medium cqa-flex cqa-items-center cqa-gap-1 hover:cqa-text-red-700 cqa-transition-colors\"\n (click)=\"onDeleteSelectedValidationRules()\">\n <mat-icon class=\"cqa-text-base\">delete</mat-icon>\n <span>Delete Selected</span>\n </button>\n <button\n type=\"button\"\n class=\"cqa-text-blue-600 cqa-text-sm cqa-font-medium cqa-flex cqa-items-center cqa-gap-1 hover:cqa-text-blue-700 cqa-transition-colors\"\n (click)=\"addValidationRule()\">\n <mat-icon class=\"cqa-text-base\">add</mat-icon>\n <span>Add Another</span>\n </button>\n </div>\n </div>\n</ng-container>\n\n\n <!-- Response Preview Section -->\n <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden cqa-mb-6\">\n <div class=\"cqa-p-3 cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Response Preview</h3>\n </div>\n <div class=\"cqa-p-4 cqa-bg-gray-50 cqa-overflow-auto\" style=\"max-height: 300px;\">\n <pre *ngIf=\"responsePreview\" class=\"cqa-text-xs cqa-font-mono cqa-text-gray-800 cqa-whitespace-pre-wrap\">{{ formatJsonResponse(responsePreview) }}</pre>\n <p *ngIf=\"!responsePreview\" class=\"cqa-text-sm cqa-text-gray-400 cqa-text-center cqa-py-8\">\n No response yet. Send a request to see the response preview.\n </p>\n </div>\n </div>\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n <!-- Cancel button (only on first step) -->\n <cqa-button\n *ngIf=\"selectedProgressStep === 'request-details'\"\n class=\"cqa-w-1/2\"\n variant=\"outlined\"\n text=\"Cancel\"\n [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <!-- Back button (on all steps except first) -->\n <cqa-button\n *ngIf=\"selectedProgressStep !== 'request-details'\"\n class=\"cqa-w-1/2\"\n variant=\"outlined\"\n text=\"Back\"\n [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onBack()\">\n </cqa-button>\n <!-- Next/Create Step button -->\n <cqa-button\n class=\"cqa-w-1/2\"\n variant=\"filled\"\n [text]=\"isLastStep() ? 'Create Step' : 'Next'\"\n [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"isLastStep() ? onCreateStep() : onNext()\">\n </cqa-button>\n </div>\n</div>\n\n", styles: [] }]
|
|
615
|
-
}], ctorParameters: function () { return [{ type: i1.FormBuilder }]; }, propDecorators: { httpMethodOptions: [{
|
|
616
|
-
type: Input
|
|
617
|
-
}], headerNameOptions: [{
|
|
618
|
-
type: Input
|
|
619
|
-
}], currentStep: [{
|
|
620
|
-
type: Input
|
|
621
|
-
}], responsePreview: [{
|
|
622
|
-
type: Input
|
|
623
|
-
}], isLoading: [{
|
|
624
|
-
type: Input
|
|
625
|
-
}], initialData: [{
|
|
626
|
-
type: Input
|
|
627
|
-
}], createStep: [{
|
|
628
|
-
type: Output
|
|
629
|
-
}], cancelled: [{
|
|
630
|
-
type: Output
|
|
631
|
-
}], sendRequest: [{
|
|
632
|
-
type: Output
|
|
633
|
-
}], importCurl: [{
|
|
634
|
-
type: Output
|
|
635
|
-
}] } });
|
|
636
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"step-builder-api.component.js","sourceRoot":"","sources":["../../../../../../src/lib/step-builder/step-builder-api/step-builder-api.component.ts","../../../../../../src/lib/step-builder/step-builder-api/step-builder-api.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, OnChanges, SimpleChanges, AfterViewInit } from '@angular/core';\nimport { FormArray, FormBuilder, FormGroup, FormControl, Validators } from '@angular/forms';\nimport { DynamicSelectFieldConfig, SelectOption } from '../../dynamic-select/dynamic-select-field.component';\n\nexport type HttpMethod = 'GET' | 'POST' | 'PUT' | 'PATCH' | 'DELETE' | 'HEAD' | 'OPTIONS';\nexport type RequestTab = 'headers' | 'body' | 'params' | 'scripts';\nexport type ProgressStep = 'request-details' | 'store-response' | 'validation';\n\nexport interface HeaderRow {\n  name: string;\n  value: string;\n}\n\nexport interface ValidationRule {\n  jsonPath: string;\n  verificationType: string;\n  expectedType: string;\n  expectedValue: any;\n  result?: 'Pass' | 'Fail' | 'Not run';\n  checked?: boolean;\n}\n\nexport interface ApiFormData {\n  method: HttpMethod;\n  url: string;\n  headers: HeaderRow[];\n  body?: string;\n  params?: HeaderRow[];\n  scripts?: string;\n  variableName?: string;\n  storeResponse?: any;\n  validation?: ValidationRule[];\n}\n\n@Component({\n  selector: 'cqa-step-builder-api',\n  templateUrl: './step-builder-api.component.html',\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' }\n})\nexport class StepBuilderApiComponent implements OnInit, OnChanges {\n  /** Options for HTTP method dropdown */\n  @Input() httpMethodOptions: SelectOption[] = [\n    { id: 'GET', value: 'GET', name: 'GET', label: 'GET' },\n    { id: 'POST', value: 'POST', name: 'POST', label: 'POST' },\n    { id: 'PUT', value: 'PUT', name: 'PUT', label: 'PUT' },\n    { id: 'PATCH', value: 'PATCH', name: 'PATCH', label: 'PATCH' },\n    { id: 'DELETE', value: 'DELETE', name: 'DELETE', label: 'DELETE' },\n    { id: 'HEAD', value: 'HEAD', name: 'HEAD', label: 'HEAD' },\n    { id: 'OPTIONS', value: 'OPTIONS', name: 'OPTIONS', label: 'OPTIONS' }\n  ];\n\n  /** Options for header name dropdown */\n  @Input() headerNameOptions: SelectOption[] = [];\n\n  /** Current progress step */\n  @Input() currentStep: ProgressStep = 'request-details';\n\n  /** Response preview data */\n  @Input() responsePreview: any = null;\n\n  /** Loading state */\n  @Input() isLoading: boolean = false;\n\n  /** Initial form data for edit mode */\n  @Input() initialData?: ApiFormData;\n\n  /** Emit when step is created */\n  @Output() createStep = new EventEmitter<ApiFormData>();\n\n  /** Emit when cancelled */\n  @Output() cancelled = new EventEmitter<void>();\n\n  /** Emit when request is sent */\n  @Output() sendRequest = new EventEmitter<{ method: HttpMethod; url: string; headers: HeaderRow[]; body?: string; params?: HeaderRow[] }>();\n\n  /** Emit when cURL is imported */\n  @Output() importCurl = new EventEmitter<void>();\n\n  apiForm: FormGroup;\n  selectedTab: RequestTab = 'headers';\n  selectedProgressStep: ProgressStep = 'request-details';\n  formSubmitted: boolean = false;\n  // progressSteps: ProgressStep[] = ['request-details', 'store-response', 'validation'];\n  \n  // Cache config objects to prevent infinite change detection loops\n  private verificationTypeConfigCache: DynamicSelectFieldConfig | null = null;\n  private expectedTypeConfigCache: DynamicSelectFieldConfig | null = null;\n  private httpMethodConfigCache: DynamicSelectFieldConfig | null = null;\n  private headerNameConfigCache: DynamicSelectFieldConfig | null = null;\n  private paramNameConfigCache: DynamicSelectFieldConfig | null = null;\n  private hasLoadedInitialData: boolean = false;\n\n  constructor(private fb: FormBuilder) {\n    // Get first HTTP method option (default to 'GET' if options not available yet)\n    const firstMethod = this.httpMethodOptions.length > 0 \n      ? this.httpMethodOptions[0].value \n      : 'GET';\n    \n    this.apiForm = this.fb.group({\n      method: [firstMethod, Validators.required],\n      url: ['', [Validators.required, this.urlValidator()]],\n      headers: this.fb.array([]),\n      body: [''],\n      params: this.fb.array([]),\n      scripts: [''],\n      variableName: [''],\n      validation: this.fb.array([])\n    });\n  }\n\n  private urlValidator() {\n    return (control: FormControl): { [key: string]: any } | null => {\n      const url = control.value;\n      if (!url) {\n        return null; // Let required validator handle empty case\n      }\n\n      // Pattern for plain URLs only (no parameters)\n      const urlPattern = /^(https?:\\/\\/)?([\\w-]+(\\.[\\w-]+)+)([\\w.,@?^=%&:/~+#-]*[\\w@?^=%&/~+#-])?$/;\n      \n      // Pattern to detect any parameter patterns: *|param|, $|param|, @|param|, or ${param}\n      const parameterPattern = /(\\*\\|[\\w-]+\\|(\\*)?|\\$\\|[\\w-]+\\|(\\*)?|@\\|[\\w-]+\\|(\\*)?|\\$\\{[^}]+\\})/;\n      \n      // Pattern for parameter-only values: *|param|, $|param|, @|param|, or ${param}\n      const envPattern = /^\\*\\|[\\w-]+\\|(\\*)?$|^\\$\\|[\\w-]+\\|(\\*)?$|^@\\|[\\w-]+\\|(\\*)?$|^\\$\\{[^}]+\\}$/;\n      \n      // Pattern to allow URLs with embedded parameters: *|param|, $|param|, @|param|, or ${param}\n      const flexibleUrlPattern = /^[@]?(https?:\\/\\/)?([\\w-]+(\\.[\\w-]+)+)([\\w.,@?^=%&:/~+#-]*(\\*\\|[\\w-]+\\|(\\*)?|\\$\\|[\\w-]+\\|(\\*)?|@\\|[\\w-]+\\|(\\*)?|\\$\\{[^}]+\\})[\\w.,@?^=%&:/~+#-]*)*$/;\n      \n      // Pattern that allows parameters anywhere after http:// or https://\n      const protocolWithParamsPattern = /^https?:\\/\\/(\\*\\|[\\w-]+\\|(\\*)?|\\$\\|[\\w-]+\\|(\\*)?|@\\|[\\w-]+\\|(\\*)?|\\$\\{[^}]+\\})[^\\s]*$|^https?:\\/\\/[^\\s]*(\\*\\|[\\w-]+\\|(\\*)?|\\$\\|[\\w-]+\\|(\\*)?|@\\|[\\w-]+\\|(\\*)?|\\$\\{[^}]+\\})[^\\s]*$/;\n      \n      // Check if URL contains parameters\n      if (parameterPattern.test(url)) {\n        // If it contains parameters, check if it matches allowed patterns\n        const envCombined = new RegExp(`${urlPattern.source}|${flexibleUrlPattern.source}|${envPattern.source}|${protocolWithParamsPattern.source}`);\n        if (!envCombined.test(url)) {\n          return { hasParameter: true };\n        }\n      } else {\n        // If no parameters, check if it's a valid plain URL\n        if (!urlPattern.test(url)) {\n          return { pattern: true };\n        }\n      }\n      \n      return null;\n    };\n  }\n\n  ngOnInit(): void {\n    // Sync currentStep input with selectedProgressStep\n    if (this.currentStep) {\n      this.selectedProgressStep = this.currentStep;\n    }\n    \n    // Set method to first option if not already set\n    if (!this.apiForm.get('method')?.value && this.httpMethodOptions.length > 0) {\n      this.apiForm.get('method')?.setValue(this.httpMethodOptions[0].value);\n    }\n    \n    // Load initial data if provided (edit mode)\n    if (this.initialData) {\n      this.loadInitialData(this.initialData);\n      this.hasLoadedInitialData = true;\n    } else {\n      // Add initial header row for new step\n      this.addHeader();\n    }\n    \n    // Initialize validation form array if starting on validation step\n    if (this.selectedProgressStep === 'validation' && this.validationFormArray.length === 0) {\n      this.addValidationRule();\n    }\n  }\n\n  private loadInitialData(data: ApiFormData): void {\n    console.log('loadInitialData: Loading data', data);\n    console.log('loadInitialData: Headers count', data.headers?.length || 0);\n    console.log('loadInitialData: Headers data', data.headers);\n    \n    // Set basic fields\n    this.apiForm.patchValue({\n      method: data.method || 'GET',\n      url: data.url || '',\n      body: data.body || '',\n      scripts: data.scripts || '',\n      variableName: data.variableName || ''\n    });\n\n    // Load headers\n    // Clear existing headers first\n    while (this.headersFormArray.length !== 0) {\n      this.headersFormArray.removeAt(0);\n    }\n    \n    if (data.headers && data.headers.length > 0) {\n      // Add headers from initial data (include all headers, even if empty)\n      data.headers.forEach(header => {\n        const headerGroup = this.fb.group({\n          name: [header.name || '', Validators.required],\n          value: [header.value || '', Validators.required]\n        });\n        this.headersFormArray.push(headerGroup);\n      });\n      console.log('loadInitialData: Loaded headers into form array, count:', this.headersFormArray.length);\n    }\n    \n    // Ensure at least one header row exists\n    if (this.headersFormArray.length === 0) {\n      console.log('loadInitialData: No headers found, adding empty header row');\n      this.addHeader();\n    }\n\n    // Load params\n    // Clear existing params first\n    while (this.paramsFormArray.length !== 0) {\n      this.paramsFormArray.removeAt(0);\n    }\n    \n    if (data.params && data.params.length > 0) {\n      // Add params from initial data (include all params, even if empty)\n      data.params.forEach(param => {\n        const paramGroup = this.fb.group({\n          name: [param.name || '', Validators.required],\n          value: [param.value || '', Validators.required]\n        });\n        this.paramsFormArray.push(paramGroup);\n      });\n    }\n\n    // Load validation rules\n    if (data.validation && data.validation.length > 0) {\n      // Clear existing validation rules\n      while (this.validationFormArray.length !== 0) {\n        this.validationFormArray.removeAt(0);\n      }\n      // Add validation rules from initial data\n      data.validation.forEach(rule => {\n        const validationGroup = this.fb.group({\n          jsonPath: [rule.jsonPath || '', Validators.required],\n          verificationType: [rule.verificationType || 'equals', Validators.required],\n          expectedType: [rule.expectedType || 'string', Validators.required],\n          expectedValue: [rule.expectedValue !== undefined ? rule.expectedValue : '', Validators.required],\n          result: [rule.result || 'Not run'],\n          checked: [rule.checked || false]\n        });\n        // Ensure all controls are enabled\n        validationGroup.enable({ emitEvent: false });\n        this.validationFormArray.push(validationGroup);\n      });\n    }\n  }\n\n  get headersFormArray(): FormArray {\n    return this.apiForm.get('headers') as FormArray;\n  }\n\n  get paramsFormArray(): FormArray {\n    return this.apiForm.get('params') as FormArray;\n  }\n\n  get validationFormArray(): FormArray {\n    return this.apiForm.get('validation') as FormArray;\n  }\n\n  addHeader(): void {\n    // Get first header name option\n    const headerOptions = this.headerNameOptions.length > 0 \n      ? this.headerNameOptions \n      : [\n          { id: 'string', value: 'string', name: 'string', label: 'string' },\n          { id: 'Content-Type', value: 'Content-Type', name: 'Content-Type', label: 'Content-Type' },\n          { id: 'Authorization', value: 'Authorization', name: 'Authorization', label: 'Authorization' },\n          { id: 'Accept', value: 'Accept', name: 'Accept', label: 'Accept' }\n        ];\n    \n    const firstHeaderName = headerOptions.length > 0 ? headerOptions[0].value : '';\n    \n    const headerGroup = this.fb.group({\n      name: [firstHeaderName, Validators.required],\n      value: ['', Validators.required]\n    });\n    this.headersFormArray.push(headerGroup);\n  }\n\n  removeHeader(index: number): void {\n    this.headersFormArray.removeAt(index);\n  }\n\n  addParam(): void {\n    const paramGroup = this.fb.group({\n      name: ['', Validators.required],\n      value: ['', Validators.required]\n    });\n    this.paramsFormArray.push(paramGroup);\n  }\n\n  removeParam(index: number): void {\n    this.paramsFormArray.removeAt(index);\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // Reset config caches when inputs change\n    if (changes['httpMethodOptions']) {\n      this.httpMethodConfigCache = null;\n      // Set method to first option if not already set or if we're not in edit mode\n      if (!this.hasLoadedInitialData && this.httpMethodOptions.length > 0) {\n        const methodControl = this.apiForm.get('method');\n        if (methodControl && !methodControl.value) {\n          methodControl.setValue(this.httpMethodOptions[0].value);\n        }\n      }\n    }\n    if (changes['headerNameOptions']) {\n      this.headerNameConfigCache = null;\n      // Update header names to first option for empty headers (only if not in edit mode)\n      if (!this.hasLoadedInitialData && this.headerNameOptions.length > 0) {\n        this.headersFormArray.controls.forEach((headerGroup: any) => {\n          const nameControl = headerGroup.get('name');\n          if (nameControl && !nameControl.value) {\n            nameControl.setValue(this.headerNameOptions[0].value);\n          }\n        });\n      }\n    }\n    \n    // Load initial data if it's set after component initialization\n    if (changes['initialData'] && !changes['initialData'].firstChange && this.initialData && !this.hasLoadedInitialData) {\n      this.loadInitialData(this.initialData);\n      this.hasLoadedInitialData = true;\n    }\n  }\n\n  getHttpMethodConfig(): DynamicSelectFieldConfig {\n    if (!this.httpMethodConfigCache) {\n      this.httpMethodConfigCache = {\n        key: 'method',\n        placeholder: 'Select method',\n        multiple: false,\n        searchable: false,\n        options: this.httpMethodOptions\n      };\n    }\n    return this.httpMethodConfigCache;\n  }\n\n  getHeaderNameConfig(index: number): DynamicSelectFieldConfig {\n    if (!this.headerNameConfigCache) {\n      const defaultOptions = [\n        { id: 'string', value: 'string', name: 'string', label: 'string' },\n        { id: 'Content-Type', value: 'Content-Type', name: 'Content-Type', label: 'Content-Type' },\n        { id: 'Authorization', value: 'Authorization', name: 'Authorization', label: 'Authorization' },\n        { id: 'Accept', value: 'Accept', name: 'Accept', label: 'Accept' }\n      ];\n      \n      this.headerNameConfigCache = {\n        key: 'name',\n        placeholder: 'Select header',\n        multiple: false,\n        searchable: true,\n        options: this.headerNameOptions.length > 0 ? this.headerNameOptions : defaultOptions\n      };\n    }\n    return this.headerNameConfigCache;\n  }\n\n  getParamNameConfig(index: number): DynamicSelectFieldConfig {\n    if (!this.paramNameConfigCache) {\n      this.paramNameConfigCache = {\n        key: 'name',\n        placeholder: 'Parameter name',\n        multiple: false,\n        searchable: false,\n        options: []\n      };\n    }\n    return this.paramNameConfigCache;\n  }\n\n  getHeaderFormGroup(index: number): FormGroup {\n    return this.headersFormArray.at(index) as FormGroup;\n  }\n\n  getParamFormGroup(index: number): FormGroup {\n    return this.paramsFormArray.at(index) as FormGroup;\n  }\n\n  onTabChange(tab: RequestTab): void {\n    this.selectedTab = tab;\n  }\n\n  onSendRequest(): void {\n    this.formSubmitted = true;\n    this.markFormGroupTouched(this.apiForm);\n    console.log('onSendRequest: apiForm', this.apiForm.value);\n    console.log('onSendRequest: apiForm valid', this.apiForm);\n    if (this.apiForm.valid) {\n      const formValue = this.apiForm.value;\n      const headers = formValue.headers.map((h: any) => ({\n        name: h.name,\n        value: h.value\n      }));\n      const params = formValue.params ? formValue.params.map((p: any) => ({\n        name: p.name,\n        value: p.value\n      })) : [];\n\n      this.sendRequest.emit({\n        method: formValue.method,\n        url: formValue.url,\n        headers,\n        body: formValue.body,\n        params\n      });\n    }\n  }\n\n  onImportCurl(): void {\n    this.importCurl.emit();\n  }\n\n  onCancel(): void {\n    this.cancelled.emit();\n  }\n\n  onCreateStep(): void {\n    this.formSubmitted = true;\n    this.markFormGroupTouched(this.apiForm);\n    \n    if (this.apiForm.valid) {\n      const formValue = this.apiForm.value;\n      const stepData: ApiFormData = {\n        method: formValue.method,\n        url: formValue.url,\n        headers: formValue.headers.map((h: any) => ({\n          name: h.name,\n          value: h.value\n        })),\n        body: formValue.body,\n        params: formValue.params ? formValue.params.map((p: any) => ({\n          name: p.name,\n          value: p.value\n        })) : [],\n        scripts: formValue.scripts,\n        variableName: formValue.variableName || '',\n        validation: formValue.validation ? formValue.validation.map((v: any) => ({\n          jsonPath: v.jsonPath,\n          verificationType: v.verificationType,\n          expectedType: v.expectedType,\n          expectedValue: v.expectedValue,\n          result: v.result || 'Not run'\n        })) : []\n      };\n      console.log('9999999', stepData);\n      this.createStep.emit(stepData);\n    }\n  }\n\n  onNext(): void {\n    // Move to next step or create step if on last step\n    const stepOrder: ProgressStep[] = ['request-details', 'store-response', 'validation'];\n    const currentIndex = stepOrder.indexOf(this.selectedProgressStep);\n    if (currentIndex < stepOrder.length - 1) {\n      this.onProgressStepChange(stepOrder[currentIndex + 1]);\n    } else {\n      this.onCreateStep();\n    }\n  }\n\n  onBack(): void {\n    // Move to previous step\n    const stepOrder: ProgressStep[] = ['request-details', 'store-response', 'validation'];\n    const currentIndex = stepOrder.indexOf(this.selectedProgressStep);\n    if (currentIndex > 0) {\n      this.onProgressStepChange(stepOrder[currentIndex - 1]);\n    }\n  }\n\n  canGoBack(): boolean {\n    const stepOrder: ProgressStep[] = ['request-details', 'store-response', 'validation'];\n    const currentIndex = stepOrder.indexOf(this.selectedProgressStep);\n    return currentIndex > 0;\n  }\n\n  isLastStep(): boolean {\n    const stepOrder: ProgressStep[] = ['request-details', 'store-response', 'validation'];\n    const currentIndex = stepOrder.indexOf(this.selectedProgressStep);\n    return currentIndex === stepOrder.length - 1;\n  }\n\n  getProgressStepClass(step: ProgressStep): string {\n    const stepOrder: ProgressStep[] = ['request-details', 'store-response', 'validation'];\n    const currentIndex = stepOrder.indexOf(this.currentStep);\n    const stepIndex = stepOrder.indexOf(step);\n\n    if (stepIndex < currentIndex) {\n      return 'cqa-text-gray-400'; // Completed\n    } else if (stepIndex === currentIndex) {\n      return 'cqa-text-purple-600 cqa-font-semibold'; // Active\n    } else {\n      return 'cqa-text-gray-400'; // Pending\n    }\n  }\n\n  getProgressStepNumber(step: ProgressStep): number {\n    const stepOrder: ProgressStep[] = ['request-details', 'store-response', 'validation'];\n    return stepOrder.indexOf(step) + 1;\n  }\n\n  getProgressStepLabel(step: ProgressStep): string {\n    const labels: Record<ProgressStep, string> = {\n      'request-details': 'Request Details',\n      'store-response': 'Store Response',\n      'validation': 'Validation'\n    };\n    return labels[step];\n  }\n\n  formatJsonResponse(data: any): string {\n    if (!data) return '';\n    try {\n      return JSON.stringify(data, null, 2);\n    } catch {\n      return String(data);\n    }\n  }\n  \n  onProgressStepChange(step: ProgressStep): void {\n    this.selectedProgressStep = step;\n    // Initialize validation form array if switching to validation step\n    if (step === 'validation' && this.validationFormArray.length === 0) {\n      this.addValidationRule();\n    }\n  }\n\n  addValidationRule(): void {\n    const validationGroup = this.fb.group({\n      checked: [false],\n      jsonPath: ['', Validators.required],\n      verificationType: ['equals'],\n      expectedType: ['string'],\n      expectedValue: ['', Validators.required],\n      result: ['Not run']\n    });\n    // Ensure all controls are enabled to avoid disabled attribute warnings\n    validationGroup.enable({ emitEvent: false });\n    this.validationFormArray.push(validationGroup);\n  }\n\n  removeValidationRule(index: number): void {\n    this.validationFormArray.removeAt(index);\n  }\n\n  getValidationFormGroup(index: number): FormGroup {\n    return this.validationFormArray.at(index) as FormGroup;\n  }\n\n  trackByValidationRuleIndex(index: number): number {\n    return index;\n  }\n\n  trackByHeaderIndex(index: number): number {\n    return index;\n  }\n\n  trackByParamIndex(index: number): number {\n    return index;\n  }\n\n  getVerificationTypeOptions(): SelectOption[] {\n    return [\n      { id: 'equals', value: 'equals', name: 'Equals', label: 'Equals' },\n      { id: 'not_equals', value: 'not_equals', name: 'Not Equals', label: 'Not Equals' },\n      { id: 'contains', value: 'contains', name: 'Contains', label: 'Contains' },\n      { id: 'not_contains', value: 'not_contains', name: 'Not Contains', label: 'Not Contains' },\n      { id: 'greater_than', value: 'greater_than', name: 'Greater Than', label: 'Greater Than' },\n      { id: 'less_than', value: 'less_than', name: 'Less Than', label: 'Less Than' },\n      { id: 'greater_than_or_equals', value: 'greater_than_or_equals', name: 'Greater Than Or Equals', label: 'Greater Than Or Equals' },\n      { id: 'less_than_or_equals', value: 'less_than_or_equals', name: 'Less Than Or Equals', label: 'Less Than Or Equals' },\n      { id: 'exists', value: 'exists', name: 'Exists', label: 'Exists' },\n      { id: 'not_exists', value: 'not_exists', name: 'Not Exists', label: 'Not Exists' }\n    ];\n  }\n\n  getExpectedTypeOptions(): SelectOption[] {\n    return [\n      { id: 'string', value: 'string', name: 'String', label: 'String' },\n      { id: 'number', value: 'number', name: 'Number', label: 'Number' },\n      { id: 'boolean', value: 'boolean', name: 'Boolean', label: 'Boolean' },\n      { id: 'object', value: 'object', name: 'Object', label: 'Object' },\n      { id: 'array', value: 'array', name: 'Array', label: 'Array' },\n      { id: 'null', value: 'null', name: 'Null', label: 'Null' }\n    ];\n  }\n\n  getVerificationTypeConfig(index: number): DynamicSelectFieldConfig {\n    // Cache the config to prevent infinite change detection loops\n    if (!this.verificationTypeConfigCache) {\n      this.verificationTypeConfigCache = {\n        key: 'verificationType',\n        placeholder: 'Select verification type',\n        multiple: false,\n        searchable: false,\n        options: this.getVerificationTypeOptions()\n      };\n    }\n    return this.verificationTypeConfigCache;\n  }\n\n  getExpectedTypeConfig(index: number): DynamicSelectFieldConfig {\n    // Cache the config to prevent infinite change detection loops\n    if (!this.expectedTypeConfigCache) {\n      this.expectedTypeConfigCache = {\n        key: 'expectedType',\n        placeholder: 'Select expected type',\n        multiple: false,\n        searchable: false,\n        options: this.getExpectedTypeOptions()\n      };\n    }\n    return this.expectedTypeConfigCache;\n  }\n\n  getResultClass(result: string): string {\n    if (result === 'Pass') {\n      return 'cqa-bg-green-500 cqa-text-white';\n    } else if (result === 'Fail') {\n      return 'cqa-bg-red-500 cqa-text-white';\n    }\n    return 'cqa-bg-gray-300 cqa-text-gray-700';\n  }\n\n  get allValidationRulesSelected(): boolean {\n    return this.validationFormArray.length > 0 && \n           this.validationFormArray.controls.every(control => control.get('checked')?.value);\n  }\n\n  get someValidationRulesSelected(): boolean {\n    return this.validationFormArray.controls.some(control => control.get('checked')?.value) && \n           !this.allValidationRulesSelected;\n  }\n\n  onSelectAllValidationRules(checked: boolean): void {\n    this.validationFormArray.controls.forEach(control => {\n      control.get('checked')?.setValue(checked);\n    });\n  }\n\n  onDeleteSelectedValidationRules(): void {\n    const indicesToRemove: number[] = [];\n    this.validationFormArray.controls.forEach((control, index) => {\n      if (control.get('checked')?.value) {\n        indicesToRemove.push(index);\n      }\n    });\n    // Remove in reverse order to maintain correct indices\n    indicesToRemove.reverse().forEach(index => {\n      this.removeValidationRule(index);\n    });\n  }\n\n  private markFormGroupTouched(formGroup: FormGroup): void {\n    Object.keys(formGroup.controls).forEach(key => {\n      const control = formGroup.get(key);\n      if (control) {\n        control.markAsTouched();\n        if (control instanceof FormGroup) {\n          this.markFormGroupTouched(control);\n        } else if (control instanceof FormArray) {\n          control.controls.forEach((ctrl: any) => {\n            if (ctrl instanceof FormGroup) {\n              this.markFormGroupTouched(ctrl);\n            } else {\n              ctrl.markAsTouched();\n            }\n          });\n        }\n      }\n    });\n  }\n\n  onUrlChange(value: string): void {\n    const urlControl = this.apiForm.get('url');\n    if (urlControl) {\n      urlControl.setValue(value);\n      urlControl.updateValueAndValidity();\n    }\n  }\n\n  onUrlBlur(): void {\n    const urlControl = this.apiForm.get('url');\n    if (urlControl) {\n      urlControl.markAsTouched();\n    }\n  }\n\n  getUrlErrors(): string[] {\n    const urlControl = this.apiForm.get('url');\n    if (!urlControl || (!this.formSubmitted && !urlControl.touched)) {\n      return [];\n    }\n\n    const errors: string[] = [];\n    \n    if (urlControl.errors?.['required']) {\n      errors.push('URL is required');\n    } else if (urlControl.errors?.['pattern']) {\n      errors.push('URL is not valid');\n    } else if (urlControl.errors?.['hasParameter']) {\n      errors.push('URL format is invalid. Please check the parameter format.');\n    }\n    \n    return errors;\n  }\n}\n\n","<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n  <!-- Header -->\n  <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n    Create API Test Step\n  </h2>\n\n  <!-- Progress Tracker -->\n  <div class=\"cqa-flex cqa-items-center cqa-gap-4 cqa-mb-6 cqa-pb-4 cqa-border-b cqa-border-gray-200\">\n    <!-- <div *ngFor=\"let step of progressSteps\"\n         class=\"cqa-flex cqa-items-center cqa-gap-2\"\n         [ngClass]=\"getProgressStepClass(step)\">\n      <span class=\"cqa-text-sm cqa-font-medium\">{{ getProgressStepNumber(step) }} {{ getProgressStepLabel(step) }}</span>\n      <span *ngIf=\"step !== 'validation'\" class=\"cqa-text-gray-300\">•</span>\n    </div> -->\n    <div class=\"cqa-flex cqa-items-center cqa-w-full cqa-justify-between\">\n         <div class=\"cqa-flex cqa-items-center cqa-gap-2\" [ngClass]=\"{'cqa-text-purple-600 cqa-font-semibold': selectedProgressStep === 'request-details'}\" (click)=\"onProgressStepChange('request-details')\">\n          <div class=\"cqa-w-8 cqa-h-8 cqa-flex cqa-items-center cqa-justify-center cqa-rounded-full cqa-bg-purple-600 cqa-text-white cqa-text-xs cqa-font-semibold\">1</div>\n          <div>Request Details</div>\n          <div></div>\n         </div>\n         <div class=\"cqa-flex cqa-items-center cqa-gap-2\" [ngClass]=\"{'cqa-text-purple-600 cqa-font-semibold': selectedProgressStep === 'store-response'}\" (click)=\"onProgressStepChange('store-response')\">\n          <div class=\"cqa-w-8 cqa-h-8 cqa-flex cqa-items-center cqa-justify-center cqa-rounded-full cqa-bg-purple-600 cqa-text-white cqa-text-xs cqa-font-semibold\">2</div>\n          <div>Store Response</div>\n          <div></div>\n         </div>\n         <div class=\"cqa-flex cqa-items-center cqa-gap-2\" [ngClass]=\"{'cqa-text-purple-600 cqa-font-semibold': selectedProgressStep === 'validation'}\" (click)=\"onProgressStepChange('validation')\">\n          <div class=\"cqa-w-8 cqa-h-8 cqa-flex cqa-items-center cqa-justify-center cqa-rounded-full cqa-bg-purple-600 cqa-text-white cqa-text-xs cqa-font-semibold\">3</div>\n          <div>Validation</div>\n          <div></div>\n         </div>\n    </div>\n  </div>\n  <ng-container *ngIf=\"selectedProgressStep === 'request-details'\">\n  <!-- API Request Configuration -->\n  <div class=\"cqa-flex cqa-items-start cqa-gap-2 cqa-mb-3\">\n    <div class=\"cqa-w-32 cqa-flex-shrink-0\">\n      <cqa-dynamic-select\n        [form]=\"apiForm\"\n        [config]=\"getHttpMethodConfig()\">\n      </cqa-dynamic-select>\n    </div>\n    <div class=\"cqa-flex-1\">\n      <cqa-custom-input\n        [placeholder]=\"'Enter API endpoint URL'\"\n        [value]=\"apiForm.get('url')?.value || ''\"\n        [fullWidth]=\"true\"\n        [errors]=\"getUrlErrors()\"\n        (valueChange)=\"onUrlChange($event)\"\n        (blurred)=\"onUrlBlur()\">\n      </cqa-custom-input>\n    </div>\n    <cqa-button\n      variant=\"outlined\"\n      text=\"Import API cURL\"\n      (clicked)=\"onImportCurl()\">\n    </cqa-button>\n    <cqa-button\n      variant=\"filled\"\n      text=\"Send Request\"\n      (clicked)=\"onSendRequest()\"\n      [disabled]=\"!apiForm.get('url')?.value || isLoading\">\n    </cqa-button>\n  </div>\n\n  <!-- Request Details Tabs -->\n  <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n    <!-- Tab Navigation -->\n    <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n      <button\n        type=\"button\"\n        class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n        [class.cqa-text-blue-600]=\"selectedTab === 'headers'\"\n        [class.cqa-border-blue-600]=\"selectedTab === 'headers'\"\n        [class.cqa-text-gray-600]=\"selectedTab !== 'headers'\"\n        [class.cqa-border-transparent]=\"selectedTab !== 'headers'\"\n        (click)=\"onTabChange('headers')\">\n        Headers\n      </button>\n      <button\n        type=\"button\"\n        class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n        [class.cqa-text-blue-600]=\"selectedTab === 'body'\"\n        [class.cqa-border-blue-600]=\"selectedTab === 'body'\"\n        [class.cqa-text-gray-600]=\"selectedTab !== 'body'\"\n        [class.cqa-border-transparent]=\"selectedTab !== 'body'\"\n        (click)=\"onTabChange('body')\">\n        Body\n      </button>\n      <button\n        type=\"button\"\n        class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n        [class.cqa-text-blue-600]=\"selectedTab === 'params'\"\n        [class.cqa-border-blue-600]=\"selectedTab === 'params'\"\n        [class.cqa-text-gray-600]=\"selectedTab !== 'params'\"\n        [class.cqa-border-transparent]=\"selectedTab !== 'params'\"\n        (click)=\"onTabChange('params')\">\n        Params\n      </button>\n      <button\n        type=\"button\"\n        class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n        [class.cqa-text-blue-600]=\"selectedTab === 'scripts'\"\n        [class.cqa-border-blue-600]=\"selectedTab === 'scripts'\"\n        [class.cqa-text-gray-600]=\"selectedTab !== 'scripts'\"\n        [class.cqa-border-transparent]=\"selectedTab !== 'scripts'\"\n        (click)=\"onTabChange('scripts')\">\n        Scripts\n      </button>\n    </div>\n\n    <!-- Tab Content -->\n    <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-4 cqa-border cqa-border-gray-200 cqa-border-t-0 cqa-rounded-b-lg\">\n      <!-- Headers Tab -->\n      <div *ngIf=\"selectedTab === 'headers'\" class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n        <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4 cqa-items-start\">\n          <div class=\"cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Header Name*</div>\n          <div class=\"cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Header Value*</div>\n        </div>\n        <div *ngFor=\"let header of headersFormArray.controls; let i = index; trackBy: trackByHeaderIndex\" \n             class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4 cqa-items-center\">\n          <div>\n            <cqa-dynamic-select\n              [form]=\"getHeaderFormGroup(i)\"\n              [config]=\"getHeaderNameConfig(i)\">\n            </cqa-dynamic-select>\n          </div>\n          <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n            <div class=\"cqa-flex-1\">\n              <cqa-custom-input\n                [placeholder]=\"'Header value'\"\n                [value]=\"getHeaderFormGroup(i).get('value')?.value || ''\"\n                [fullWidth]=\"true\"\n                (valueChange)=\"getHeaderFormGroup(i).get('value')?.setValue($event)\">\n              </cqa-custom-input>\n            </div>\n            <button\n              type=\"button\"\n              class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-8 cqa-h-8 cqa-rounded cqa-text-red-500 hover:cqa-text-red-700 hover:cqa-bg-red-50 cqa-transition-colors\"\n              (click)=\"removeHeader(i)\"\n              [attr.aria-label]=\"'Remove header'\">\n              <mat-icon class=\"cqa-text-lg\">delete</mat-icon>\n            </button>\n          </div>\n        </div>\n        <button\n          type=\"button\"\n          class=\"cqa-text-blue-600 cqa-text-sm cqa-font-medium cqa-self-start cqa-flex cqa-items-center cqa-gap-1 hover:cqa-text-blue-700 cqa-transition-colors\"\n          (click)=\"addHeader()\">\n          <mat-icon class=\"cqa-text-base\">add</mat-icon>\n          <span>Add Header</span>\n        </button>\n      </div>\n\n      <!-- Body Tab -->\n      <div *ngIf=\"selectedTab === 'body'\" class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n        <cqa-custom-textarea\n          [placeholder]=\"'Enter request body (JSON, XML, etc.)'\"\n          [value]=\"apiForm.get('body')?.value || ''\"\n          [fullWidth]=\"true\"\n          [rows]=\"10\"\n          (valueChange)=\"apiForm.get('body')?.setValue($event)\">\n        </cqa-custom-textarea>\n      </div>\n\n      <!-- Params Tab -->\n      <div *ngIf=\"selectedTab === 'params'\" class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n        <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4 cqa-items-start\">\n          <div class=\"cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Parameter Name</div>\n          <div class=\"cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Parameter Value</div>\n        </div>\n        <div *ngFor=\"let param of paramsFormArray.controls; let i = index; trackBy: trackByParamIndex\" \n             class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4 cqa-items-center\">\n          <div>\n            <cqa-custom-input\n              [placeholder]=\"'Parameter name'\"\n              [value]=\"getParamFormGroup(i).get('name')?.value || ''\"\n              [fullWidth]=\"true\"\n              (valueChange)=\"getParamFormGroup(i).get('name')?.setValue($event)\">\n            </cqa-custom-input>\n          </div>\n          <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n            <div class=\"cqa-flex-1\">\n              <cqa-custom-input\n                [placeholder]=\"'Parameter value'\"\n                [value]=\"getParamFormGroup(i).get('value')?.value || ''\"\n                [fullWidth]=\"true\"\n                (valueChange)=\"getParamFormGroup(i).get('value')?.setValue($event)\">\n              </cqa-custom-input>\n            </div>\n            <button\n              type=\"button\"\n              class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-8 cqa-h-8 cqa-rounded cqa-text-red-500 hover:cqa-text-red-700 hover:cqa-bg-red-50 cqa-transition-colors\"\n              (click)=\"removeParam(i)\"\n              [attr.aria-label]=\"'Remove parameter'\">\n              <mat-icon class=\"cqa-text-lg\">delete</mat-icon>\n            </button>\n          </div>\n        </div>\n        <button\n          type=\"button\"\n          class=\"cqa-text-blue-600 cqa-text-sm cqa-font-medium cqa-self-start cqa-flex cqa-items-center cqa-gap-1 hover:cqa-text-blue-700 cqa-transition-colors\"\n          (click)=\"addParam()\">\n          <mat-icon class=\"cqa-text-base\">add</mat-icon>\n          <span>Add Parameter</span>\n        </button>\n      </div>\n\n      <!-- Scripts Tab -->\n      <div *ngIf=\"selectedTab === 'scripts'\" class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n        <cqa-custom-textarea\n          [placeholder]=\"'Enter scripts (JavaScript, etc.)'\"\n          [value]=\"apiForm.get('scripts')?.value || ''\"\n          [fullWidth]=\"true\"\n          [rows]=\"10\"\n          (valueChange)=\"apiForm.get('scripts')?.setValue($event)\">\n        </cqa-custom-textarea>\n      </div>\n    </div>\n  </div>\n\n\n</ng-container>\n<ng-container *ngIf=\"selectedProgressStep === 'store-response'\">\n  <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-max-h-[500px] cqa-overflow-y-auto\">\n    <!-- Store Response Title -->\n    <h3 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n      Store Response\n    </h3>\n\n    <!-- Variable Name Input -->\n    <div class=\"cqa-mb-6\">\n      <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-gray-700 cqa-mb-1 cqa-block\">\n        Variable Name\n      </label>\n      <cqa-custom-input\n        [placeholder]=\"'Variable Name'\"\n        [value]=\"apiForm.get('variableName')?.value || ''\"\n        [fullWidth]=\"true\"\n        (valueChange)=\"apiForm.get('variableName')?.setValue($event)\">\n      </cqa-custom-input>\n    </div>\n  </div>\n</ng-container>\n<ng-container *ngIf=\"selectedProgressStep === 'validation'\">\n  <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-max-h-[500px] cqa-overflow-y-auto\">\n    <!-- Validation Rules Title -->\n    <h3 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n      Validation Rules\n    </h3>\n\n    <!-- Validation Rules Table -->\n    <div class=\"cqa-bg-white cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden cqa-mb-4\">\n      <!-- Table Header -->\n      <div class=\"cqa-grid cqa-grid-cols-6 cqa-gap-2 cqa-p-3 cqa-bg-gray-50 cqa-border-b cqa-border-gray-200 cqa-items-center\">\n        <div class=\"cqa-flex cqa-items-center\">\n          <mat-checkbox\n            [checked]=\"allValidationRulesSelected\"\n            [indeterminate]=\"someValidationRulesSelected\"\n            (change)=\"onSelectAllValidationRules($event.checked)\">\n          </mat-checkbox>\n        </div>\n        <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">jsonPath</div>\n        <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">verificationType</div>\n        <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">expectedType</div>\n        <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">expectedValue</div>\n        <div class=\"cqa-text-[10px] cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">Result</div>\n      </div>\n\n      <!-- Table Body -->\n      <div class=\"cqa-flex cqa-flex-col\">\n        <div *ngFor=\"let validationRule of validationFormArray.controls; let i = index; trackBy: trackByValidationRuleIndex\" \n             [formGroup]=\"getValidationFormGroup(i)\"\n             class=\"cqa-grid cqa-grid-cols-6 cqa-gap-2 cqa-p-3 cqa-border-b cqa-border-gray-200 cqa-items-center cqa-last:border-b-0\">\n          <!-- Checkbox -->\n          <div class=\"cqa-flex cqa-items-center\">\n            <mat-checkbox formControlName=\"checked\">\n            </mat-checkbox>\n          </div>\n\n          <!-- jsonPath -->\n          <div>\n            <cqa-custom-input\n              [placeholder]=\"'jsonPath'\"\n              [value]=\"getValidationFormGroup(i).get('jsonPath')?.value || ''\"\n              [fullWidth]=\"true\"\n              (valueChange)=\"getValidationFormGroup(i).get('jsonPath')?.setValue($event)\">\n            </cqa-custom-input>\n          </div>\n\n          <!-- verificationType -->\n          <div>\n            <cqa-dynamic-select\n              [form]=\"getValidationFormGroup(i)\"\n              [config]=\"getVerificationTypeConfig(i)\">\n            </cqa-dynamic-select>\n          </div>\n\n          <!-- expectedType -->\n          <div>\n            <cqa-dynamic-select\n              [form]=\"getValidationFormGroup(i)\"\n              [config]=\"getExpectedTypeConfig(i)\">\n            </cqa-dynamic-select>\n          </div>\n\n          <!-- expectedValue -->\n          <div>\n            <cqa-custom-input\n              [placeholder]=\"'expectedValue'\"\n              [value]=\"getValidationFormGroup(i).get('expectedValue')?.value || ''\"\n              [fullWidth]=\"true\"\n              (valueChange)=\"getValidationFormGroup(i).get('expectedValue')?.setValue($event)\">\n            </cqa-custom-input>\n          </div>\n\n          <!-- Result and Delete -->\n          <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n            <button\n              type=\"button\"\n              class=\"cqa-px-3 cqa-py-1 cqa-rounded cqa-text-[10px] cqa-font-medium\"\n              [ngClass]=\"getResultClass(getValidationFormGroup(i).get('result')?.value || 'Not run')\">\n              {{ getValidationFormGroup(i).get('result')?.value || 'Not run' }}\n            </button>\n            <button\n              type=\"button\"\n              class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-6 cqa-h-6 cqa-rounded cqa-text-red-500 hover:cqa-text-red-700 hover:cqa-bg-red-50 cqa-transition-colors\"\n              (click)=\"removeValidationRule(i)\"\n              [attr.aria-label]=\"'Delete validation rule'\">\n              <mat-icon class=\"cqa-text-base\">delete</mat-icon>\n            </button>\n          </div>\n        </div>\n      </div>\n    </div>\n\n    <!-- Add Another Rule and Delete Selected -->\n    <div class=\"cqa-flex cqa-justify-between cqa-items-center cqa-mb-4\">\n      <button\n        *ngIf=\"someValidationRulesSelected || allValidationRulesSelected\"\n        type=\"button\"\n        class=\"cqa-text-red-600 cqa-text-sm cqa-font-medium cqa-flex cqa-items-center cqa-gap-1 hover:cqa-text-red-700 cqa-transition-colors\"\n        (click)=\"onDeleteSelectedValidationRules()\">\n        <mat-icon class=\"cqa-text-base\">delete</mat-icon>\n        <span>Delete Selected</span>\n      </button>\n      <button\n        type=\"button\"\n        class=\"cqa-text-blue-600 cqa-text-sm cqa-font-medium cqa-flex cqa-items-center cqa-gap-1 hover:cqa-text-blue-700 cqa-transition-colors\"\n        (click)=\"addValidationRule()\">\n        <mat-icon class=\"cqa-text-base\">add</mat-icon>\n        <span>Add Another</span>\n      </button>\n    </div>\n  </div>\n</ng-container>\n\n\n  <!-- Response Preview Section -->\n  <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden cqa-mb-6\">\n    <div class=\"cqa-p-3 cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n      <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Response Preview</h3>\n    </div>\n    <div class=\"cqa-p-4 cqa-bg-gray-50 cqa-overflow-auto\" style=\"max-height: 300px;\">\n      <pre *ngIf=\"responsePreview\" class=\"cqa-text-xs cqa-font-mono cqa-text-gray-800 cqa-whitespace-pre-wrap\">{{ formatJsonResponse(responsePreview) }}</pre>\n      <p *ngIf=\"!responsePreview\" class=\"cqa-text-sm cqa-text-gray-400 cqa-text-center cqa-py-8\">\n        No response yet. Send a request to see the response preview.\n      </p>\n    </div>\n  </div>\n  <!-- Action Buttons -->\n  <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n    <!-- Cancel button (only on first step) -->\n    <cqa-button\n      *ngIf=\"selectedProgressStep === 'request-details'\"\n      class=\"cqa-w-1/2\"\n      variant=\"outlined\"\n      text=\"Cancel\"\n      [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCancel()\">\n    </cqa-button>\n    <!-- Back button (on all steps except first) -->\n    <cqa-button\n      *ngIf=\"selectedProgressStep !== 'request-details'\"\n      class=\"cqa-w-1/2\"\n      variant=\"outlined\"\n      text=\"Back\"\n      [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onBack()\">\n    </cqa-button>\n    <!-- Next/Create Step button -->\n    <cqa-button\n      class=\"cqa-w-1/2\"\n      variant=\"filled\"\n      [text]=\"isLastStep() ? 'Create Step' : 'Next'\"\n      [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"isLastStep() ? onCreateStep() : onNext()\">\n    </cqa-button>\n  </div>\n</div>\n\n"]}
|