@cqa-lib/cqa-ui 1.1.184 → 1.1.186
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/execution-screen/main-step-collapse/main-step-collapse.component.mjs +20 -3
- package/esm2020/lib/simulator/simulator.component.mjs +188 -346
- package/esm2020/lib/step-builder/step-builder-api/step-builder-api.component.mjs +510 -0
- package/esm2020/lib/step-builder/step-builder-custom-code/step-builder-custom-code.component.mjs +92 -7
- package/esm2020/lib/test-case-details/create-step-group/create-step-group.component.mjs +114 -0
- package/esm2020/lib/test-case-details/delete-steps/delete-steps.component.mjs +104 -0
- package/esm2020/lib/ui-kit.module.mjs +17 -2
- package/esm2020/public-api.mjs +4 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +1065 -398
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +1014 -356
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/execution-screen/main-step-collapse/main-step-collapse.component.d.ts +6 -3
- package/lib/simulator/simulator.component.d.ts +10 -22
- package/lib/step-builder/step-builder-api/step-builder-api.component.d.ts +115 -0
- package/lib/step-builder/step-builder-custom-code/step-builder-custom-code.component.d.ts +17 -3
- package/lib/test-case-details/create-step-group/create-step-group.component.d.ts +30 -0
- package/lib/test-case-details/delete-steps/delete-steps.component.d.ts +26 -0
- package/lib/ui-kit.module.d.ts +35 -32
- package/package.json +1 -1
- package/public-api.d.ts +3 -0
- package/styles.css +1 -1
|
@@ -0,0 +1,510 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
2
|
+
import { 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
|
+
// progressSteps: ProgressStep[] = ['request-details', 'store-response', 'validation'];
|
|
44
|
+
// Cache config objects to prevent infinite change detection loops
|
|
45
|
+
this.verificationTypeConfigCache = null;
|
|
46
|
+
this.expectedTypeConfigCache = null;
|
|
47
|
+
this.httpMethodConfigCache = null;
|
|
48
|
+
this.headerNameConfigCache = null;
|
|
49
|
+
this.paramNameConfigCache = null;
|
|
50
|
+
this.hasLoadedInitialData = false;
|
|
51
|
+
this.apiForm = this.fb.group({
|
|
52
|
+
method: ['GET', Validators.required],
|
|
53
|
+
url: ['', Validators.required],
|
|
54
|
+
headers: this.fb.array([]),
|
|
55
|
+
body: [''],
|
|
56
|
+
params: this.fb.array([]),
|
|
57
|
+
scripts: [''],
|
|
58
|
+
variableName: [''],
|
|
59
|
+
validation: this.fb.array([])
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
ngOnInit() {
|
|
63
|
+
// Sync currentStep input with selectedProgressStep
|
|
64
|
+
if (this.currentStep) {
|
|
65
|
+
this.selectedProgressStep = this.currentStep;
|
|
66
|
+
}
|
|
67
|
+
// Load initial data if provided (edit mode)
|
|
68
|
+
if (this.initialData) {
|
|
69
|
+
this.loadInitialData(this.initialData);
|
|
70
|
+
this.hasLoadedInitialData = true;
|
|
71
|
+
}
|
|
72
|
+
else {
|
|
73
|
+
// Add initial header row for new step
|
|
74
|
+
this.addHeader();
|
|
75
|
+
}
|
|
76
|
+
// Initialize validation form array if starting on validation step
|
|
77
|
+
if (this.selectedProgressStep === 'validation' && this.validationFormArray.length === 0) {
|
|
78
|
+
this.addValidationRule();
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
loadInitialData(data) {
|
|
82
|
+
console.log('loadInitialData: Loading data', data);
|
|
83
|
+
console.log('loadInitialData: Headers count', data.headers?.length || 0);
|
|
84
|
+
console.log('loadInitialData: Headers data', data.headers);
|
|
85
|
+
// Set basic fields
|
|
86
|
+
this.apiForm.patchValue({
|
|
87
|
+
method: data.method || 'GET',
|
|
88
|
+
url: data.url || '',
|
|
89
|
+
body: data.body || '',
|
|
90
|
+
scripts: data.scripts || '',
|
|
91
|
+
variableName: data.variableName || ''
|
|
92
|
+
});
|
|
93
|
+
// Load headers
|
|
94
|
+
// Clear existing headers first
|
|
95
|
+
while (this.headersFormArray.length !== 0) {
|
|
96
|
+
this.headersFormArray.removeAt(0);
|
|
97
|
+
}
|
|
98
|
+
if (data.headers && data.headers.length > 0) {
|
|
99
|
+
// Add headers from initial data (include all headers, even if empty)
|
|
100
|
+
data.headers.forEach(header => {
|
|
101
|
+
const headerGroup = this.fb.group({
|
|
102
|
+
name: [header.name || '', Validators.required],
|
|
103
|
+
value: [header.value || '', Validators.required]
|
|
104
|
+
});
|
|
105
|
+
this.headersFormArray.push(headerGroup);
|
|
106
|
+
});
|
|
107
|
+
console.log('loadInitialData: Loaded headers into form array, count:', this.headersFormArray.length);
|
|
108
|
+
}
|
|
109
|
+
// Ensure at least one header row exists
|
|
110
|
+
if (this.headersFormArray.length === 0) {
|
|
111
|
+
console.log('loadInitialData: No headers found, adding empty header row');
|
|
112
|
+
this.addHeader();
|
|
113
|
+
}
|
|
114
|
+
// Load params
|
|
115
|
+
// Clear existing params first
|
|
116
|
+
while (this.paramsFormArray.length !== 0) {
|
|
117
|
+
this.paramsFormArray.removeAt(0);
|
|
118
|
+
}
|
|
119
|
+
if (data.params && data.params.length > 0) {
|
|
120
|
+
// Add params from initial data (include all params, even if empty)
|
|
121
|
+
data.params.forEach(param => {
|
|
122
|
+
const paramGroup = this.fb.group({
|
|
123
|
+
name: [param.name || '', Validators.required],
|
|
124
|
+
value: [param.value || '', Validators.required]
|
|
125
|
+
});
|
|
126
|
+
this.paramsFormArray.push(paramGroup);
|
|
127
|
+
});
|
|
128
|
+
}
|
|
129
|
+
// Load validation rules
|
|
130
|
+
if (data.validation && data.validation.length > 0) {
|
|
131
|
+
// Clear existing validation rules
|
|
132
|
+
while (this.validationFormArray.length !== 0) {
|
|
133
|
+
this.validationFormArray.removeAt(0);
|
|
134
|
+
}
|
|
135
|
+
// Add validation rules from initial data
|
|
136
|
+
data.validation.forEach(rule => {
|
|
137
|
+
const validationGroup = this.fb.group({
|
|
138
|
+
jsonPath: [rule.jsonPath || '', Validators.required],
|
|
139
|
+
verificationType: [rule.verificationType || 'equals', Validators.required],
|
|
140
|
+
expectedType: [rule.expectedType || 'string', Validators.required],
|
|
141
|
+
expectedValue: [rule.expectedValue !== undefined ? rule.expectedValue : '', Validators.required],
|
|
142
|
+
result: [rule.result || 'Not run'],
|
|
143
|
+
checked: [rule.checked || false]
|
|
144
|
+
});
|
|
145
|
+
// Ensure all controls are enabled
|
|
146
|
+
validationGroup.enable({ emitEvent: false });
|
|
147
|
+
this.validationFormArray.push(validationGroup);
|
|
148
|
+
});
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
get headersFormArray() {
|
|
152
|
+
return this.apiForm.get('headers');
|
|
153
|
+
}
|
|
154
|
+
get paramsFormArray() {
|
|
155
|
+
return this.apiForm.get('params');
|
|
156
|
+
}
|
|
157
|
+
get validationFormArray() {
|
|
158
|
+
return this.apiForm.get('validation');
|
|
159
|
+
}
|
|
160
|
+
addHeader() {
|
|
161
|
+
const headerGroup = this.fb.group({
|
|
162
|
+
name: ['', Validators.required],
|
|
163
|
+
value: ['', Validators.required]
|
|
164
|
+
});
|
|
165
|
+
this.headersFormArray.push(headerGroup);
|
|
166
|
+
}
|
|
167
|
+
removeHeader(index) {
|
|
168
|
+
this.headersFormArray.removeAt(index);
|
|
169
|
+
}
|
|
170
|
+
addParam() {
|
|
171
|
+
const paramGroup = this.fb.group({
|
|
172
|
+
name: ['', Validators.required],
|
|
173
|
+
value: ['', Validators.required]
|
|
174
|
+
});
|
|
175
|
+
this.paramsFormArray.push(paramGroup);
|
|
176
|
+
}
|
|
177
|
+
removeParam(index) {
|
|
178
|
+
this.paramsFormArray.removeAt(index);
|
|
179
|
+
}
|
|
180
|
+
ngOnChanges(changes) {
|
|
181
|
+
// Reset config caches when inputs change
|
|
182
|
+
if (changes['httpMethodOptions']) {
|
|
183
|
+
this.httpMethodConfigCache = null;
|
|
184
|
+
}
|
|
185
|
+
if (changes['headerNameOptions']) {
|
|
186
|
+
this.headerNameConfigCache = null;
|
|
187
|
+
}
|
|
188
|
+
// Load initial data if it's set after component initialization
|
|
189
|
+
if (changes['initialData'] && !changes['initialData'].firstChange && this.initialData && !this.hasLoadedInitialData) {
|
|
190
|
+
this.loadInitialData(this.initialData);
|
|
191
|
+
this.hasLoadedInitialData = true;
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
getHttpMethodConfig() {
|
|
195
|
+
if (!this.httpMethodConfigCache) {
|
|
196
|
+
this.httpMethodConfigCache = {
|
|
197
|
+
key: 'method',
|
|
198
|
+
placeholder: 'Select method',
|
|
199
|
+
multiple: false,
|
|
200
|
+
searchable: false,
|
|
201
|
+
options: this.httpMethodOptions
|
|
202
|
+
};
|
|
203
|
+
}
|
|
204
|
+
return this.httpMethodConfigCache;
|
|
205
|
+
}
|
|
206
|
+
getHeaderNameConfig(index) {
|
|
207
|
+
if (!this.headerNameConfigCache) {
|
|
208
|
+
const defaultOptions = [
|
|
209
|
+
{ id: 'string', value: 'string', name: 'string', label: 'string' },
|
|
210
|
+
{ id: 'Content-Type', value: 'Content-Type', name: 'Content-Type', label: 'Content-Type' },
|
|
211
|
+
{ id: 'Authorization', value: 'Authorization', name: 'Authorization', label: 'Authorization' },
|
|
212
|
+
{ id: 'Accept', value: 'Accept', name: 'Accept', label: 'Accept' }
|
|
213
|
+
];
|
|
214
|
+
this.headerNameConfigCache = {
|
|
215
|
+
key: 'name',
|
|
216
|
+
placeholder: 'Select header',
|
|
217
|
+
multiple: false,
|
|
218
|
+
searchable: true,
|
|
219
|
+
options: this.headerNameOptions.length > 0 ? this.headerNameOptions : defaultOptions
|
|
220
|
+
};
|
|
221
|
+
}
|
|
222
|
+
return this.headerNameConfigCache;
|
|
223
|
+
}
|
|
224
|
+
getParamNameConfig(index) {
|
|
225
|
+
if (!this.paramNameConfigCache) {
|
|
226
|
+
this.paramNameConfigCache = {
|
|
227
|
+
key: 'name',
|
|
228
|
+
placeholder: 'Parameter name',
|
|
229
|
+
multiple: false,
|
|
230
|
+
searchable: false,
|
|
231
|
+
options: []
|
|
232
|
+
};
|
|
233
|
+
}
|
|
234
|
+
return this.paramNameConfigCache;
|
|
235
|
+
}
|
|
236
|
+
getHeaderFormGroup(index) {
|
|
237
|
+
return this.headersFormArray.at(index);
|
|
238
|
+
}
|
|
239
|
+
getParamFormGroup(index) {
|
|
240
|
+
return this.paramsFormArray.at(index);
|
|
241
|
+
}
|
|
242
|
+
onTabChange(tab) {
|
|
243
|
+
this.selectedTab = tab;
|
|
244
|
+
}
|
|
245
|
+
onSendRequest() {
|
|
246
|
+
if (this.apiForm.valid) {
|
|
247
|
+
const formValue = this.apiForm.value;
|
|
248
|
+
const headers = formValue.headers.map((h) => ({
|
|
249
|
+
name: h.name,
|
|
250
|
+
value: h.value
|
|
251
|
+
}));
|
|
252
|
+
const params = formValue.params ? formValue.params.map((p) => ({
|
|
253
|
+
name: p.name,
|
|
254
|
+
value: p.value
|
|
255
|
+
})) : [];
|
|
256
|
+
this.sendRequest.emit({
|
|
257
|
+
method: formValue.method,
|
|
258
|
+
url: formValue.url,
|
|
259
|
+
headers,
|
|
260
|
+
body: formValue.body,
|
|
261
|
+
params
|
|
262
|
+
});
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
onImportCurl() {
|
|
266
|
+
this.importCurl.emit();
|
|
267
|
+
}
|
|
268
|
+
onCancel() {
|
|
269
|
+
this.cancelled.emit();
|
|
270
|
+
}
|
|
271
|
+
onCreateStep() {
|
|
272
|
+
if (this.apiForm.valid) {
|
|
273
|
+
const formValue = this.apiForm.value;
|
|
274
|
+
const stepData = {
|
|
275
|
+
method: formValue.method,
|
|
276
|
+
url: formValue.url,
|
|
277
|
+
headers: formValue.headers.map((h) => ({
|
|
278
|
+
name: h.name,
|
|
279
|
+
value: h.value
|
|
280
|
+
})),
|
|
281
|
+
body: formValue.body,
|
|
282
|
+
params: formValue.params ? formValue.params.map((p) => ({
|
|
283
|
+
name: p.name,
|
|
284
|
+
value: p.value
|
|
285
|
+
})) : [],
|
|
286
|
+
scripts: formValue.scripts,
|
|
287
|
+
variableName: formValue.variableName || '',
|
|
288
|
+
validation: formValue.validation ? formValue.validation.map((v) => ({
|
|
289
|
+
jsonPath: v.jsonPath,
|
|
290
|
+
verificationType: v.verificationType,
|
|
291
|
+
expectedType: v.expectedType,
|
|
292
|
+
expectedValue: v.expectedValue,
|
|
293
|
+
result: v.result || 'Not run'
|
|
294
|
+
})) : []
|
|
295
|
+
};
|
|
296
|
+
this.createStep.emit(stepData);
|
|
297
|
+
}
|
|
298
|
+
}
|
|
299
|
+
onNext() {
|
|
300
|
+
// Move to next step or create step if on last step
|
|
301
|
+
const stepOrder = ['request-details', 'store-response', 'validation'];
|
|
302
|
+
const currentIndex = stepOrder.indexOf(this.selectedProgressStep);
|
|
303
|
+
if (currentIndex < stepOrder.length - 1) {
|
|
304
|
+
this.onProgressStepChange(stepOrder[currentIndex + 1]);
|
|
305
|
+
}
|
|
306
|
+
else {
|
|
307
|
+
this.onCreateStep();
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
onBack() {
|
|
311
|
+
// Move to previous step
|
|
312
|
+
const stepOrder = ['request-details', 'store-response', 'validation'];
|
|
313
|
+
const currentIndex = stepOrder.indexOf(this.selectedProgressStep);
|
|
314
|
+
if (currentIndex > 0) {
|
|
315
|
+
this.onProgressStepChange(stepOrder[currentIndex - 1]);
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
canGoBack() {
|
|
319
|
+
const stepOrder = ['request-details', 'store-response', 'validation'];
|
|
320
|
+
const currentIndex = stepOrder.indexOf(this.selectedProgressStep);
|
|
321
|
+
return currentIndex > 0;
|
|
322
|
+
}
|
|
323
|
+
isLastStep() {
|
|
324
|
+
const stepOrder = ['request-details', 'store-response', 'validation'];
|
|
325
|
+
const currentIndex = stepOrder.indexOf(this.selectedProgressStep);
|
|
326
|
+
return currentIndex === stepOrder.length - 1;
|
|
327
|
+
}
|
|
328
|
+
getProgressStepClass(step) {
|
|
329
|
+
const stepOrder = ['request-details', 'store-response', 'validation'];
|
|
330
|
+
const currentIndex = stepOrder.indexOf(this.currentStep);
|
|
331
|
+
const stepIndex = stepOrder.indexOf(step);
|
|
332
|
+
if (stepIndex < currentIndex) {
|
|
333
|
+
return 'cqa-text-gray-400'; // Completed
|
|
334
|
+
}
|
|
335
|
+
else if (stepIndex === currentIndex) {
|
|
336
|
+
return 'cqa-text-purple-600 cqa-font-semibold'; // Active
|
|
337
|
+
}
|
|
338
|
+
else {
|
|
339
|
+
return 'cqa-text-gray-400'; // Pending
|
|
340
|
+
}
|
|
341
|
+
}
|
|
342
|
+
getProgressStepNumber(step) {
|
|
343
|
+
const stepOrder = ['request-details', 'store-response', 'validation'];
|
|
344
|
+
return stepOrder.indexOf(step) + 1;
|
|
345
|
+
}
|
|
346
|
+
getProgressStepLabel(step) {
|
|
347
|
+
const labels = {
|
|
348
|
+
'request-details': 'Request Details',
|
|
349
|
+
'store-response': 'Store Response',
|
|
350
|
+
'validation': 'Validation'
|
|
351
|
+
};
|
|
352
|
+
return labels[step];
|
|
353
|
+
}
|
|
354
|
+
formatJsonResponse(data) {
|
|
355
|
+
if (!data)
|
|
356
|
+
return '';
|
|
357
|
+
try {
|
|
358
|
+
return JSON.stringify(data, null, 2);
|
|
359
|
+
}
|
|
360
|
+
catch {
|
|
361
|
+
return String(data);
|
|
362
|
+
}
|
|
363
|
+
}
|
|
364
|
+
onProgressStepChange(step) {
|
|
365
|
+
this.selectedProgressStep = step;
|
|
366
|
+
// Initialize validation form array if switching to validation step
|
|
367
|
+
if (step === 'validation' && this.validationFormArray.length === 0) {
|
|
368
|
+
this.addValidationRule();
|
|
369
|
+
}
|
|
370
|
+
}
|
|
371
|
+
addValidationRule() {
|
|
372
|
+
const validationGroup = this.fb.group({
|
|
373
|
+
checked: [false],
|
|
374
|
+
jsonPath: ['', Validators.required],
|
|
375
|
+
verificationType: ['equals'],
|
|
376
|
+
expectedType: ['string'],
|
|
377
|
+
expectedValue: ['', Validators.required],
|
|
378
|
+
result: ['Not run']
|
|
379
|
+
});
|
|
380
|
+
// Ensure all controls are enabled to avoid disabled attribute warnings
|
|
381
|
+
validationGroup.enable({ emitEvent: false });
|
|
382
|
+
this.validationFormArray.push(validationGroup);
|
|
383
|
+
}
|
|
384
|
+
removeValidationRule(index) {
|
|
385
|
+
this.validationFormArray.removeAt(index);
|
|
386
|
+
}
|
|
387
|
+
getValidationFormGroup(index) {
|
|
388
|
+
return this.validationFormArray.at(index);
|
|
389
|
+
}
|
|
390
|
+
trackByValidationRuleIndex(index) {
|
|
391
|
+
return index;
|
|
392
|
+
}
|
|
393
|
+
trackByHeaderIndex(index) {
|
|
394
|
+
return index;
|
|
395
|
+
}
|
|
396
|
+
trackByParamIndex(index) {
|
|
397
|
+
return index;
|
|
398
|
+
}
|
|
399
|
+
getVerificationTypeOptions() {
|
|
400
|
+
return [
|
|
401
|
+
{ id: 'equals', value: 'equals', name: 'Equals', label: 'Equals' },
|
|
402
|
+
{ id: 'not_equals', value: 'not_equals', name: 'Not Equals', label: 'Not Equals' },
|
|
403
|
+
{ id: 'contains', value: 'contains', name: 'Contains', label: 'Contains' },
|
|
404
|
+
{ id: 'not_contains', value: 'not_contains', name: 'Not Contains', label: 'Not Contains' },
|
|
405
|
+
{ id: 'greater_than', value: 'greater_than', name: 'Greater Than', label: 'Greater Than' },
|
|
406
|
+
{ id: 'less_than', value: 'less_than', name: 'Less Than', label: 'Less Than' },
|
|
407
|
+
{ id: 'greater_than_or_equals', value: 'greater_than_or_equals', name: 'Greater Than Or Equals', label: 'Greater Than Or Equals' },
|
|
408
|
+
{ id: 'less_than_or_equals', value: 'less_than_or_equals', name: 'Less Than Or Equals', label: 'Less Than Or Equals' },
|
|
409
|
+
{ id: 'exists', value: 'exists', name: 'Exists', label: 'Exists' },
|
|
410
|
+
{ id: 'not_exists', value: 'not_exists', name: 'Not Exists', label: 'Not Exists' }
|
|
411
|
+
];
|
|
412
|
+
}
|
|
413
|
+
getExpectedTypeOptions() {
|
|
414
|
+
return [
|
|
415
|
+
{ id: 'string', value: 'string', name: 'String', label: 'String' },
|
|
416
|
+
{ id: 'number', value: 'number', name: 'Number', label: 'Number' },
|
|
417
|
+
{ id: 'boolean', value: 'boolean', name: 'Boolean', label: 'Boolean' },
|
|
418
|
+
{ id: 'object', value: 'object', name: 'Object', label: 'Object' },
|
|
419
|
+
{ id: 'array', value: 'array', name: 'Array', label: 'Array' },
|
|
420
|
+
{ id: 'null', value: 'null', name: 'Null', label: 'Null' }
|
|
421
|
+
];
|
|
422
|
+
}
|
|
423
|
+
getVerificationTypeConfig(index) {
|
|
424
|
+
// Cache the config to prevent infinite change detection loops
|
|
425
|
+
if (!this.verificationTypeConfigCache) {
|
|
426
|
+
this.verificationTypeConfigCache = {
|
|
427
|
+
key: 'verificationType',
|
|
428
|
+
placeholder: 'Select verification type',
|
|
429
|
+
multiple: false,
|
|
430
|
+
searchable: false,
|
|
431
|
+
options: this.getVerificationTypeOptions()
|
|
432
|
+
};
|
|
433
|
+
}
|
|
434
|
+
return this.verificationTypeConfigCache;
|
|
435
|
+
}
|
|
436
|
+
getExpectedTypeConfig(index) {
|
|
437
|
+
// Cache the config to prevent infinite change detection loops
|
|
438
|
+
if (!this.expectedTypeConfigCache) {
|
|
439
|
+
this.expectedTypeConfigCache = {
|
|
440
|
+
key: 'expectedType',
|
|
441
|
+
placeholder: 'Select expected type',
|
|
442
|
+
multiple: false,
|
|
443
|
+
searchable: false,
|
|
444
|
+
options: this.getExpectedTypeOptions()
|
|
445
|
+
};
|
|
446
|
+
}
|
|
447
|
+
return this.expectedTypeConfigCache;
|
|
448
|
+
}
|
|
449
|
+
getResultClass(result) {
|
|
450
|
+
if (result === 'Pass') {
|
|
451
|
+
return 'cqa-bg-green-500 cqa-text-white';
|
|
452
|
+
}
|
|
453
|
+
else if (result === 'Fail') {
|
|
454
|
+
return 'cqa-bg-red-500 cqa-text-white';
|
|
455
|
+
}
|
|
456
|
+
return 'cqa-bg-gray-300 cqa-text-gray-700';
|
|
457
|
+
}
|
|
458
|
+
get allValidationRulesSelected() {
|
|
459
|
+
return this.validationFormArray.length > 0 &&
|
|
460
|
+
this.validationFormArray.controls.every(control => control.get('checked')?.value);
|
|
461
|
+
}
|
|
462
|
+
get someValidationRulesSelected() {
|
|
463
|
+
return this.validationFormArray.controls.some(control => control.get('checked')?.value) &&
|
|
464
|
+
!this.allValidationRulesSelected;
|
|
465
|
+
}
|
|
466
|
+
onSelectAllValidationRules(checked) {
|
|
467
|
+
this.validationFormArray.controls.forEach(control => {
|
|
468
|
+
control.get('checked')?.setValue(checked);
|
|
469
|
+
});
|
|
470
|
+
}
|
|
471
|
+
onDeleteSelectedValidationRules() {
|
|
472
|
+
const indicesToRemove = [];
|
|
473
|
+
this.validationFormArray.controls.forEach((control, index) => {
|
|
474
|
+
if (control.get('checked')?.value) {
|
|
475
|
+
indicesToRemove.push(index);
|
|
476
|
+
}
|
|
477
|
+
});
|
|
478
|
+
// Remove in reverse order to maintain correct indices
|
|
479
|
+
indicesToRemove.reverse().forEach(index => {
|
|
480
|
+
this.removeValidationRule(index);
|
|
481
|
+
});
|
|
482
|
+
}
|
|
483
|
+
}
|
|
484
|
+
StepBuilderApiComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepBuilderApiComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component });
|
|
485
|
+
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\n class=\"cqa-flex cqa-items-center cqa-gap-2\">\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>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>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>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-center cqa-gap-2 cqa-mb-6\">\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 (valueChange)=\"apiForm.get('url')?.setValue($event)\">\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"] }, { 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", "customClass"], 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"] }] });
|
|
486
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepBuilderApiComponent, decorators: [{
|
|
487
|
+
type: Component,
|
|
488
|
+
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\n class=\"cqa-flex cqa-items-center cqa-gap-2\">\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>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>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>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-center cqa-gap-2 cqa-mb-6\">\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 (valueChange)=\"apiForm.get('url')?.setValue($event)\">\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: [] }]
|
|
489
|
+
}], ctorParameters: function () { return [{ type: i1.FormBuilder }]; }, propDecorators: { httpMethodOptions: [{
|
|
490
|
+
type: Input
|
|
491
|
+
}], headerNameOptions: [{
|
|
492
|
+
type: Input
|
|
493
|
+
}], currentStep: [{
|
|
494
|
+
type: Input
|
|
495
|
+
}], responsePreview: [{
|
|
496
|
+
type: Input
|
|
497
|
+
}], isLoading: [{
|
|
498
|
+
type: Input
|
|
499
|
+
}], initialData: [{
|
|
500
|
+
type: Input
|
|
501
|
+
}], createStep: [{
|
|
502
|
+
type: Output
|
|
503
|
+
}], cancelled: [{
|
|
504
|
+
type: Output
|
|
505
|
+
}], sendRequest: [{
|
|
506
|
+
type: Output
|
|
507
|
+
}], importCurl: [{
|
|
508
|
+
type: Output
|
|
509
|
+
}] } });
|
|
510
|
+
//# 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  // 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    this.apiForm = this.fb.group({\n      method: ['GET', Validators.required],\n      url: ['', Validators.required],\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  ngOnInit(): void {\n    // Sync currentStep input with selectedProgressStep\n    if (this.currentStep) {\n      this.selectedProgressStep = this.currentStep;\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    const headerGroup = this.fb.group({\n      name: ['', 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    }\n    if (changes['headerNameOptions']) {\n      this.headerNameConfigCache = null;\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    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    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      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\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\n         class=\"cqa-flex cqa-items-center cqa-gap-2\">\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>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>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>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-center cqa-gap-2 cqa-mb-6\">\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        (valueChange)=\"apiForm.get('url')?.setValue($event)\">\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"]}
|