@cqa-lib/cqa-ui 1.1.189 → 1.1.190
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2020/lib/add-prerequisite-cases-section/add-prerequisite-cases-section.component.mjs +1 -1
- package/esm2020/lib/button/button.component.mjs +3 -3
- package/esm2020/lib/compare-runs/compare-runs.component.mjs +1 -1
- package/esm2020/lib/custom-textarea/custom-textarea.component.mjs +3 -6
- package/esm2020/lib/dashboards/dashboard-header/dashboard-header.component.mjs +1 -1
- package/esm2020/lib/dynamic-select/dynamic-select-field.component.mjs +58 -3
- package/esm2020/lib/filters/dynamic-filter/dynamic-filter.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-action/step-builder-action.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-ai-agent/step-builder-ai-agent.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-api/step-builder-api.component.mjs +3 -2
- package/esm2020/lib/step-builder/step-builder-condition/step-builder-condition.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-custom-code/step-builder-custom-code.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-database/step-builder-database.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-document/step-builder-document.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-document-generation-template-step/step-builder-document-generation-template-step.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-loop/step-builder-loop.component.mjs +1 -1
- package/esm2020/lib/test-case-details/api-edit-step/api-edit-step.component.mjs +1122 -0
- package/esm2020/lib/test-case-details/api-step/api-step.component.mjs +48 -9
- package/esm2020/lib/test-case-details/condition-step/condition-step.component.mjs +1 -1
- package/esm2020/lib/test-case-details/custom-edit-step/custom-edit-step.component.mjs +1 -1
- package/esm2020/lib/test-case-details/element-popup/element-popup.component.mjs +1 -1
- package/esm2020/lib/test-case-details/loop-step/loop-step.component.mjs +1 -1
- package/esm2020/lib/test-case-details/step-details-drawer/step-details-drawer.component.mjs +1 -1
- package/esm2020/lib/ui-kit.module.mjs +6 -1
- package/esm2020/public-api.mjs +2 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +1370 -120
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +1320 -113
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/custom-textarea/custom-textarea.component.d.ts +1 -2
- package/lib/dynamic-select/dynamic-select-field.component.d.ts +13 -1
- package/lib/test-case-details/api-edit-step/api-edit-step.component.d.ts +346 -0
- package/lib/test-case-details/api-step/api-step.component.d.ts +18 -2
- package/lib/ui-kit.module.d.ts +35 -34
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
- package/src/lib/assets/images/delete.svg +3 -0
- package/styles.css +1 -1
|
@@ -0,0 +1,1122 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, ViewChild, } from '@angular/core';
|
|
2
|
+
import { FormControl } from '@angular/forms';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/forms";
|
|
5
|
+
import * as i2 from "../../button/button.component";
|
|
6
|
+
import * as i3 from "../../dynamic-select/dynamic-select-field.component";
|
|
7
|
+
import * as i4 from "../../custom-input/custom-input.component";
|
|
8
|
+
import * as i5 from "../../custom-textarea/custom-textarea.component";
|
|
9
|
+
import * as i6 from "../../segment-control/segment-control.component";
|
|
10
|
+
import * as i7 from "@angular/common";
|
|
11
|
+
export const API_EDIT_STEP_LABELS = {
|
|
12
|
+
REQUEST_DETAILS: 'Request Details',
|
|
13
|
+
STORE_RESPONSE: 'Store Response',
|
|
14
|
+
VALIDATION: 'Validation',
|
|
15
|
+
};
|
|
16
|
+
const METHODS_WITHOUT_BODY = ['GET', 'HEAD', 'DELETE'];
|
|
17
|
+
/** Auto-close pairs for payload JSON editor: opening char -> closing char. */
|
|
18
|
+
const PAYLOAD_AUTO_CLOSE_PAIRS = { '{': '}', '[': ']', '"': '"', "'": "'" };
|
|
19
|
+
export class ApiEditStepComponent {
|
|
20
|
+
constructor(fb, cdr) {
|
|
21
|
+
this.fb = fb;
|
|
22
|
+
this.cdr = cdr;
|
|
23
|
+
/** Emits the cURL string when user clicks Import (value from the textarea control). */
|
|
24
|
+
this.importCurl = new EventEmitter();
|
|
25
|
+
/** Emits when user cancels the Import cURL panel (clicks Cancel). */
|
|
26
|
+
this.importCurlCancel = new EventEmitter();
|
|
27
|
+
/** Emits when user clicks Send Request, with environment, method, url, and headers. */
|
|
28
|
+
this.sendRequest = new EventEmitter();
|
|
29
|
+
this.back = new EventEmitter();
|
|
30
|
+
this.next = new EventEmitter();
|
|
31
|
+
/** Emits when user clicks Create with all entered details: step1 (environment, HTTP method, URL, headers, body), step2 (variable name), step3 (verifications). */
|
|
32
|
+
this.create = new EventEmitter();
|
|
33
|
+
/** Emits whenever headers change (add, remove, or edit) so the parent can reflect them in the canvas/controls. */
|
|
34
|
+
this.headersChange = new EventEmitter();
|
|
35
|
+
/** Form control for Import cURL textarea; value is emitted when user clicks Import. */
|
|
36
|
+
this.importCurlControl = new FormControl('');
|
|
37
|
+
this.variableName = '';
|
|
38
|
+
this.variableNameError = '';
|
|
39
|
+
/** Controls which body content is visible: headers (default) or import-curl. */
|
|
40
|
+
this.bodyView = 'headers';
|
|
41
|
+
this.stepLabels = [
|
|
42
|
+
{ index: 1, label: API_EDIT_STEP_LABELS.REQUEST_DETAILS },
|
|
43
|
+
{ index: 2, label: API_EDIT_STEP_LABELS.STORE_RESPONSE },
|
|
44
|
+
{ index: 3, label: API_EDIT_STEP_LABELS.VALIDATION },
|
|
45
|
+
];
|
|
46
|
+
this.currentStep = 1;
|
|
47
|
+
this.minStep = 1;
|
|
48
|
+
this.maxStep = 3;
|
|
49
|
+
/** HTTP method options: array of strings or objects with id, name, value, label (passed from parent). */
|
|
50
|
+
this.httpMethodOptions = [];
|
|
51
|
+
/** Config for method dropdown (updated when httpMethodOptions changes to avoid new reference every CD) */
|
|
52
|
+
this.methodSelectConfig = {
|
|
53
|
+
key: 'method',
|
|
54
|
+
placeholder: 'Method',
|
|
55
|
+
searchable: false,
|
|
56
|
+
options: [],
|
|
57
|
+
};
|
|
58
|
+
/** Environment options: array of strings or objects with id, name, value, label (passed from parent). */
|
|
59
|
+
this.environmentOptions = [];
|
|
60
|
+
/** Config for environment dropdown (updated when environmentOptions changes to avoid new reference every CD) */
|
|
61
|
+
this.environmentSelectConfig = {
|
|
62
|
+
key: 'environment',
|
|
63
|
+
placeholder: 'Environment',
|
|
64
|
+
searchable: false,
|
|
65
|
+
options: [],
|
|
66
|
+
};
|
|
67
|
+
/** Cached environment value from last selection (so Create payload has the selected value). */
|
|
68
|
+
this.currentEnvironmentValue = '';
|
|
69
|
+
/** Cached HTTP method from last selection (so Create payload has the selected value). */
|
|
70
|
+
this.currentMethodValue = '';
|
|
71
|
+
this.url = '';
|
|
72
|
+
this.payloadTabs = [
|
|
73
|
+
{ value: 'headers', label: 'Headers' },
|
|
74
|
+
{ value: 'body', label: 'Body' },
|
|
75
|
+
{ value: 'params', label: 'Params' },
|
|
76
|
+
{ value: 'scripts', label: 'Scripts' },
|
|
77
|
+
];
|
|
78
|
+
this.activePayloadTab = 'headers';
|
|
79
|
+
/** Step 3: Response verification tabs */
|
|
80
|
+
this.responseVerificationTabs = [
|
|
81
|
+
{ value: 'response-body', label: 'Response Body' },
|
|
82
|
+
{ value: 'status', label: 'Status' },
|
|
83
|
+
];
|
|
84
|
+
this.activeResponseVerificationTab = 'response-body';
|
|
85
|
+
/** Payload type for Body/Params/Scripts: raw, x-www-form-urlencoded, form-data */
|
|
86
|
+
this.payloadType = 'raw';
|
|
87
|
+
/** Segment options for payload type (used by cqa-segment-control) */
|
|
88
|
+
this.payloadTypeSegments = [
|
|
89
|
+
{ label: 'Raw', value: 'raw' },
|
|
90
|
+
{ label: 'x-www-form-urlencoded', value: 'x-www-form-urlencoded' },
|
|
91
|
+
{ label: 'Form Data', value: 'form-data' },
|
|
92
|
+
];
|
|
93
|
+
/** Format options: array of strings or objects (passed from parent). Falls back to JSON, XML, HTML, Text when empty. */
|
|
94
|
+
this.formatOptions = [];
|
|
95
|
+
/** Config for Format dropdown (updated when formatOptions changes, like Method dropdown) */
|
|
96
|
+
this.payloadFormatSelectConfig = {
|
|
97
|
+
key: 'format',
|
|
98
|
+
placeholder: 'Select',
|
|
99
|
+
searchable: false,
|
|
100
|
+
options: [...ApiEditStepComponent.DEFAULT_FORMAT_OPTIONS],
|
|
101
|
+
};
|
|
102
|
+
/** Payload text area value for Body/Params/Scripts */
|
|
103
|
+
this.payloadText = '';
|
|
104
|
+
/** JSON parse error when format is JSON and payload is invalid; null when valid or not JSON. */
|
|
105
|
+
this.payloadJsonError = null;
|
|
106
|
+
/** Header name options: array of strings or objects with id, name, value, label (passed from parent). Falls back to built-in list when empty. */
|
|
107
|
+
this.headerNameOptions = [];
|
|
108
|
+
/** Config for header name dropdown (updated when headerNameOptions changes to avoid new reference every CD) */
|
|
109
|
+
this.headerNameSelectConfig = {
|
|
110
|
+
key: 'name',
|
|
111
|
+
placeholder: 'Select header',
|
|
112
|
+
searchable: true,
|
|
113
|
+
allowCustomValue: true,
|
|
114
|
+
options: [],
|
|
115
|
+
};
|
|
116
|
+
this.defaultHeaders = [
|
|
117
|
+
{ name: 'Content-Type', type: 'string', value: 'application/json' },
|
|
118
|
+
];
|
|
119
|
+
/** Config for Type dropdown in key-type-value rows (Text, File) */
|
|
120
|
+
this.keyTypeValueTypeSelectConfig = {
|
|
121
|
+
key: 'type',
|
|
122
|
+
placeholder: 'Text',
|
|
123
|
+
searchable: false,
|
|
124
|
+
options: [
|
|
125
|
+
{ id: 'text', value: 'text', name: 'Text', label: 'Text' },
|
|
126
|
+
{ id: 'file', value: 'file', name: 'File', label: 'File' },
|
|
127
|
+
],
|
|
128
|
+
};
|
|
129
|
+
/** Verification options: array of strings or objects (passed from parent). Falls back to built-in list when empty. */
|
|
130
|
+
this.verificationOptions = [];
|
|
131
|
+
/** Config for Verification dropdown (updated when verificationOptions changes, like Method dropdown) */
|
|
132
|
+
this.verificationSelectConfig = {
|
|
133
|
+
key: 'verification',
|
|
134
|
+
placeholder: 'Equals',
|
|
135
|
+
searchable: false,
|
|
136
|
+
options: [...ApiEditStepComponent.DEFAULT_VERIFICATION_OPTIONS],
|
|
137
|
+
};
|
|
138
|
+
/** Data Type options: array of strings or objects (passed from parent). Falls back to built-in list when empty. */
|
|
139
|
+
this.verificationDataTypeOptions = [];
|
|
140
|
+
/** Config for Data Type dropdown (updated when verificationDataTypeOptions changes, like Method dropdown) */
|
|
141
|
+
this.verificationDataTypeSelectConfig = {
|
|
142
|
+
key: 'dataType',
|
|
143
|
+
placeholder: 'String',
|
|
144
|
+
searchable: false,
|
|
145
|
+
options: [...ApiEditStepComponent.DEFAULT_DATA_TYPE_OPTIONS],
|
|
146
|
+
};
|
|
147
|
+
/** Config for Expected Value when Data Type is Boolean (true/false dropdown) */
|
|
148
|
+
this.verificationExpectedValueBooleanSelectConfig = {
|
|
149
|
+
key: 'expectedValue',
|
|
150
|
+
placeholder: 'Select',
|
|
151
|
+
searchable: false,
|
|
152
|
+
options: [
|
|
153
|
+
{ id: 'true', value: 'true', name: 'true', label: 'true' },
|
|
154
|
+
{ id: 'false', value: 'false', name: 'false', label: 'false' },
|
|
155
|
+
],
|
|
156
|
+
};
|
|
157
|
+
/** Config for Status tab Verification dropdown (Equals, Not equal, Less than, Greater than) */
|
|
158
|
+
/** Status tab Verification options (Equals, Not equal, etc.). Falls back to built-in list when empty. */
|
|
159
|
+
this.statusVerificationOptions = [];
|
|
160
|
+
/** Config for Status tab Verification dropdown (updated when statusVerificationOptions changes). */
|
|
161
|
+
this.statusVerificationSelectConfig = {
|
|
162
|
+
key: 'verification',
|
|
163
|
+
placeholder: 'Equals',
|
|
164
|
+
searchable: false,
|
|
165
|
+
options: [...ApiEditStepComponent.DEFAULT_STATUS_VERIFICATION_OPTIONS],
|
|
166
|
+
};
|
|
167
|
+
this.responsePreview = ``;
|
|
168
|
+
}
|
|
169
|
+
/** The label for the current step only (used when showing a single step in the indicator). */
|
|
170
|
+
get activeStepLabel() {
|
|
171
|
+
return this.stepLabels[this.currentStep - 1];
|
|
172
|
+
}
|
|
173
|
+
/** Percentage to translate the step strip (0, -33.333, -66.666) for CSS transition. */
|
|
174
|
+
get stripTranslatePercent() {
|
|
175
|
+
return (this.currentStep - 1) * -33.333;
|
|
176
|
+
}
|
|
177
|
+
setStep(step) {
|
|
178
|
+
if (step >= this.minStep && step <= this.maxStep) {
|
|
179
|
+
this.currentStep = step;
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
applyInitialStep(value) {
|
|
183
|
+
if (value == null)
|
|
184
|
+
return;
|
|
185
|
+
const clamped = Math.max(this.minStep, Math.min(this.maxStep, Math.floor(value)));
|
|
186
|
+
this.currentStep = clamped;
|
|
187
|
+
}
|
|
188
|
+
updateMethodSelectConfig() {
|
|
189
|
+
const options = (this.httpMethodOptions ?? []).map((m) => ApiEditStepComponent.toSelectOption(m));
|
|
190
|
+
this.methodSelectConfig = {
|
|
191
|
+
...this.methodSelectConfig,
|
|
192
|
+
options,
|
|
193
|
+
};
|
|
194
|
+
}
|
|
195
|
+
getMethodValues() {
|
|
196
|
+
return (this.httpMethodOptions ?? [])
|
|
197
|
+
.map((m) => ApiEditStepComponent.getOptionValue(m))
|
|
198
|
+
.filter((v) => v != null);
|
|
199
|
+
}
|
|
200
|
+
static toSelectOption(item) {
|
|
201
|
+
if (typeof item === 'string') {
|
|
202
|
+
return { id: item, value: item, name: item, label: item };
|
|
203
|
+
}
|
|
204
|
+
const o = item;
|
|
205
|
+
return {
|
|
206
|
+
id: o.id ?? o.value,
|
|
207
|
+
value: o.value ?? o.id,
|
|
208
|
+
name: o.name ?? o.label ?? String(o.value ?? o.id),
|
|
209
|
+
label: o.label ?? o.name ?? String(o.value ?? o.id),
|
|
210
|
+
};
|
|
211
|
+
}
|
|
212
|
+
static getOptionValue(item) {
|
|
213
|
+
if (typeof item === 'string')
|
|
214
|
+
return item;
|
|
215
|
+
const o = item;
|
|
216
|
+
const v = o.value ?? o.id;
|
|
217
|
+
return v != null ? String(v) : undefined;
|
|
218
|
+
}
|
|
219
|
+
updateEnvironmentSelectConfig() {
|
|
220
|
+
const options = (this.environmentOptions ?? []).map((e) => ApiEditStepComponent.toSelectOption(e));
|
|
221
|
+
this.environmentSelectConfig = {
|
|
222
|
+
...this.environmentSelectConfig,
|
|
223
|
+
options,
|
|
224
|
+
};
|
|
225
|
+
}
|
|
226
|
+
getEnvironmentValues() {
|
|
227
|
+
return (this.environmentOptions ?? [])
|
|
228
|
+
.map((e) => ApiEditStepComponent.getOptionValue(e))
|
|
229
|
+
.filter((v) => v != null);
|
|
230
|
+
}
|
|
231
|
+
/** Current HTTP method (from form or default: first of httpMethodOptions). Treats empty string as unset. */
|
|
232
|
+
get selectedMethod() {
|
|
233
|
+
const value = this.methodForm?.get('method')?.value;
|
|
234
|
+
const firstValue = this.httpMethodOptions?.length
|
|
235
|
+
? ApiEditStepComponent.getOptionValue(this.httpMethodOptions[0])
|
|
236
|
+
: undefined;
|
|
237
|
+
const resolved = value != null && value !== '' ? (typeof value === 'object' ? ApiEditStepComponent.getOptionValue(value) : String(value)) : undefined;
|
|
238
|
+
return resolved ?? firstValue ?? 'GET';
|
|
239
|
+
}
|
|
240
|
+
set selectedMethod(v) {
|
|
241
|
+
this.methodForm?.patchValue({ method: v });
|
|
242
|
+
}
|
|
243
|
+
/** True when the selected HTTP method is GET, HEAD, or DELETE (no request body). */
|
|
244
|
+
get isMethodWithoutBody() {
|
|
245
|
+
const method = this.selectedMethod != null ? String(this.selectedMethod).toUpperCase() : '';
|
|
246
|
+
return METHODS_WITHOUT_BODY.includes(method);
|
|
247
|
+
}
|
|
248
|
+
/** Current environment (from form or default: first of environmentOptions). Treats empty string as unset. */
|
|
249
|
+
get selectedEnvironment() {
|
|
250
|
+
const value = this.environmentForm?.get('environment')?.value;
|
|
251
|
+
const firstValue = this.environmentOptions?.length
|
|
252
|
+
? ApiEditStepComponent.getOptionValue(this.environmentOptions[0])
|
|
253
|
+
: undefined;
|
|
254
|
+
const resolved = value != null && value !== '' ? (typeof value === 'object' ? ApiEditStepComponent.getOptionValue(value) : String(value)) : undefined;
|
|
255
|
+
return resolved ?? firstValue ?? 'Development';
|
|
256
|
+
}
|
|
257
|
+
set selectedEnvironment(v) {
|
|
258
|
+
this.environmentForm?.patchValue({ environment: v });
|
|
259
|
+
}
|
|
260
|
+
/** Called when user selects an environment; keeps currentEnvironmentValue in sync for create payload. */
|
|
261
|
+
onEnvironmentSelectionChange(event) {
|
|
262
|
+
const v = event.value;
|
|
263
|
+
this.currentEnvironmentValue =
|
|
264
|
+
v != null
|
|
265
|
+
? typeof v === 'object'
|
|
266
|
+
? (ApiEditStepComponent.getOptionValue(v) ?? '')
|
|
267
|
+
: String(v)
|
|
268
|
+
: '';
|
|
269
|
+
}
|
|
270
|
+
/** Called when user selects an HTTP method; keeps currentMethodValue in sync for create payload. */
|
|
271
|
+
onMethodSelectionChange(event) {
|
|
272
|
+
const v = event.value;
|
|
273
|
+
this.currentMethodValue =
|
|
274
|
+
v != null
|
|
275
|
+
? typeof v === 'object'
|
|
276
|
+
? (ApiEditStepComponent.getOptionValue(v) ?? '')
|
|
277
|
+
: String(v)
|
|
278
|
+
: '';
|
|
279
|
+
}
|
|
280
|
+
onPayloadTypeChange(val) {
|
|
281
|
+
this.payloadType = val;
|
|
282
|
+
}
|
|
283
|
+
updatePayloadFormatSelectConfig() {
|
|
284
|
+
const options = this.formatOptions?.length > 0
|
|
285
|
+
? this.formatOptions.map((f) => ApiEditStepComponent.toSelectOption(f))
|
|
286
|
+
: ApiEditStepComponent.DEFAULT_FORMAT_OPTIONS;
|
|
287
|
+
this.payloadFormatSelectConfig = {
|
|
288
|
+
...this.payloadFormatSelectConfig,
|
|
289
|
+
options,
|
|
290
|
+
};
|
|
291
|
+
}
|
|
292
|
+
getFormatValues() {
|
|
293
|
+
const source = this.formatOptions?.length > 0
|
|
294
|
+
? this.formatOptions
|
|
295
|
+
: ApiEditStepComponent.DEFAULT_FORMAT_OPTIONS;
|
|
296
|
+
return source
|
|
297
|
+
.map((f) => ApiEditStepComponent.getOptionValue(f))
|
|
298
|
+
.filter((v) => v != null);
|
|
299
|
+
}
|
|
300
|
+
/** Line numbers for payload textarea (1, 2, 3, ...) based on current line count. */
|
|
301
|
+
get payloadLineNumbers() {
|
|
302
|
+
const n = Math.max(1, (this.payloadText || '').split(/\r?\n/).length);
|
|
303
|
+
return Array.from({ length: n }, (_, i) => i + 1);
|
|
304
|
+
}
|
|
305
|
+
/** Whether current payload format is JSON (so we validate and show JSON editor behavior). */
|
|
306
|
+
get isPayloadFormatJson() {
|
|
307
|
+
const format = this.payloadFormatForm?.get('format')?.value;
|
|
308
|
+
return format === 'json';
|
|
309
|
+
}
|
|
310
|
+
/** Error messages for the payload textarea (red border + bottom line). */
|
|
311
|
+
get payloadJsonErrors() {
|
|
312
|
+
return this.payloadJsonError ? ['Invalid JSON format. Please check your syntax.'] : [];
|
|
313
|
+
}
|
|
314
|
+
/** Tooltip content for the line-level error icon (parse error on line X, excerpt, caret, message). */
|
|
315
|
+
get payloadJsonErrorTooltip() {
|
|
316
|
+
if (!this.payloadJsonError)
|
|
317
|
+
return '';
|
|
318
|
+
const e = this.payloadJsonError;
|
|
319
|
+
const caret = ' '.repeat(e.caretOffset) + '^';
|
|
320
|
+
return `Parse error on line ${e.line}:\n\n${e.excerpt}\n${caret}\n\n${e.message}`;
|
|
321
|
+
}
|
|
322
|
+
onPayloadTextChange(value) {
|
|
323
|
+
this.payloadText = value;
|
|
324
|
+
this.validatePayloadJson();
|
|
325
|
+
}
|
|
326
|
+
onPayloadInput(event) {
|
|
327
|
+
const target = event.target;
|
|
328
|
+
this.onPayloadTextChange(target?.value ?? '');
|
|
329
|
+
}
|
|
330
|
+
onPayloadKeydown(event) {
|
|
331
|
+
if (!this.isPayloadFormatJson)
|
|
332
|
+
return;
|
|
333
|
+
const key = event.key;
|
|
334
|
+
const close = PAYLOAD_AUTO_CLOSE_PAIRS[key];
|
|
335
|
+
if (!close)
|
|
336
|
+
return;
|
|
337
|
+
const target = event.target;
|
|
338
|
+
if (!target)
|
|
339
|
+
return;
|
|
340
|
+
event.preventDefault();
|
|
341
|
+
const start = target.selectionStart ?? 0;
|
|
342
|
+
const end = target.selectionEnd ?? start;
|
|
343
|
+
const before = this.payloadText.slice(0, start);
|
|
344
|
+
const after = this.payloadText.slice(end);
|
|
345
|
+
const insertPos = start + 1;
|
|
346
|
+
const newValue = before + key + close + after;
|
|
347
|
+
this.payloadText = newValue;
|
|
348
|
+
this.validatePayloadJson();
|
|
349
|
+
setTimeout(() => {
|
|
350
|
+
this.payloadTextareaRef?.nativeElement?.setSelectionRange(insertPos, insertPos);
|
|
351
|
+
}, 0);
|
|
352
|
+
}
|
|
353
|
+
/** Validate payload when format is JSON; sets payloadJsonError or clears it. */
|
|
354
|
+
validatePayloadJson() {
|
|
355
|
+
if (!this.isPayloadFormatJson) {
|
|
356
|
+
this.payloadJsonError = null;
|
|
357
|
+
return;
|
|
358
|
+
}
|
|
359
|
+
const text = this.payloadText ?? '';
|
|
360
|
+
if (text.trim() === '') {
|
|
361
|
+
this.payloadJsonError = null;
|
|
362
|
+
return;
|
|
363
|
+
}
|
|
364
|
+
try {
|
|
365
|
+
JSON.parse(text);
|
|
366
|
+
this.payloadJsonError = null;
|
|
367
|
+
}
|
|
368
|
+
catch (err) {
|
|
369
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
370
|
+
const position = this.extractPositionFromJsonError(message);
|
|
371
|
+
const { line, column } = this.positionToLineColumn(text, position);
|
|
372
|
+
const lines = text.split(/\r?\n/);
|
|
373
|
+
const lineIndex = Math.max(0, line - 1);
|
|
374
|
+
const lineContent = lines[lineIndex] ?? '';
|
|
375
|
+
const excerpt = lineContent || ' ';
|
|
376
|
+
const caretOffset = lineIndex === line - 1 ? Math.min(column - 1, lineContent.length) : 0;
|
|
377
|
+
this.payloadJsonError = {
|
|
378
|
+
line,
|
|
379
|
+
column,
|
|
380
|
+
message: this.normalizeJsonErrorMessage(message),
|
|
381
|
+
excerpt,
|
|
382
|
+
caretOffset,
|
|
383
|
+
};
|
|
384
|
+
}
|
|
385
|
+
this.cdr.markForCheck();
|
|
386
|
+
}
|
|
387
|
+
extractPositionFromJsonError(message) {
|
|
388
|
+
const match = message.match(/position\s+(\d+)/i) || message.match(/at\s+position\s+(\d+)/i);
|
|
389
|
+
return match ? parseInt(match[1], 10) : 0;
|
|
390
|
+
}
|
|
391
|
+
positionToLineColumn(text, position) {
|
|
392
|
+
const before = text.slice(0, Math.min(position, text.length));
|
|
393
|
+
const lines = before.split(/\r?\n/);
|
|
394
|
+
const line = lines.length;
|
|
395
|
+
const lastLine = lines[lines.length - 1] ?? '';
|
|
396
|
+
return { line, column: lastLine.length + 1 };
|
|
397
|
+
}
|
|
398
|
+
normalizeJsonErrorMessage(message) {
|
|
399
|
+
if (message.includes('Expecting') || message.includes('expecting'))
|
|
400
|
+
return message;
|
|
401
|
+
const tokenMatch = message.match(/Unexpected token\s+['"]?([^'"]+)['"]?/i)
|
|
402
|
+
|| message.match(/unexpected character.*?([^\s]+)/i);
|
|
403
|
+
const token = tokenMatch ? tokenMatch[1].trim() : 'undefined';
|
|
404
|
+
return `Expecting 'STRING', 'NUMBER', 'NULL', 'TRUE', 'FALSE', '{', '[', got '${token}'`;
|
|
405
|
+
}
|
|
406
|
+
/** Typed accessor for template (FormArray.controls is AbstractControl[]) */
|
|
407
|
+
get headerRows() {
|
|
408
|
+
return (this.headersFormArray?.controls ?? []);
|
|
409
|
+
}
|
|
410
|
+
updateHeaderNameSelectConfig() {
|
|
411
|
+
const source = this.headerNameOptions?.length > 0
|
|
412
|
+
? this.headerNameOptions
|
|
413
|
+
: ApiEditStepComponent.DEFAULT_HEADER_NAME_OPTIONS;
|
|
414
|
+
const baseOptions = source.map((h) => ApiEditStepComponent.toSelectOption(h));
|
|
415
|
+
const baseValues = new Set(baseOptions.map((o) => String(o.value ?? o.id ?? '').trim()).filter(Boolean));
|
|
416
|
+
const currentNamesFromForm = this.getCurrentHeaderNamesFromForm();
|
|
417
|
+
const extraOptions = [];
|
|
418
|
+
for (const name of currentNamesFromForm) {
|
|
419
|
+
const key = name.trim();
|
|
420
|
+
if (key && !baseValues.has(key)) {
|
|
421
|
+
baseValues.add(key);
|
|
422
|
+
extraOptions.push({ id: key, value: key, name: key, label: key });
|
|
423
|
+
}
|
|
424
|
+
}
|
|
425
|
+
const options = [...baseOptions, ...extraOptions];
|
|
426
|
+
this.headerNameSelectConfig = {
|
|
427
|
+
...this.headerNameSelectConfig,
|
|
428
|
+
options,
|
|
429
|
+
};
|
|
430
|
+
}
|
|
431
|
+
/** Collects distinct non-empty header names currently in the headers form (for dynamic options). */
|
|
432
|
+
getCurrentHeaderNamesFromForm() {
|
|
433
|
+
const names = [];
|
|
434
|
+
const seen = new Set();
|
|
435
|
+
if (!this.headersFormArray?.controls?.length)
|
|
436
|
+
return names;
|
|
437
|
+
for (const control of this.headersFormArray.controls) {
|
|
438
|
+
const name = control.get('name')?.value;
|
|
439
|
+
const str = name != null ? String(name).trim() : '';
|
|
440
|
+
if (str && !seen.has(str)) {
|
|
441
|
+
seen.add(str);
|
|
442
|
+
names.push(str);
|
|
443
|
+
}
|
|
444
|
+
}
|
|
445
|
+
return names;
|
|
446
|
+
}
|
|
447
|
+
get keyValueRows() {
|
|
448
|
+
return (this.keyValueFormArray?.controls ?? []);
|
|
449
|
+
}
|
|
450
|
+
get keyTypeValueRows() {
|
|
451
|
+
return (this.keyTypeValueFormArray?.controls ?? []);
|
|
452
|
+
}
|
|
453
|
+
get verificationRows() {
|
|
454
|
+
return (this.verificationFormArray?.controls ?? []);
|
|
455
|
+
}
|
|
456
|
+
updateVerificationSelectConfig() {
|
|
457
|
+
const options = this.verificationOptions?.length > 0
|
|
458
|
+
? this.verificationOptions.map((v) => ApiEditStepComponent.toSelectOption(v))
|
|
459
|
+
: ApiEditStepComponent.DEFAULT_VERIFICATION_OPTIONS;
|
|
460
|
+
this.verificationSelectConfig = {
|
|
461
|
+
...this.verificationSelectConfig,
|
|
462
|
+
options,
|
|
463
|
+
};
|
|
464
|
+
}
|
|
465
|
+
getVerificationValues() {
|
|
466
|
+
const source = this.verificationOptions?.length > 0
|
|
467
|
+
? this.verificationOptions
|
|
468
|
+
: ApiEditStepComponent.DEFAULT_VERIFICATION_OPTIONS;
|
|
469
|
+
return source
|
|
470
|
+
.map((v) => ApiEditStepComponent.getOptionValue(v))
|
|
471
|
+
.filter((v) => v != null);
|
|
472
|
+
}
|
|
473
|
+
getDefaultVerification() {
|
|
474
|
+
const values = this.getVerificationValues();
|
|
475
|
+
return values.length > 0 ? values[0] : 'equals';
|
|
476
|
+
}
|
|
477
|
+
updateVerificationDataTypeSelectConfig() {
|
|
478
|
+
const options = this.verificationDataTypeOptions?.length > 0
|
|
479
|
+
? this.verificationDataTypeOptions.map((d) => ApiEditStepComponent.toSelectOption(d))
|
|
480
|
+
: ApiEditStepComponent.DEFAULT_DATA_TYPE_OPTIONS;
|
|
481
|
+
this.verificationDataTypeSelectConfig = {
|
|
482
|
+
...this.verificationDataTypeSelectConfig,
|
|
483
|
+
options,
|
|
484
|
+
};
|
|
485
|
+
}
|
|
486
|
+
getDataTypeValues() {
|
|
487
|
+
const source = this.verificationDataTypeOptions?.length > 0
|
|
488
|
+
? this.verificationDataTypeOptions
|
|
489
|
+
: ApiEditStepComponent.DEFAULT_DATA_TYPE_OPTIONS;
|
|
490
|
+
return source
|
|
491
|
+
.map((d) => ApiEditStepComponent.getOptionValue(d))
|
|
492
|
+
.filter((v) => v != null);
|
|
493
|
+
}
|
|
494
|
+
getDefaultDataType() {
|
|
495
|
+
const values = this.getDataTypeValues();
|
|
496
|
+
return values.length > 0 ? values[0] : 'string';
|
|
497
|
+
}
|
|
498
|
+
get statusVerificationRows() {
|
|
499
|
+
return (this.statusVerificationFormArray?.controls ?? []);
|
|
500
|
+
}
|
|
501
|
+
updateStatusVerificationSelectConfig() {
|
|
502
|
+
const options = this.statusVerificationOptions?.length > 0
|
|
503
|
+
? this.statusVerificationOptions.map((o) => ApiEditStepComponent.toSelectOption(o))
|
|
504
|
+
: ApiEditStepComponent.DEFAULT_STATUS_VERIFICATION_OPTIONS;
|
|
505
|
+
this.statusVerificationSelectConfig = {
|
|
506
|
+
...this.statusVerificationSelectConfig,
|
|
507
|
+
options,
|
|
508
|
+
};
|
|
509
|
+
}
|
|
510
|
+
getStatusVerificationValues() {
|
|
511
|
+
const source = this.statusVerificationOptions?.length > 0
|
|
512
|
+
? this.statusVerificationOptions
|
|
513
|
+
: ApiEditStepComponent.DEFAULT_STATUS_VERIFICATION_OPTIONS;
|
|
514
|
+
return source
|
|
515
|
+
.map((o) => ApiEditStepComponent.getOptionValue(o))
|
|
516
|
+
.filter((v) => v != null);
|
|
517
|
+
}
|
|
518
|
+
getDefaultStatusVerification() {
|
|
519
|
+
const values = this.getStatusVerificationValues();
|
|
520
|
+
return values.length > 0 ? values[0] : 'equals';
|
|
521
|
+
}
|
|
522
|
+
ngOnInit() {
|
|
523
|
+
this.applyInitialStep(this.initialStep);
|
|
524
|
+
this.updateMethodSelectConfig();
|
|
525
|
+
this.updateHeaderNameSelectConfig();
|
|
526
|
+
const firstMethodValue = this.httpMethodOptions?.length
|
|
527
|
+
? ApiEditStepComponent.getOptionValue(this.httpMethodOptions[0])
|
|
528
|
+
: undefined;
|
|
529
|
+
const defaultMethod = this.initialMethod ?? firstMethodValue ?? 'GET';
|
|
530
|
+
this.currentMethodValue = typeof defaultMethod === 'string' ? defaultMethod : (ApiEditStepComponent.getOptionValue(defaultMethod) ?? '');
|
|
531
|
+
this.methodForm = this.fb.group({
|
|
532
|
+
method: [defaultMethod],
|
|
533
|
+
});
|
|
534
|
+
const firstValue = this.environmentOptions?.length
|
|
535
|
+
? ApiEditStepComponent.getOptionValue(this.environmentOptions[0])
|
|
536
|
+
: undefined;
|
|
537
|
+
const defaultEnv = this.initialEnvironment ?? firstValue ?? 'Development';
|
|
538
|
+
this.currentEnvironmentValue = typeof defaultEnv === 'string' ? defaultEnv : (ApiEditStepComponent.getOptionValue(defaultEnv) ?? '');
|
|
539
|
+
this.environmentForm = this.fb.group({
|
|
540
|
+
environment: [defaultEnv],
|
|
541
|
+
});
|
|
542
|
+
this.environmentFormChangesSub = this.environmentForm.get('environment')?.valueChanges?.subscribe((v) => {
|
|
543
|
+
const s = v != null && v !== '' ? (typeof v === 'object' ? ApiEditStepComponent.getOptionValue(v) : String(v)) : '';
|
|
544
|
+
this.currentEnvironmentValue = s ?? '';
|
|
545
|
+
});
|
|
546
|
+
this.methodFormChangesSub = this.methodForm.get('method')?.valueChanges?.subscribe((v) => {
|
|
547
|
+
const s = v != null && v !== '' ? (typeof v === 'object' ? ApiEditStepComponent.getOptionValue(v) : String(v)) : '';
|
|
548
|
+
this.currentMethodValue = s ?? '';
|
|
549
|
+
});
|
|
550
|
+
this.buildHeadersFormArray(this.initialHeaders ?? [{ name: '', type: 'string', value: '' }]);
|
|
551
|
+
this.headerNameOptionsChangesSub = this.headersFormArray?.valueChanges?.subscribe(() => {
|
|
552
|
+
this.updateHeaderNameSelectConfig();
|
|
553
|
+
this.headersChange.emit(this.headers);
|
|
554
|
+
});
|
|
555
|
+
this.updateHeaderNameSelectConfig();
|
|
556
|
+
this.headersChange.emit(this.headers);
|
|
557
|
+
this.updateEnvironmentSelectConfig();
|
|
558
|
+
this.updatePayloadFormatSelectConfig();
|
|
559
|
+
const firstFormatValue = this.getFormatValues()[0];
|
|
560
|
+
const defaultFormat = this.initialFormat ?? firstFormatValue ?? 'json';
|
|
561
|
+
this.payloadFormatForm = this.fb.group({ format: [defaultFormat] });
|
|
562
|
+
this.buildKeyValueFormArray();
|
|
563
|
+
this.buildKeyTypeValueFormArray();
|
|
564
|
+
this.updateVerificationSelectConfig();
|
|
565
|
+
this.updateVerificationDataTypeSelectConfig();
|
|
566
|
+
this.buildVerificationFormArray();
|
|
567
|
+
this.updateStatusVerificationSelectConfig();
|
|
568
|
+
this.buildStatusVerificationFormArray();
|
|
569
|
+
this.methodChangesSub = this.methodForm.get('method')?.valueChanges?.subscribe((value) => {
|
|
570
|
+
const method = value != null ? String(value).toUpperCase() : '';
|
|
571
|
+
if (METHODS_WITHOUT_BODY.includes(method) && this.activePayloadTab !== 'headers') {
|
|
572
|
+
this.activePayloadTab = 'headers';
|
|
573
|
+
this.cdr.detectChanges();
|
|
574
|
+
}
|
|
575
|
+
});
|
|
576
|
+
this.formatChangesSub = this.payloadFormatForm.get('format')?.valueChanges?.subscribe(() => {
|
|
577
|
+
this.validatePayloadJson();
|
|
578
|
+
});
|
|
579
|
+
this.validatePayloadJson();
|
|
580
|
+
}
|
|
581
|
+
ngAfterViewInit() {
|
|
582
|
+
this.setupPayloadTextareaScrollSync();
|
|
583
|
+
}
|
|
584
|
+
ngOnDestroy() {
|
|
585
|
+
this.methodChangesSub?.unsubscribe();
|
|
586
|
+
this.formatChangesSub?.unsubscribe();
|
|
587
|
+
this.headerNameOptionsChangesSub?.unsubscribe();
|
|
588
|
+
this.environmentFormChangesSub?.unsubscribe();
|
|
589
|
+
this.methodFormChangesSub?.unsubscribe();
|
|
590
|
+
}
|
|
591
|
+
/** Sync line numbers column scroll with payload textarea scroll (Postman-style). */
|
|
592
|
+
setupPayloadTextareaScrollSync() {
|
|
593
|
+
setTimeout(() => {
|
|
594
|
+
const host = this.payloadEditorWithLinesRef?.nativeElement;
|
|
595
|
+
if (!host)
|
|
596
|
+
return;
|
|
597
|
+
const lineNumbersEl = host.querySelector('.cqa-api-edit-step-payload-line-numbers');
|
|
598
|
+
const textarea = host.querySelector('textarea');
|
|
599
|
+
if (!lineNumbersEl || !textarea)
|
|
600
|
+
return;
|
|
601
|
+
textarea.addEventListener('scroll', () => {
|
|
602
|
+
lineNumbersEl.scrollTop = textarea.scrollTop;
|
|
603
|
+
});
|
|
604
|
+
}, 0);
|
|
605
|
+
}
|
|
606
|
+
buildKeyValueFormArray(rows) {
|
|
607
|
+
const initial = rows ?? [{ key: '', value: '' }];
|
|
608
|
+
const groups = initial.map((r) => this.fb.group({ key: [r.key], value: [r.value] }));
|
|
609
|
+
this.keyValueFormArray = this.fb.array(groups);
|
|
610
|
+
}
|
|
611
|
+
buildKeyTypeValueFormArray(rows) {
|
|
612
|
+
const initial = rows ?? [{ key: '', type: 'text', value: '' }];
|
|
613
|
+
const groups = initial.map((r) => this.fb.group({ key: [r.key], type: [r.type], value: [r.value] }));
|
|
614
|
+
this.keyTypeValueFormArray = this.fb.array(groups);
|
|
615
|
+
}
|
|
616
|
+
buildVerificationFormArray(rows) {
|
|
617
|
+
const defaultVerification = this.getDefaultVerification();
|
|
618
|
+
const defaultDataType = this.getDefaultDataType();
|
|
619
|
+
const initial = rows ?? [{ jsonPath: '', verification: defaultVerification, dataType: defaultDataType, expectedValue: '' }];
|
|
620
|
+
const groups = initial.map((r) => this.fb.group({
|
|
621
|
+
jsonPath: [r.jsonPath],
|
|
622
|
+
verification: [r.verification],
|
|
623
|
+
dataType: [r.dataType],
|
|
624
|
+
expectedValue: [r.expectedValue],
|
|
625
|
+
}));
|
|
626
|
+
this.verificationFormArray = this.fb.array(groups);
|
|
627
|
+
}
|
|
628
|
+
buildStatusVerificationFormArray(rows) {
|
|
629
|
+
const defaultVerification = this.getDefaultStatusVerification();
|
|
630
|
+
const initial = rows ?? [{ verification: defaultVerification, expectedValue: '' }];
|
|
631
|
+
const groups = initial.map((r) => this.fb.group({
|
|
632
|
+
verification: [r.verification],
|
|
633
|
+
expectedValue: [r.expectedValue],
|
|
634
|
+
}));
|
|
635
|
+
this.statusVerificationFormArray = this.fb.array(groups);
|
|
636
|
+
}
|
|
637
|
+
buildHeadersFormArray(rows) {
|
|
638
|
+
const groups = rows.map((h) => this.fb.group({
|
|
639
|
+
name: [h.name],
|
|
640
|
+
value: [h.value],
|
|
641
|
+
}));
|
|
642
|
+
if (this.headersFormArray) {
|
|
643
|
+
this.headersFormArray.clear();
|
|
644
|
+
groups.forEach((g) => this.headersFormArray.push(g));
|
|
645
|
+
}
|
|
646
|
+
else {
|
|
647
|
+
this.headersFormArray = this.fb.array(groups);
|
|
648
|
+
}
|
|
649
|
+
}
|
|
650
|
+
/** Handler: show import cURL panel (called when user clicks "Import API cURL"). */
|
|
651
|
+
openImportCurlPanel() {
|
|
652
|
+
this.bodyView = 'import-curl';
|
|
653
|
+
this.cdr.detectChanges();
|
|
654
|
+
}
|
|
655
|
+
/** Handler: emit cURL value from control and close panel. Called when user clicks Import button. */
|
|
656
|
+
onImportCurlConfirm() {
|
|
657
|
+
const value = this.importCurlControl.value ?? '';
|
|
658
|
+
this.importCurl.emit(typeof value === 'string' ? value : '');
|
|
659
|
+
this.closeImportCurlPanel();
|
|
660
|
+
}
|
|
661
|
+
/** Handler: emit cancel and close panel. Called when user clicks Cancel in import cURL panel. */
|
|
662
|
+
onCancelImportCurl() {
|
|
663
|
+
this.importCurlCancel.emit();
|
|
664
|
+
this.closeImportCurlPanel();
|
|
665
|
+
}
|
|
666
|
+
/** Handler: show headers section (called from Cancel or after Import in import panel). */
|
|
667
|
+
closeImportCurlPanel() {
|
|
668
|
+
this.bodyView = 'headers';
|
|
669
|
+
this.cdr.detectChanges();
|
|
670
|
+
}
|
|
671
|
+
onSendRequest() {
|
|
672
|
+
const environment = this.selectedEnvironment != null ? String(this.selectedEnvironment) : '';
|
|
673
|
+
const method = this.selectedMethod != null ? String(this.selectedMethod) : '';
|
|
674
|
+
this.sendRequest.emit({
|
|
675
|
+
environment,
|
|
676
|
+
method,
|
|
677
|
+
url: this.url ?? '',
|
|
678
|
+
headers: this.headers,
|
|
679
|
+
});
|
|
680
|
+
}
|
|
681
|
+
onVariableNameChange() {
|
|
682
|
+
this.variableNameError = '';
|
|
683
|
+
}
|
|
684
|
+
onBack() {
|
|
685
|
+
if (this.currentStep > 1) {
|
|
686
|
+
this.setStep(this.currentStep - 1);
|
|
687
|
+
}
|
|
688
|
+
else {
|
|
689
|
+
this.back.emit();
|
|
690
|
+
}
|
|
691
|
+
}
|
|
692
|
+
onNext() {
|
|
693
|
+
if (this.currentStep === 2 && !this.variableName?.trim()) {
|
|
694
|
+
this.variableNameError = 'Variable Name is required.';
|
|
695
|
+
return;
|
|
696
|
+
}
|
|
697
|
+
this.variableNameError = '';
|
|
698
|
+
if (this.currentStep < this.stepLabels.length) {
|
|
699
|
+
this.setStep(this.currentStep + 1);
|
|
700
|
+
}
|
|
701
|
+
else {
|
|
702
|
+
this.next.emit();
|
|
703
|
+
}
|
|
704
|
+
}
|
|
705
|
+
setPayloadTab(value) {
|
|
706
|
+
this.activePayloadTab = value;
|
|
707
|
+
}
|
|
708
|
+
setResponseVerificationTab(value) {
|
|
709
|
+
this.activeResponseVerificationTab = value;
|
|
710
|
+
}
|
|
711
|
+
addHeader() {
|
|
712
|
+
this.headersFormArray.push(this.fb.group({ name: [''], value: [''] }));
|
|
713
|
+
}
|
|
714
|
+
removeHeader(index) {
|
|
715
|
+
this.headersFormArray.removeAt(index);
|
|
716
|
+
}
|
|
717
|
+
/** When user adds a custom header name via the dropdown "Add '…'" option, set the control value. */
|
|
718
|
+
onHeaderNameAddCustomValue(event, row) {
|
|
719
|
+
const control = row.get(event.key);
|
|
720
|
+
if (control && event.value != null) {
|
|
721
|
+
control.setValue(event.value.trim());
|
|
722
|
+
}
|
|
723
|
+
}
|
|
724
|
+
trackByHeader(index) {
|
|
725
|
+
return index;
|
|
726
|
+
}
|
|
727
|
+
addKeyValueRow() {
|
|
728
|
+
this.keyValueFormArray.push(this.fb.group({ key: [''], value: [''] }));
|
|
729
|
+
}
|
|
730
|
+
removeKeyValueRow(index) {
|
|
731
|
+
this.keyValueFormArray.removeAt(index);
|
|
732
|
+
}
|
|
733
|
+
trackByKeyValue(index) {
|
|
734
|
+
return index;
|
|
735
|
+
}
|
|
736
|
+
addKeyTypeValueRow() {
|
|
737
|
+
this.keyTypeValueFormArray.push(this.fb.group({ key: [''], type: ['text'], value: [''] }));
|
|
738
|
+
}
|
|
739
|
+
removeKeyTypeValueRow(index) {
|
|
740
|
+
this.keyTypeValueFormArray.removeAt(index);
|
|
741
|
+
}
|
|
742
|
+
trackByKeyTypeValue(index) {
|
|
743
|
+
return index;
|
|
744
|
+
}
|
|
745
|
+
addVerificationRow() {
|
|
746
|
+
this.verificationFormArray.push(this.fb.group({
|
|
747
|
+
jsonPath: [''],
|
|
748
|
+
verification: [this.getDefaultVerification()],
|
|
749
|
+
dataType: [this.getDefaultDataType()],
|
|
750
|
+
expectedValue: [''],
|
|
751
|
+
}));
|
|
752
|
+
}
|
|
753
|
+
removeVerificationRow(index) {
|
|
754
|
+
this.verificationFormArray.removeAt(index);
|
|
755
|
+
}
|
|
756
|
+
trackByVerification(index) {
|
|
757
|
+
return index;
|
|
758
|
+
}
|
|
759
|
+
/** When Data Type changes in Response Body verification, reset Expected Value for that row. */
|
|
760
|
+
onVerificationDataTypeChange(row, event) {
|
|
761
|
+
if (event.key === 'dataType') {
|
|
762
|
+
row.get('expectedValue')?.setValue('', { emitEvent: false });
|
|
763
|
+
}
|
|
764
|
+
}
|
|
765
|
+
addStatusVerificationRow() {
|
|
766
|
+
this.statusVerificationFormArray.push(this.fb.group({ verification: [this.getDefaultStatusVerification()], expectedValue: [''] }));
|
|
767
|
+
}
|
|
768
|
+
removeStatusVerificationRow(index) {
|
|
769
|
+
this.statusVerificationFormArray.removeAt(index);
|
|
770
|
+
}
|
|
771
|
+
trackByStatusVerification(index) {
|
|
772
|
+
return index;
|
|
773
|
+
}
|
|
774
|
+
/** Emit all entered details (environment, HTTP method, URL, headers, body, step2 variable, step3 verifications) when user clicks Create. */
|
|
775
|
+
onCreate() {
|
|
776
|
+
console.log('9999999');
|
|
777
|
+
try {
|
|
778
|
+
const payload = this.getCreatePayload();
|
|
779
|
+
console.log('9999999 payload', payload);
|
|
780
|
+
this.create.emit(payload);
|
|
781
|
+
}
|
|
782
|
+
catch (err) {
|
|
783
|
+
console.error('ApiEditStep getCreatePayload error:', err);
|
|
784
|
+
this.create.emit(this.getCreatePayloadFallback());
|
|
785
|
+
}
|
|
786
|
+
}
|
|
787
|
+
/** Minimal payload when getCreatePayload throws (so create always emits). */
|
|
788
|
+
getCreatePayloadFallback() {
|
|
789
|
+
return {
|
|
790
|
+
step1: {
|
|
791
|
+
environment: '',
|
|
792
|
+
method: '',
|
|
793
|
+
url: this.url ?? '',
|
|
794
|
+
headers: [],
|
|
795
|
+
activePayloadTab: this.activePayloadTab,
|
|
796
|
+
payloadType: this.payloadType,
|
|
797
|
+
payloadFormat: '',
|
|
798
|
+
payloadText: this.payloadText ?? '',
|
|
799
|
+
keyValueRows: [],
|
|
800
|
+
keyTypeValueRows: [],
|
|
801
|
+
},
|
|
802
|
+
step2: { variableName: this.variableName ?? '' },
|
|
803
|
+
step3: {
|
|
804
|
+
activeResponseVerificationTab: this.activeResponseVerificationTab,
|
|
805
|
+
responseBodyVerifications: [],
|
|
806
|
+
statusVerifications: [],
|
|
807
|
+
},
|
|
808
|
+
};
|
|
809
|
+
}
|
|
810
|
+
/** Normalize form control value to string (handles object or string). */
|
|
811
|
+
static getControlValue(control) {
|
|
812
|
+
const v = control?.value;
|
|
813
|
+
if (v == null || v === '')
|
|
814
|
+
return '';
|
|
815
|
+
return typeof v === 'object' ? (ApiEditStepComponent.getOptionValue(v) ?? '') : String(v);
|
|
816
|
+
}
|
|
817
|
+
/** Build full create payload: environment, HTTP method, URL, headers, body (step1), variable name (step2), verifications (step3). */
|
|
818
|
+
getCreatePayload() {
|
|
819
|
+
const envFromForm = ApiEditStepComponent.getControlValue(this.environmentForm?.get('environment'));
|
|
820
|
+
const environment = envFromForm !== '' ? envFromForm : (this.currentEnvironmentValue !== '' ? this.currentEnvironmentValue : this.selectedEnvironment);
|
|
821
|
+
const methodFromForm = ApiEditStepComponent.getControlValue(this.methodForm?.get('method'));
|
|
822
|
+
const method = methodFromForm !== '' ? methodFromForm : (this.currentMethodValue !== '' ? this.currentMethodValue : this.selectedMethod);
|
|
823
|
+
const format = this.payloadFormatForm?.get('format')?.value ?? '';
|
|
824
|
+
const keyValueFormArray = this.keyValueFormArray;
|
|
825
|
+
const keyValueRows = (keyValueFormArray?.controls ?? []).map((c) => ({
|
|
826
|
+
key: c.get('key')?.value ?? '',
|
|
827
|
+
value: c.get('value')?.value ?? '',
|
|
828
|
+
}));
|
|
829
|
+
const keyTypeValueFormArray = this.keyTypeValueFormArray;
|
|
830
|
+
const keyTypeValueRows = (keyTypeValueFormArray?.controls ?? []).map((c) => ({
|
|
831
|
+
key: c.get('key')?.value ?? '',
|
|
832
|
+
type: c.get('type')?.value ?? 'text',
|
|
833
|
+
value: c.get('value')?.value ?? '',
|
|
834
|
+
}));
|
|
835
|
+
const verificationFormArray = this.verificationFormArray;
|
|
836
|
+
const responseBodyVerifications = (verificationFormArray?.controls ?? []).map((c) => ({
|
|
837
|
+
jsonPath: c.get('jsonPath')?.value ?? '',
|
|
838
|
+
verification: c.get('verification')?.value ?? '',
|
|
839
|
+
dataType: c.get('dataType')?.value ?? '',
|
|
840
|
+
expectedValue: c.get('expectedValue')?.value ?? '',
|
|
841
|
+
}));
|
|
842
|
+
const statusVerificationFormArray = this.statusVerificationFormArray;
|
|
843
|
+
const statusVerifications = (statusVerificationFormArray?.controls ?? []).map((c) => ({
|
|
844
|
+
verification: c.get('verification')?.value ?? '',
|
|
845
|
+
expectedValue: c.get('expectedValue')?.value ?? '',
|
|
846
|
+
}));
|
|
847
|
+
return {
|
|
848
|
+
step1: {
|
|
849
|
+
environment,
|
|
850
|
+
method,
|
|
851
|
+
url: this.url ?? '',
|
|
852
|
+
headers: this.headers,
|
|
853
|
+
activePayloadTab: this.activePayloadTab,
|
|
854
|
+
payloadType: this.payloadType,
|
|
855
|
+
payloadFormat: format,
|
|
856
|
+
payloadText: this.payloadText ?? '',
|
|
857
|
+
keyValueRows,
|
|
858
|
+
keyTypeValueRows,
|
|
859
|
+
},
|
|
860
|
+
step2: {
|
|
861
|
+
variableName: this.variableName ?? '',
|
|
862
|
+
},
|
|
863
|
+
step3: {
|
|
864
|
+
activeResponseVerificationTab: this.activeResponseVerificationTab,
|
|
865
|
+
responseBodyVerifications,
|
|
866
|
+
statusVerifications,
|
|
867
|
+
},
|
|
868
|
+
};
|
|
869
|
+
}
|
|
870
|
+
/** Current header rows from form (for consumers that read headers) */
|
|
871
|
+
get headers() {
|
|
872
|
+
if (!this.headersFormArray)
|
|
873
|
+
return [];
|
|
874
|
+
return this.headersFormArray.controls.map((c) => ({
|
|
875
|
+
name: c.get('name')?.value ?? '',
|
|
876
|
+
type: 'string',
|
|
877
|
+
value: c.get('value')?.value ?? '',
|
|
878
|
+
}));
|
|
879
|
+
}
|
|
880
|
+
ngOnChanges(changes) {
|
|
881
|
+
if (changes['initialMethod']?.currentValue != null && this.methodForm) {
|
|
882
|
+
const v = changes['initialMethod'].currentValue;
|
|
883
|
+
this.currentMethodValue = typeof v === 'string' ? v : (ApiEditStepComponent.getOptionValue(v) ?? '');
|
|
884
|
+
this.methodForm.patchValue({ method: v });
|
|
885
|
+
}
|
|
886
|
+
if (changes['httpMethodOptions']) {
|
|
887
|
+
this.updateMethodSelectConfig();
|
|
888
|
+
if (this.methodForm && this.httpMethodOptions?.length) {
|
|
889
|
+
const current = this.methodForm.get('method')?.value;
|
|
890
|
+
const values = this.getMethodValues();
|
|
891
|
+
const valid = values.includes(current);
|
|
892
|
+
if (!valid) {
|
|
893
|
+
const firstValue = ApiEditStepComponent.getOptionValue(this.httpMethodOptions[0]);
|
|
894
|
+
if (firstValue != null) {
|
|
895
|
+
this.currentMethodValue = firstValue;
|
|
896
|
+
this.methodForm.patchValue({ method: firstValue });
|
|
897
|
+
}
|
|
898
|
+
}
|
|
899
|
+
}
|
|
900
|
+
}
|
|
901
|
+
if (changes['initialEnvironment']?.currentValue != null && this.environmentForm) {
|
|
902
|
+
const v = changes['initialEnvironment'].currentValue;
|
|
903
|
+
this.currentEnvironmentValue = typeof v === 'string' ? v : (ApiEditStepComponent.getOptionValue(v) ?? '');
|
|
904
|
+
this.environmentForm.patchValue({ environment: v });
|
|
905
|
+
}
|
|
906
|
+
if (changes['environmentOptions']) {
|
|
907
|
+
this.updateEnvironmentSelectConfig();
|
|
908
|
+
if (this.environmentForm && this.environmentOptions?.length) {
|
|
909
|
+
const current = this.environmentForm.get('environment')?.value;
|
|
910
|
+
const values = this.getEnvironmentValues();
|
|
911
|
+
const valid = values.includes(current);
|
|
912
|
+
if (!valid) {
|
|
913
|
+
const firstValue = ApiEditStepComponent.getOptionValue(this.environmentOptions[0]);
|
|
914
|
+
if (firstValue != null) {
|
|
915
|
+
this.currentEnvironmentValue = firstValue;
|
|
916
|
+
this.environmentForm.patchValue({ environment: firstValue });
|
|
917
|
+
}
|
|
918
|
+
}
|
|
919
|
+
}
|
|
920
|
+
}
|
|
921
|
+
if (changes['initialStep']) {
|
|
922
|
+
this.applyInitialStep(changes['initialStep'].currentValue);
|
|
923
|
+
}
|
|
924
|
+
if (changes['initialUrl']?.currentValue != null) {
|
|
925
|
+
this.url = changes['initialUrl'].currentValue;
|
|
926
|
+
}
|
|
927
|
+
if (changes['initialPayloadTab']?.currentValue != null) {
|
|
928
|
+
this.activePayloadTab = changes['initialPayloadTab'].currentValue;
|
|
929
|
+
}
|
|
930
|
+
if (changes['headerNameOptions']) {
|
|
931
|
+
this.updateHeaderNameSelectConfig();
|
|
932
|
+
}
|
|
933
|
+
if (changes['initialHeaders']?.currentValue != null) {
|
|
934
|
+
this.buildHeadersFormArray(changes['initialHeaders'].currentValue);
|
|
935
|
+
this.updateHeaderNameSelectConfig();
|
|
936
|
+
}
|
|
937
|
+
if (changes['initialResponsePreview']?.currentValue != null) {
|
|
938
|
+
this.responsePreview = changes['initialResponsePreview'].currentValue;
|
|
939
|
+
}
|
|
940
|
+
if (changes['initialFormat']?.currentValue != null && this.payloadFormatForm) {
|
|
941
|
+
this.payloadFormatForm.patchValue({ format: changes['initialFormat'].currentValue });
|
|
942
|
+
}
|
|
943
|
+
if (changes['formatOptions']) {
|
|
944
|
+
this.updatePayloadFormatSelectConfig();
|
|
945
|
+
if (this.payloadFormatForm) {
|
|
946
|
+
const current = this.payloadFormatForm.get('format')?.value;
|
|
947
|
+
const values = this.getFormatValues();
|
|
948
|
+
const valid = values.includes(current);
|
|
949
|
+
if (!valid && values.length > 0) {
|
|
950
|
+
const firstValue = values[0];
|
|
951
|
+
this.payloadFormatForm.patchValue({ format: firstValue });
|
|
952
|
+
}
|
|
953
|
+
}
|
|
954
|
+
}
|
|
955
|
+
if (changes['verificationOptions']) {
|
|
956
|
+
this.updateVerificationSelectConfig();
|
|
957
|
+
const values = this.getVerificationValues();
|
|
958
|
+
const defaultVerification = this.getDefaultVerification();
|
|
959
|
+
if (this.verificationFormArray && values.length > 0) {
|
|
960
|
+
this.verificationFormArray.controls.forEach((control) => {
|
|
961
|
+
const current = control.get('verification')?.value;
|
|
962
|
+
if (current && !values.includes(current)) {
|
|
963
|
+
control.patchValue({ verification: defaultVerification });
|
|
964
|
+
}
|
|
965
|
+
});
|
|
966
|
+
}
|
|
967
|
+
}
|
|
968
|
+
if (changes['verificationDataTypeOptions']) {
|
|
969
|
+
this.updateVerificationDataTypeSelectConfig();
|
|
970
|
+
const values = this.getDataTypeValues();
|
|
971
|
+
const defaultDataType = this.getDefaultDataType();
|
|
972
|
+
if (this.verificationFormArray && values.length > 0) {
|
|
973
|
+
this.verificationFormArray.controls.forEach((control) => {
|
|
974
|
+
const current = control.get('dataType')?.value;
|
|
975
|
+
if (current && !values.includes(current)) {
|
|
976
|
+
control.patchValue({ dataType: defaultDataType });
|
|
977
|
+
}
|
|
978
|
+
});
|
|
979
|
+
}
|
|
980
|
+
}
|
|
981
|
+
if (changes['statusVerificationOptions']) {
|
|
982
|
+
this.updateStatusVerificationSelectConfig();
|
|
983
|
+
const values = this.getStatusVerificationValues();
|
|
984
|
+
const defaultVerification = this.getDefaultStatusVerification();
|
|
985
|
+
if (this.statusVerificationFormArray && values.length > 0) {
|
|
986
|
+
this.statusVerificationFormArray.controls.forEach((control) => {
|
|
987
|
+
const current = control.get('verification')?.value;
|
|
988
|
+
if (current && !values.includes(current)) {
|
|
989
|
+
control.patchValue({ verification: defaultVerification });
|
|
990
|
+
}
|
|
991
|
+
});
|
|
992
|
+
}
|
|
993
|
+
}
|
|
994
|
+
}
|
|
995
|
+
}
|
|
996
|
+
ApiEditStepComponent.DEFAULT_FORMAT_OPTIONS = [
|
|
997
|
+
{ id: 'json', value: 'json', name: 'JSON', label: 'JSON' },
|
|
998
|
+
{ id: 'xml', value: 'xml', name: 'XML', label: 'XML' },
|
|
999
|
+
{ id: 'html', value: 'html', name: 'HTML', label: 'HTML' },
|
|
1000
|
+
{ id: 'text', value: 'text', name: 'Text', label: 'Text' },
|
|
1001
|
+
];
|
|
1002
|
+
ApiEditStepComponent.DEFAULT_HEADER_NAME_OPTIONS = [
|
|
1003
|
+
'Accept',
|
|
1004
|
+
'Accept-Charset',
|
|
1005
|
+
'Accept-Encoding',
|
|
1006
|
+
'Accept-Language',
|
|
1007
|
+
'Access-Control-Request-Headers',
|
|
1008
|
+
'Access-Control-Request-Method',
|
|
1009
|
+
'Authorization',
|
|
1010
|
+
'Cache-Control',
|
|
1011
|
+
'Content-MD5',
|
|
1012
|
+
'Content-Length',
|
|
1013
|
+
'Content-Transfer-Encoding',
|
|
1014
|
+
'Content-Type',
|
|
1015
|
+
'Cookie',
|
|
1016
|
+
'Date',
|
|
1017
|
+
'Expect',
|
|
1018
|
+
'From',
|
|
1019
|
+
'Host',
|
|
1020
|
+
'If-Match',
|
|
1021
|
+
'If-Modified-Since',
|
|
1022
|
+
'If-None-Match',
|
|
1023
|
+
'If-Unmodified-Since',
|
|
1024
|
+
'If-Range',
|
|
1025
|
+
'Keep-Alive',
|
|
1026
|
+
'Max-Forwards',
|
|
1027
|
+
'Origin',
|
|
1028
|
+
'Pragma',
|
|
1029
|
+
'Proxy-Authorization',
|
|
1030
|
+
'Range',
|
|
1031
|
+
'Referer',
|
|
1032
|
+
'TE',
|
|
1033
|
+
'Trailer',
|
|
1034
|
+
'Transfer-Encoding',
|
|
1035
|
+
'Upgrade',
|
|
1036
|
+
'User-Agent',
|
|
1037
|
+
'Via',
|
|
1038
|
+
'Warning',
|
|
1039
|
+
'X-Requested-With',
|
|
1040
|
+
'X-Do-Not-Track',
|
|
1041
|
+
'DNT',
|
|
1042
|
+
'x-api-key',
|
|
1043
|
+
'Connection',
|
|
1044
|
+
];
|
|
1045
|
+
ApiEditStepComponent.DEFAULT_VERIFICATION_OPTIONS = [
|
|
1046
|
+
{ id: 'equals', value: 'equals', name: 'Equals', label: 'Equals' },
|
|
1047
|
+
{ id: 'not-equal', value: 'not-equal', name: 'Not equal', label: 'Not equal' },
|
|
1048
|
+
{ id: 'less-than', value: 'less-than', name: 'Less than', label: 'Less than' },
|
|
1049
|
+
{ id: 'greater-than', value: 'greater-than', name: 'Greater than', label: 'Greater than' },
|
|
1050
|
+
{ id: 'is-null', value: 'is-null', name: 'Is null', label: 'Is null' },
|
|
1051
|
+
{ id: 'is-not-null', value: 'is-not-null', name: 'Is not null', label: 'Is not null' },
|
|
1052
|
+
];
|
|
1053
|
+
ApiEditStepComponent.DEFAULT_DATA_TYPE_OPTIONS = [
|
|
1054
|
+
{ id: 'string', value: 'string', name: 'String', label: 'String' },
|
|
1055
|
+
{ id: 'number', value: 'number', name: 'Number', label: 'Number' },
|
|
1056
|
+
{ id: 'boolean', value: 'boolean', name: 'Boolean', label: 'Boolean' },
|
|
1057
|
+
{ id: 'array', value: 'array', name: 'Array', label: 'Array' },
|
|
1058
|
+
{ id: 'object', value: 'object', name: 'Object', label: 'Object' },
|
|
1059
|
+
];
|
|
1060
|
+
ApiEditStepComponent.DEFAULT_STATUS_VERIFICATION_OPTIONS = [
|
|
1061
|
+
{ id: 'equals', value: 'equals', name: 'Equals', label: 'Equals' },
|
|
1062
|
+
{ id: 'not-equal', value: 'not-equal', name: 'Not equal', label: 'Not equal' },
|
|
1063
|
+
{ id: 'less-than', value: 'less-than', name: 'Less than', label: 'Less than' },
|
|
1064
|
+
{ id: 'greater-than', value: 'greater-than', name: 'Greater than', label: 'Greater than' },
|
|
1065
|
+
];
|
|
1066
|
+
ApiEditStepComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ApiEditStepComponent, deps: [{ token: i1.FormBuilder }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
1067
|
+
ApiEditStepComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: ApiEditStepComponent, selector: "cqa-api-edit-step", inputs: { initialMethod: "initialMethod", initialEnvironment: "initialEnvironment", initialStep: "initialStep", initialUrl: "initialUrl", initialPayloadTab: "initialPayloadTab", initialHeaders: "initialHeaders", initialResponsePreview: "initialResponsePreview", httpMethodOptions: "httpMethodOptions", environmentOptions: "environmentOptions", formatOptions: "formatOptions", initialFormat: "initialFormat", headerNameOptions: "headerNameOptions", verificationOptions: "verificationOptions", verificationDataTypeOptions: "verificationDataTypeOptions", statusVerificationOptions: "statusVerificationOptions" }, outputs: { importCurl: "importCurl", importCurlCancel: "importCurlCancel", sendRequest: "sendRequest", back: "back", next: "next", create: "create", headersChange: "headersChange" }, host: { classAttribute: "cqa-ui-root" }, viewQueries: [{ propertyName: "payloadEditorWithLinesRef", first: true, predicate: ["payloadEditorWithLinesRef"], descendants: true }, { propertyName: "payloadTextareaRef", first: true, predicate: ["payloadTextareaRef"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-api-edit-step-container\">\n <!-- Title -->\n <h2\n class=\"cqa-api-edit-step-title cqa-font-inter cqa-text-[12px] cqa-font-semibold cqa-leading-[100%] cqa-tracking-normal cqa-text-[#000000]\">\n Create API Test Step</h2>\n\n <!-- Step indicator: all three steps visible, active step highlighted (Postman-style) -->\n <div class=\"cqa-api-edit-step-indicator\">\n <ng-container *ngFor=\"let step of stepLabels; let i = index\">\n <cqa-button type=\"button\" variant=\"text\"\n customClass=\"cqa-api-edit-step-indicator-item cqa-api-edit-step-indicator-item--clickable\"\n [attr.aria-label]=\"'Step ' + step.index + ': ' + step.label\" [attr.aria-pressed]=\"step.index === currentStep\"\n (clicked)=\"setStep(step.index)\">\n <span class=\"cqa-api-edit-step-indicator-circle\"\n [class.cqa-api-edit-step-indicator-circle--active]=\"step.index === currentStep\">\n {{ step.index }}\n </span>\n <span class=\"cqa-api-edit-step-indicator-label\"\n [class.cqa-api-edit-step-indicator-label--active]=\"step.index === currentStep\">\n {{ step.label }}\n </span>\n <div *ngIf=\"i < stepLabels.length - 1\" class=\"cqa-api-edit-step-indicator-line\" aria-hidden=\"true\"></div>\n </cqa-button>\n </ng-container>\n </div>\n\n <!-- Step content viewport: smooth slide between steps -->\n <div class=\"cqa-api-edit-step-viewport\">\n <div class=\"cqa-api-edit-step-strip\" [style.transform]=\"'translateX(' + stripTranslatePercent + '%)'\">\n <!-- Step 1: Environment, request, body, response -->\n <div class=\"cqa-api-edit-step-panel\">\n <!-- Environment row: new line, select aligned right -->\n <div class=\"cqa-api-edit-step-environment-row\">\n <cqa-dynamic-select *ngIf=\"environmentForm\" [form]=\"environmentForm\" [config]=\"environmentSelectConfig\"\n class=\"cqa-api-edit-step-environment-select\" aria-label=\"Environment\"\n (selectionChange)=\"onEnvironmentSelectionChange($event)\">\n </cqa-dynamic-select>\n </div>\n\n <!-- Request row: method, URL, buttons -->\n <div class=\"cqa-api-edit-step-request-row\">\n <cqa-dynamic-select *ngIf=\"methodForm\" [form]=\"methodForm\" [config]=\"methodSelectConfig\"\n class=\"cqa-api-edit-step-method-select\" aria-label=\"HTTP method\"\n (selectionChange)=\"onMethodSelectionChange($event)\">\n </cqa-dynamic-select>\n <div class=\"cqa-api-edit-step-url-wrap\">\n <cqa-custom-input [(value)]=\"url\" [label]=\"''\" placeholder=\"\" [fullWidth]=\"true\" size=\"md\">\n </cqa-custom-input>\n </div>\n <div class=\"cqa-api-edit-step-import-curl-trigger\" style=\"display: contents\" (click)=\"openImportCurlPanel()\">\n <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Import API cURL\"\n customClass=\"cqa-api-edit-step-btn-outline cqa-font-inter cqa-text-[14px] cqa-font-semibold cqa-leading-[100%] cqa-tracking-normal cqa-text-[#414146]\">\n </cqa-button>\n </div>\n <cqa-button variant=\"filled\" btnSize=\"lg\" text=\"Send Request\" (clicked)=\"onSendRequest()\"\n customClass=\"cqa-api-edit-step-btn-primary cqa-font-inter cqa-text-[14px] cqa-font-semibold cqa-leading-[100%] cqa-tracking-normal cqa-text-[#FBFCFF]\">\n </cqa-button>\n </div>\n\n\n\n <!-- Body content: header section (default) OR import cURL section -->\n <ng-container *ngIf=\"bodyView === 'import-curl'\">\n <div class=\"cqa-api-edit-step-import-curl-panel\">\n <div class=\"cqa-api-edit-step-import-curl-header\">\n <h3 class=\"cqa-api-edit-step-import-curl-title\">Import API cURL</h3>\n </div>\n <div class=\"cqa-api-edit-step-import-curl-separator\"></div>\n <div class=\"cqa-api-edit-step-import-curl-body\">\n <cqa-custom-textarea [value]=\"importCurlControl.value\"\n (valueChange)=\"importCurlControl.setValue($event)\"\n placeholder=\"Paste your cURL command here (e.g., curl -X POST https://api.example.com/users)\"\n [fullWidth]=\"true\" [rows]=\"8\" resize=\"vertical\" size=\"md\"\n class=\"cqa-api-edit-step-import-curl-textarea\">\n </cqa-custom-textarea>\n </div>\n <div class=\"cqa-api-edit-step-import-curl-footer\">\n <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Cancel\"\n customClass=\"cqa-api-edit-step-import-curl-btn-cancel\" (clicked)=\"onCancelImportCurl()\"></cqa-button>\n <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"Import\"\n customClass=\"cqa-api-edit-step-import-curl-btn-import\" (clicked)=\"onImportCurlConfirm()\"></cqa-button>\n </div>\n </div>\n </ng-container>\n\n <ng-container *ngIf=\"bodyView === 'headers'\">\n <div class=\"cqa-api-edit-step-tabs-wrapper\">\n <div class=\"cqa-api-edit-step-tabs\">\n <cqa-button *ngFor=\"let tab of payloadTabs\" type=\"button\" variant=\"text\" [text]=\"tab.label\"\n [customClass]=\"'cqa-api-edit-step-tab' + (activePayloadTab === tab.value ? ' cqa-api-edit-step-tab--active' : '') + (isMethodWithoutBody && tab.value !== 'headers' ? ' cqa-api-edit-step-tab--disabled' : '')\"\n [disabled]=\"isMethodWithoutBody && tab.value !== 'headers'\"\n (clicked)=\"setPayloadTab(tab.value)\">\n </cqa-button>\n </div>\n\n <!-- Payload content (Headers) -->\n <div *ngIf=\"activePayloadTab === 'headers'\" class=\"cqa-api-edit-step-payload\">\n <div class=\"cqa-api-edit-step-headers-grid\">\n <span class=\"cqa-api-edit-step-headers-label\">Header Name</span>\n <span class=\"cqa-api-edit-step-headers-label\">Header Value</span>\n <span class=\"cqa-api-edit-step-headers-label cqa-api-edit-step-headers-label--empty\"\n aria-hidden=\"true\"></span>\n </div>\n <div *ngFor=\"let row of headerRows; let i = index; trackBy: trackByHeader\" [formGroup]=\"row\"\n class=\"cqa-api-edit-step-header-row\">\n <cqa-dynamic-select [form]=\"row\" [config]=\"headerNameSelectConfig\"\n (addCustomValue)=\"onHeaderNameAddCustomValue($event, row)\"\n class=\"cqa-api-edit-step-header-type-select cqa-w-full\">\n </cqa-dynamic-select>\n <cqa-custom-input [value]=\"row.get('value')?.value ?? ''\"\n (valueChange)=\"row.get('value')?.setValue($event)\" placeholder=\"Value\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-header-value-input\" ariaLabel=\"Header value\">\n </cqa-custom-input>\n <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-header-delete'\"\n [tooltip]=\"'Remove header'\" (clicked)=\"removeHeader(i)\">\n <svg class=\"cqa-api-edit-step-header-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#F9BFBF\" />\n </svg>\n </cqa-button>\n </div>\n <div class=\"cqa-api-edit-step-add-header-wrap\">\n <cqa-button type=\"button\" variant=\"text\" text=\"+ Add Header\"\n [customClass]=\"'cqa-api-edit-step-add-header-link'\" (clicked)=\"addHeader()\">\n </cqa-button>\n </div>\n </div>\n\n <!-- Payload content (Body only): type, format, text area, Back/Next -->\n <div *ngIf=\"activePayloadTab === 'body'\"\n class=\"cqa-api-edit-step-payload cqa-api-edit-step-payload-editor\">\n <div class=\"cqa-api-edit-step-payload-type-row\">\n <span class=\"cqa-api-edit-step-payload-type-label\">Type</span>\n <div class=\"cqa-api-edit-step-payload-type-radios\">\n <cqa-segment-control [value]=\"payloadType\" [segments]=\"payloadTypeSegments\"\n (valueChange)=\"onPayloadTypeChange($event)\"\n class=\"cqa-api-edit-step-payload-type-segment\">\n </cqa-segment-control>\n </div>\n <div *ngIf=\"payloadType === 'raw'\" class=\"cqa-api-edit-step-payload-format-wrap\">\n <span class=\"cqa-api-edit-step-payload-format-label\">Format:</span>\n <cqa-dynamic-select *ngIf=\"payloadFormatForm\" [form]=\"payloadFormatForm\"\n [config]=\"payloadFormatSelectConfig\" class=\"cqa-api-edit-step-payload-format-select\"\n aria-label=\"Format\">\n </cqa-dynamic-select>\n </div>\n </div>\n <!-- Raw: text area with line numbers (Postman-style payload body) -->\n <div *ngIf=\"payloadType === 'raw'\" class=\"cqa-api-edit-step-payload-body\" #payloadEditorWithLinesRef>\n <div class=\"cqa-api-edit-step-payload-editor-with-lines\">\n <div class=\"cqa-api-edit-step-payload-line-numbers\" aria-hidden=\"true\">\n <span *ngFor=\"let n of payloadLineNumbers\" class=\"cqa-api-edit-step-payload-line-num\">\n <span *ngIf=\"payloadJsonError && payloadJsonError.line === n\"\n class=\"cqa-api-edit-step-payload-line-error-icon\" [title]=\"payloadJsonErrorTooltip\"\n aria-label=\"Parse error on this line\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" aria-hidden=\"true\">\n <circle cx=\"7\" cy=\"7\" r=\"6\" fill=\"#EF4444\"/>\n <path d=\"M4 4l6 6M10 4l-6 6\" stroke=\"white\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\n </svg>\n </span>\n <span class=\"cqa-api-edit-step-payload-line-num-text\">{{ n }}</span>\n </span>\n </div>\n <div class=\"cqa-api-edit-step-payload-textarea-cell\">\n <div class=\"cqa-api-edit-step-payload-textarea cqa-w-full cqa-flex cqa-flex-col cqa-min-h-0\"\n [ngClass]=\"{'cqa-api-edit-step-payload-textarea--error': payloadJsonError}\">\n <textarea #payloadTextareaRef\n class=\"cqa-api-edit-step-payload-textarea-input cqa-w-full cqa-resize-y cqa-outline-none cqa-box-border\"\n [ngClass]=\"{'cqa-api-edit-step-payload-textarea-input--error': payloadJsonError}\"\n [value]=\"payloadText\"\n [attr.rows]=\"10\"\n placeholder=\"\"\n [attr.aria-label]=\"'Payload'\"\n [attr.aria-invalid]=\"!!payloadJsonError\"\n (input)=\"onPayloadInput($event)\"\n (keydown)=\"onPayloadKeydown($event)\">\n </textarea>\n </div>\n </div>\n </div>\n <p *ngIf=\"payloadJsonError\" class=\"cqa-api-edit-step-payload-json-error-msg\">\n Invalid JSON format. Please check your syntax.\n </p>\n </div>\n\n <!-- x-www-form-urlencoded: Key\u2013Value rows, add/remove dynamically -->\n <div *ngIf=\"payloadType === 'x-www-form-urlencoded'\" class=\"cqa-api-edit-step-key-value\">\n <div class=\"cqa-api-edit-step-key-value-grid cqa-api-edit-step-key-value-header\">\n <span class=\"cqa-api-edit-step-key-value-label\">Key</span>\n <span class=\"cqa-api-edit-step-key-value-label\">Value</span>\n <span class=\"cqa-api-edit-step-key-value-label cqa-api-edit-step-key-value-label--empty\"\n aria-hidden=\"true\"></span>\n </div>\n <div *ngFor=\"let row of keyValueRows; let i = index; trackBy: trackByKeyValue\" [formGroup]=\"row\"\n class=\"cqa-api-edit-step-key-value-row\">\n <cqa-custom-input [value]=\"row.get('key')?.value ?? ''\"\n (valueChange)=\"row.get('key')?.setValue($event)\" placeholder=\"Key\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-key-value-input\" ariaLabel=\"Key\">\n </cqa-custom-input>\n <cqa-custom-input [value]=\"row.get('value')?.value ?? ''\"\n (valueChange)=\"row.get('value')?.setValue($event)\" placeholder=\"Value\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-key-value-input\" ariaLabel=\"Value\">\n </cqa-custom-input>\n <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-key-value-delete'\"\n [tooltip]=\"'Remove row'\" (clicked)=\"removeKeyValueRow(i)\">\n <svg class=\"cqa-api-edit-step-key-value-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#F9BFBF\" />\n </svg>\n </cqa-button>\n </div>\n <div class=\"cqa-api-edit-step-key-value-add-wrap\">\n <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"+ Add Row\"\n customClass=\"cqa-api-edit-step-key-value-add-btn\" (clicked)=\"addKeyValueRow()\">\n </cqa-button>\n </div>\n </div>\n\n <!-- Form Data: Key\u2013Type\u2013Value rows; Type is a dropdown (Text/File), add/remove dynamically -->\n <div *ngIf=\"payloadType === 'form-data'\" class=\"cqa-api-edit-step-key-type-value\">\n <div class=\"cqa-api-edit-step-key-type-value-header\">\n <span class=\"cqa-api-edit-step-key-type-value-label\">Key</span>\n <span class=\"cqa-api-edit-step-key-type-value-label\">Type</span>\n <span class=\"cqa-api-edit-step-key-type-value-label\">Value</span>\n <span class=\"cqa-api-edit-step-key-type-value-label cqa-api-edit-step-key-type-value-label--empty\"\n aria-hidden=\"true\"></span>\n </div>\n <div *ngFor=\"let row of keyTypeValueRows; let i = index; trackBy: trackByKeyTypeValue\" [formGroup]=\"row\"\n class=\"cqa-api-edit-step-key-type-value-row\">\n <cqa-custom-input [value]=\"row.get('key')?.value ?? ''\"\n (valueChange)=\"row.get('key')?.setValue($event)\" placeholder=\"Key\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-key-type-value-input\" ariaLabel=\"Key\">\n </cqa-custom-input>\n <cqa-dynamic-select [form]=\"row\" [config]=\"keyTypeValueTypeSelectConfig\"\n class=\"cqa-api-edit-step-key-type-value-type-select cqa-w-full\">\n </cqa-dynamic-select>\n <cqa-custom-input [value]=\"row.get('value')?.value ?? ''\"\n (valueChange)=\"row.get('value')?.setValue($event)\" placeholder=\"Value\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-key-type-value-input\" ariaLabel=\"Value\">\n </cqa-custom-input>\n <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-key-type-value-delete'\"\n [tooltip]=\"'Remove row'\" (clicked)=\"removeKeyTypeValueRow(i)\">\n <svg class=\"cqa-api-edit-step-key-type-value-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#F9BFBF\" />\n </svg>\n </cqa-button>\n </div>\n <div class=\"cqa-api-edit-step-key-type-value-add-wrap\">\n <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"+ Add Row\"\n customClass=\"cqa-api-edit-step-key-type-value-add-btn\" (clicked)=\"addKeyTypeValueRow()\">\n </cqa-button>\n </div>\n </div>\n </div>\n </div>\n </ng-container>\n\n </div>\n <!-- Step 2: Variable Name: input, validation, Back / Next -->\n <div class=\"cqa-api-edit-step-panel\">\n <div class=\"cqa-api-edit-step-variable-section\">\n <div class=\"cqa-api-edit-step-variable-input-wrap\">\n <cqa-custom-input [(value)]=\"variableName\" [label]=\"''\" placeholder=\"Variable Name\" [fullWidth]=\"true\"\n size=\"md\" (valueChange)=\"onVariableNameChange()\" aria-label=\"Variable Name\">\n </cqa-custom-input>\n </div>\n <p *ngIf=\"variableNameError\" class=\"cqa-api-edit-step-variable-error\" role=\"alert\">{{ variableNameError }}</p>\n </div>\n </div>\n <!-- Step 3: Response Body / Status tabs -->\n <div class=\"cqa-api-edit-step-panel\">\n <div class=\"cqa-api-edit-step-tabs-wrapper\">\n <div class=\"cqa-api-edit-step-tabs\">\n <cqa-button *ngFor=\"let tab of responseVerificationTabs\" type=\"button\" variant=\"text\" [text]=\"tab.label\"\n [customClass]=\"'cqa-api-edit-step-tab' + (activeResponseVerificationTab === tab.value ? ' cqa-api-edit-step-tab--active' : '')\"\n (clicked)=\"setResponseVerificationTab(tab.value)\">\n </cqa-button>\n </div>\n\n <!-- Response Body tab content: verification grid -->\n <div *ngIf=\"activeResponseVerificationTab === 'response-body'\" class=\"cqa-api-edit-step-step3-content\">\n <div class=\"cqa-api-edit-step-verification-header-row\">\n <span></span>\n <cqa-button type=\"button\" variant=\"text\" text=\"Add Verification\"\n customClass=\"cqa-api-edit-step-verification-add-link\" (clicked)=\"addVerificationRow()\">\n </cqa-button>\n </div>\n <div class=\"cqa-api-edit-step-verification\">\n <div class=\"cqa-api-edit-step-verification-grid cqa-api-edit-step-verification-grid-header\">\n <span class=\"cqa-api-edit-step-verification-label\">S.no</span>\n <span class=\"cqa-api-edit-step-verification-label\">JSON Path</span>\n <span class=\"cqa-api-edit-step-verification-label\">Verification</span>\n <span class=\"cqa-api-edit-step-verification-label\">Data Type</span>\n <span class=\"cqa-api-edit-step-verification-label\">Expected Value</span>\n <span class=\"cqa-api-edit-step-verification-label cqa-api-edit-step-verification-label--empty\"\n aria-hidden=\"true\"></span>\n </div>\n <div *ngFor=\"let row of verificationRows; let i = index; trackBy: trackByVerification\" [formGroup]=\"row\"\n class=\"cqa-api-edit-step-verification-row\">\n <span class=\"cqa-api-edit-step-verification-sno\">{{ i + 1 }}</span>\n <cqa-custom-input [value]=\"row.get('jsonPath')?.value ?? ''\"\n (valueChange)=\"row.get('jsonPath')?.setValue($event)\" placeholder=\"Json Path\" [fullWidth]=\"true\"\n size=\"sm\" class=\"cqa-api-edit-step-verification-input\" ariaLabel=\"JSON Path\">\n </cqa-custom-input>\n <cqa-dynamic-select [form]=\"row\" [config]=\"verificationSelectConfig\"\n class=\"cqa-api-edit-step-verification-select cqa-w-full\">\n </cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"row\" [config]=\"verificationDataTypeSelectConfig\"\n class=\"cqa-api-edit-step-verification-select cqa-w-full\"\n (selectionChange)=\"onVerificationDataTypeChange(row, $event)\">\n </cqa-dynamic-select>\n <!-- Expected Value: text for String/Array/Object, number for Number, dropdown for Boolean -->\n <ng-container [ngSwitch]=\"row.get('dataType')?.value\">\n <cqa-dynamic-select *ngSwitchCase=\"'boolean'\" [form]=\"row\"\n [config]=\"verificationExpectedValueBooleanSelectConfig\"\n class=\"cqa-api-edit-step-verification-select cqa-api-edit-step-verification-expected-select cqa-w-full\">\n </cqa-dynamic-select>\n <cqa-custom-input *ngSwitchCase=\"'number'\" [value]=\"row.get('expectedValue')?.value ?? ''\"\n (valueChange)=\"row.get('expectedValue')?.setValue($event)\" placeholder=\"Expected Value (number)\"\n type=\"number\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-verification-input cqa-api-edit-step-verification-expected-number\"\n ariaLabel=\"Expected Value\">\n </cqa-custom-input>\n <cqa-custom-input *ngSwitchDefault [value]=\"row.get('expectedValue')?.value ?? ''\"\n (valueChange)=\"row.get('expectedValue')?.setValue($event)\"\n placeholder=\"Expected Value in String\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-verification-input\" ariaLabel=\"Expected Value\">\n </cqa-custom-input>\n </ng-container>\n <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-verification-delete'\"\n [tooltip]=\"'Remove row'\" (clicked)=\"removeVerificationRow(i)\">\n <svg class=\"cqa-api-edit-step-verification-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#F9BFBF\" />\n </svg>\n </cqa-button>\n </div>\n </div>\n </div>\n\n <!-- Status tab content: S.no, Verification dropdown, Expected Value, add/remove rows -->\n <div *ngIf=\"activeResponseVerificationTab === 'status'\" class=\"cqa-api-edit-step-step3-content\">\n <div class=\"cqa-api-edit-step-verification-header-row\">\n <span></span>\n <cqa-button type=\"button\" variant=\"text\" text=\"Add Verification\"\n customClass=\"cqa-api-edit-step-verification-add-link\" (clicked)=\"addStatusVerificationRow()\">\n </cqa-button>\n </div>\n <div class=\"cqa-api-edit-step-status-verification\">\n <div class=\"cqa-api-edit-step-status-verification-header\">\n <span class=\"cqa-api-edit-step-verification-label\">S.no</span>\n <span class=\"cqa-api-edit-step-verification-label\">Verification</span>\n <span class=\"cqa-api-edit-step-verification-label\">Expected Value</span>\n <span class=\"cqa-api-edit-step-verification-label cqa-api-edit-step-verification-label--empty\"\n aria-hidden=\"true\"></span>\n </div>\n <div *ngFor=\"let row of statusVerificationRows; let i = index; trackBy: trackByStatusVerification\"\n [formGroup]=\"row\" class=\"cqa-api-edit-step-status-verification-row\">\n <span class=\"cqa-api-edit-step-verification-sno\">{{ i + 1 }}</span>\n <cqa-dynamic-select [form]=\"row\" [config]=\"statusVerificationSelectConfig\"\n class=\"cqa-api-edit-step-status-verification-select cqa-w-full\">\n </cqa-dynamic-select>\n <cqa-custom-input [value]=\"row.get('expectedValue')?.value ?? ''\"\n (valueChange)=\"row.get('expectedValue')?.setValue($event)\" placeholder=\"Expected Value\"\n type=\"number\" [fullWidth]=\"true\" size=\"sm\" class=\"cqa-api-edit-step-verification-input\"\n ariaLabel=\"Expected Value\">\n </cqa-custom-input>\n <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-verification-delete'\"\n [tooltip]=\"'Remove row'\" (clicked)=\"removeStatusVerificationRow(i)\">\n <svg class=\"cqa-api-edit-step-verification-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#F9BFBF\" />\n </svg>\n </cqa-button>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Response Preview -->\n <div class=\"cqa-api-edit-step-response\">\n <h3 class=\"cqa-api-edit-step-response-title\">Response Preview</h3>\n <pre class=\"cqa-api-edit-step-response-content\">{{ responsePreview }}</pre>\n </div>\n\n <!-- Step actions: one row, full width. Step 1: Cancel + Next; Step 2: Back + Next; Step 3: Back + Create -->\n <div class=\"cqa-api-edit-step-actions\">\n <ng-container [ngSwitch]=\"currentStep\">\n <ng-container *ngSwitchCase=\"1\">\n <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Cancel\"\n customClass=\"cqa-api-edit-step-actions-btn-cancel\" (clicked)=\"onBack()\">\n </cqa-button>\n <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"Next\"\n customClass=\"cqa-api-edit-step-actions-btn-next\" (clicked)=\"onNext()\">\n </cqa-button>\n </ng-container>\n <ng-container *ngSwitchCase=\"2\">\n <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Back\"\n customClass=\"cqa-api-edit-step-actions-btn-back\" (clicked)=\"onBack()\">\n </cqa-button>\n <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"Next\"\n customClass=\"cqa-api-edit-step-actions-btn-next\" (clicked)=\"onNext()\">\n </cqa-button>\n </ng-container>\n <ng-container *ngSwitchCase=\"3\">\n <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Back\"\n customClass=\"cqa-api-edit-step-actions-btn-back\" (clicked)=\"onBack()\">\n </cqa-button>\n <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"Create\"\n customClass=\"cqa-api-edit-step-actions-btn-create\" (clicked)=\"onCreate()\">\n </cqa-button>\n </ng-container>\n </ng-container>\n </div>\n</div>", components: [{ type: i2.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }, { type: i3.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }, { type: i4.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: i5.CustomTextareaComponent, selector: "cqa-custom-textarea", inputs: ["label", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "rows", "cols", "resize", "textareaInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused"] }, { type: i6.SegmentControlComponent, selector: "cqa-segment-control", inputs: ["segments", "value", "disabled", "containerBgColor"], outputs: ["valueChange"] }], directives: [{ type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { type: i7.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i7.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { type: i7.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { type: i7.NgSwitchDefault, selector: "[ngSwitchDefault]" }], changeDetection: i0.ChangeDetectionStrategy.Default });
|
|
1068
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ApiEditStepComponent, decorators: [{
|
|
1069
|
+
type: Component,
|
|
1070
|
+
args: [{ selector: 'cqa-api-edit-step', host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.Default, template: "<div class=\"cqa-api-edit-step-container\">\n <!-- Title -->\n <h2\n class=\"cqa-api-edit-step-title cqa-font-inter cqa-text-[12px] cqa-font-semibold cqa-leading-[100%] cqa-tracking-normal cqa-text-[#000000]\">\n Create API Test Step</h2>\n\n <!-- Step indicator: all three steps visible, active step highlighted (Postman-style) -->\n <div class=\"cqa-api-edit-step-indicator\">\n <ng-container *ngFor=\"let step of stepLabels; let i = index\">\n <cqa-button type=\"button\" variant=\"text\"\n customClass=\"cqa-api-edit-step-indicator-item cqa-api-edit-step-indicator-item--clickable\"\n [attr.aria-label]=\"'Step ' + step.index + ': ' + step.label\" [attr.aria-pressed]=\"step.index === currentStep\"\n (clicked)=\"setStep(step.index)\">\n <span class=\"cqa-api-edit-step-indicator-circle\"\n [class.cqa-api-edit-step-indicator-circle--active]=\"step.index === currentStep\">\n {{ step.index }}\n </span>\n <span class=\"cqa-api-edit-step-indicator-label\"\n [class.cqa-api-edit-step-indicator-label--active]=\"step.index === currentStep\">\n {{ step.label }}\n </span>\n <div *ngIf=\"i < stepLabels.length - 1\" class=\"cqa-api-edit-step-indicator-line\" aria-hidden=\"true\"></div>\n </cqa-button>\n </ng-container>\n </div>\n\n <!-- Step content viewport: smooth slide between steps -->\n <div class=\"cqa-api-edit-step-viewport\">\n <div class=\"cqa-api-edit-step-strip\" [style.transform]=\"'translateX(' + stripTranslatePercent + '%)'\">\n <!-- Step 1: Environment, request, body, response -->\n <div class=\"cqa-api-edit-step-panel\">\n <!-- Environment row: new line, select aligned right -->\n <div class=\"cqa-api-edit-step-environment-row\">\n <cqa-dynamic-select *ngIf=\"environmentForm\" [form]=\"environmentForm\" [config]=\"environmentSelectConfig\"\n class=\"cqa-api-edit-step-environment-select\" aria-label=\"Environment\"\n (selectionChange)=\"onEnvironmentSelectionChange($event)\">\n </cqa-dynamic-select>\n </div>\n\n <!-- Request row: method, URL, buttons -->\n <div class=\"cqa-api-edit-step-request-row\">\n <cqa-dynamic-select *ngIf=\"methodForm\" [form]=\"methodForm\" [config]=\"methodSelectConfig\"\n class=\"cqa-api-edit-step-method-select\" aria-label=\"HTTP method\"\n (selectionChange)=\"onMethodSelectionChange($event)\">\n </cqa-dynamic-select>\n <div class=\"cqa-api-edit-step-url-wrap\">\n <cqa-custom-input [(value)]=\"url\" [label]=\"''\" placeholder=\"\" [fullWidth]=\"true\" size=\"md\">\n </cqa-custom-input>\n </div>\n <div class=\"cqa-api-edit-step-import-curl-trigger\" style=\"display: contents\" (click)=\"openImportCurlPanel()\">\n <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Import API cURL\"\n customClass=\"cqa-api-edit-step-btn-outline cqa-font-inter cqa-text-[14px] cqa-font-semibold cqa-leading-[100%] cqa-tracking-normal cqa-text-[#414146]\">\n </cqa-button>\n </div>\n <cqa-button variant=\"filled\" btnSize=\"lg\" text=\"Send Request\" (clicked)=\"onSendRequest()\"\n customClass=\"cqa-api-edit-step-btn-primary cqa-font-inter cqa-text-[14px] cqa-font-semibold cqa-leading-[100%] cqa-tracking-normal cqa-text-[#FBFCFF]\">\n </cqa-button>\n </div>\n\n\n\n <!-- Body content: header section (default) OR import cURL section -->\n <ng-container *ngIf=\"bodyView === 'import-curl'\">\n <div class=\"cqa-api-edit-step-import-curl-panel\">\n <div class=\"cqa-api-edit-step-import-curl-header\">\n <h3 class=\"cqa-api-edit-step-import-curl-title\">Import API cURL</h3>\n </div>\n <div class=\"cqa-api-edit-step-import-curl-separator\"></div>\n <div class=\"cqa-api-edit-step-import-curl-body\">\n <cqa-custom-textarea [value]=\"importCurlControl.value\"\n (valueChange)=\"importCurlControl.setValue($event)\"\n placeholder=\"Paste your cURL command here (e.g., curl -X POST https://api.example.com/users)\"\n [fullWidth]=\"true\" [rows]=\"8\" resize=\"vertical\" size=\"md\"\n class=\"cqa-api-edit-step-import-curl-textarea\">\n </cqa-custom-textarea>\n </div>\n <div class=\"cqa-api-edit-step-import-curl-footer\">\n <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Cancel\"\n customClass=\"cqa-api-edit-step-import-curl-btn-cancel\" (clicked)=\"onCancelImportCurl()\"></cqa-button>\n <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"Import\"\n customClass=\"cqa-api-edit-step-import-curl-btn-import\" (clicked)=\"onImportCurlConfirm()\"></cqa-button>\n </div>\n </div>\n </ng-container>\n\n <ng-container *ngIf=\"bodyView === 'headers'\">\n <div class=\"cqa-api-edit-step-tabs-wrapper\">\n <div class=\"cqa-api-edit-step-tabs\">\n <cqa-button *ngFor=\"let tab of payloadTabs\" type=\"button\" variant=\"text\" [text]=\"tab.label\"\n [customClass]=\"'cqa-api-edit-step-tab' + (activePayloadTab === tab.value ? ' cqa-api-edit-step-tab--active' : '') + (isMethodWithoutBody && tab.value !== 'headers' ? ' cqa-api-edit-step-tab--disabled' : '')\"\n [disabled]=\"isMethodWithoutBody && tab.value !== 'headers'\"\n (clicked)=\"setPayloadTab(tab.value)\">\n </cqa-button>\n </div>\n\n <!-- Payload content (Headers) -->\n <div *ngIf=\"activePayloadTab === 'headers'\" class=\"cqa-api-edit-step-payload\">\n <div class=\"cqa-api-edit-step-headers-grid\">\n <span class=\"cqa-api-edit-step-headers-label\">Header Name</span>\n <span class=\"cqa-api-edit-step-headers-label\">Header Value</span>\n <span class=\"cqa-api-edit-step-headers-label cqa-api-edit-step-headers-label--empty\"\n aria-hidden=\"true\"></span>\n </div>\n <div *ngFor=\"let row of headerRows; let i = index; trackBy: trackByHeader\" [formGroup]=\"row\"\n class=\"cqa-api-edit-step-header-row\">\n <cqa-dynamic-select [form]=\"row\" [config]=\"headerNameSelectConfig\"\n (addCustomValue)=\"onHeaderNameAddCustomValue($event, row)\"\n class=\"cqa-api-edit-step-header-type-select cqa-w-full\">\n </cqa-dynamic-select>\n <cqa-custom-input [value]=\"row.get('value')?.value ?? ''\"\n (valueChange)=\"row.get('value')?.setValue($event)\" placeholder=\"Value\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-header-value-input\" ariaLabel=\"Header value\">\n </cqa-custom-input>\n <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-header-delete'\"\n [tooltip]=\"'Remove header'\" (clicked)=\"removeHeader(i)\">\n <svg class=\"cqa-api-edit-step-header-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#F9BFBF\" />\n </svg>\n </cqa-button>\n </div>\n <div class=\"cqa-api-edit-step-add-header-wrap\">\n <cqa-button type=\"button\" variant=\"text\" text=\"+ Add Header\"\n [customClass]=\"'cqa-api-edit-step-add-header-link'\" (clicked)=\"addHeader()\">\n </cqa-button>\n </div>\n </div>\n\n <!-- Payload content (Body only): type, format, text area, Back/Next -->\n <div *ngIf=\"activePayloadTab === 'body'\"\n class=\"cqa-api-edit-step-payload cqa-api-edit-step-payload-editor\">\n <div class=\"cqa-api-edit-step-payload-type-row\">\n <span class=\"cqa-api-edit-step-payload-type-label\">Type</span>\n <div class=\"cqa-api-edit-step-payload-type-radios\">\n <cqa-segment-control [value]=\"payloadType\" [segments]=\"payloadTypeSegments\"\n (valueChange)=\"onPayloadTypeChange($event)\"\n class=\"cqa-api-edit-step-payload-type-segment\">\n </cqa-segment-control>\n </div>\n <div *ngIf=\"payloadType === 'raw'\" class=\"cqa-api-edit-step-payload-format-wrap\">\n <span class=\"cqa-api-edit-step-payload-format-label\">Format:</span>\n <cqa-dynamic-select *ngIf=\"payloadFormatForm\" [form]=\"payloadFormatForm\"\n [config]=\"payloadFormatSelectConfig\" class=\"cqa-api-edit-step-payload-format-select\"\n aria-label=\"Format\">\n </cqa-dynamic-select>\n </div>\n </div>\n <!-- Raw: text area with line numbers (Postman-style payload body) -->\n <div *ngIf=\"payloadType === 'raw'\" class=\"cqa-api-edit-step-payload-body\" #payloadEditorWithLinesRef>\n <div class=\"cqa-api-edit-step-payload-editor-with-lines\">\n <div class=\"cqa-api-edit-step-payload-line-numbers\" aria-hidden=\"true\">\n <span *ngFor=\"let n of payloadLineNumbers\" class=\"cqa-api-edit-step-payload-line-num\">\n <span *ngIf=\"payloadJsonError && payloadJsonError.line === n\"\n class=\"cqa-api-edit-step-payload-line-error-icon\" [title]=\"payloadJsonErrorTooltip\"\n aria-label=\"Parse error on this line\">\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" aria-hidden=\"true\">\n <circle cx=\"7\" cy=\"7\" r=\"6\" fill=\"#EF4444\"/>\n <path d=\"M4 4l6 6M10 4l-6 6\" stroke=\"white\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\n </svg>\n </span>\n <span class=\"cqa-api-edit-step-payload-line-num-text\">{{ n }}</span>\n </span>\n </div>\n <div class=\"cqa-api-edit-step-payload-textarea-cell\">\n <div class=\"cqa-api-edit-step-payload-textarea cqa-w-full cqa-flex cqa-flex-col cqa-min-h-0\"\n [ngClass]=\"{'cqa-api-edit-step-payload-textarea--error': payloadJsonError}\">\n <textarea #payloadTextareaRef\n class=\"cqa-api-edit-step-payload-textarea-input cqa-w-full cqa-resize-y cqa-outline-none cqa-box-border\"\n [ngClass]=\"{'cqa-api-edit-step-payload-textarea-input--error': payloadJsonError}\"\n [value]=\"payloadText\"\n [attr.rows]=\"10\"\n placeholder=\"\"\n [attr.aria-label]=\"'Payload'\"\n [attr.aria-invalid]=\"!!payloadJsonError\"\n (input)=\"onPayloadInput($event)\"\n (keydown)=\"onPayloadKeydown($event)\">\n </textarea>\n </div>\n </div>\n </div>\n <p *ngIf=\"payloadJsonError\" class=\"cqa-api-edit-step-payload-json-error-msg\">\n Invalid JSON format. Please check your syntax.\n </p>\n </div>\n\n <!-- x-www-form-urlencoded: Key\u2013Value rows, add/remove dynamically -->\n <div *ngIf=\"payloadType === 'x-www-form-urlencoded'\" class=\"cqa-api-edit-step-key-value\">\n <div class=\"cqa-api-edit-step-key-value-grid cqa-api-edit-step-key-value-header\">\n <span class=\"cqa-api-edit-step-key-value-label\">Key</span>\n <span class=\"cqa-api-edit-step-key-value-label\">Value</span>\n <span class=\"cqa-api-edit-step-key-value-label cqa-api-edit-step-key-value-label--empty\"\n aria-hidden=\"true\"></span>\n </div>\n <div *ngFor=\"let row of keyValueRows; let i = index; trackBy: trackByKeyValue\" [formGroup]=\"row\"\n class=\"cqa-api-edit-step-key-value-row\">\n <cqa-custom-input [value]=\"row.get('key')?.value ?? ''\"\n (valueChange)=\"row.get('key')?.setValue($event)\" placeholder=\"Key\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-key-value-input\" ariaLabel=\"Key\">\n </cqa-custom-input>\n <cqa-custom-input [value]=\"row.get('value')?.value ?? ''\"\n (valueChange)=\"row.get('value')?.setValue($event)\" placeholder=\"Value\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-key-value-input\" ariaLabel=\"Value\">\n </cqa-custom-input>\n <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-key-value-delete'\"\n [tooltip]=\"'Remove row'\" (clicked)=\"removeKeyValueRow(i)\">\n <svg class=\"cqa-api-edit-step-key-value-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#F9BFBF\" />\n </svg>\n </cqa-button>\n </div>\n <div class=\"cqa-api-edit-step-key-value-add-wrap\">\n <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"+ Add Row\"\n customClass=\"cqa-api-edit-step-key-value-add-btn\" (clicked)=\"addKeyValueRow()\">\n </cqa-button>\n </div>\n </div>\n\n <!-- Form Data: Key\u2013Type\u2013Value rows; Type is a dropdown (Text/File), add/remove dynamically -->\n <div *ngIf=\"payloadType === 'form-data'\" class=\"cqa-api-edit-step-key-type-value\">\n <div class=\"cqa-api-edit-step-key-type-value-header\">\n <span class=\"cqa-api-edit-step-key-type-value-label\">Key</span>\n <span class=\"cqa-api-edit-step-key-type-value-label\">Type</span>\n <span class=\"cqa-api-edit-step-key-type-value-label\">Value</span>\n <span class=\"cqa-api-edit-step-key-type-value-label cqa-api-edit-step-key-type-value-label--empty\"\n aria-hidden=\"true\"></span>\n </div>\n <div *ngFor=\"let row of keyTypeValueRows; let i = index; trackBy: trackByKeyTypeValue\" [formGroup]=\"row\"\n class=\"cqa-api-edit-step-key-type-value-row\">\n <cqa-custom-input [value]=\"row.get('key')?.value ?? ''\"\n (valueChange)=\"row.get('key')?.setValue($event)\" placeholder=\"Key\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-key-type-value-input\" ariaLabel=\"Key\">\n </cqa-custom-input>\n <cqa-dynamic-select [form]=\"row\" [config]=\"keyTypeValueTypeSelectConfig\"\n class=\"cqa-api-edit-step-key-type-value-type-select cqa-w-full\">\n </cqa-dynamic-select>\n <cqa-custom-input [value]=\"row.get('value')?.value ?? ''\"\n (valueChange)=\"row.get('value')?.setValue($event)\" placeholder=\"Value\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-key-type-value-input\" ariaLabel=\"Value\">\n </cqa-custom-input>\n <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-key-type-value-delete'\"\n [tooltip]=\"'Remove row'\" (clicked)=\"removeKeyTypeValueRow(i)\">\n <svg class=\"cqa-api-edit-step-key-type-value-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#F9BFBF\" />\n </svg>\n </cqa-button>\n </div>\n <div class=\"cqa-api-edit-step-key-type-value-add-wrap\">\n <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"+ Add Row\"\n customClass=\"cqa-api-edit-step-key-type-value-add-btn\" (clicked)=\"addKeyTypeValueRow()\">\n </cqa-button>\n </div>\n </div>\n </div>\n </div>\n </ng-container>\n\n </div>\n <!-- Step 2: Variable Name: input, validation, Back / Next -->\n <div class=\"cqa-api-edit-step-panel\">\n <div class=\"cqa-api-edit-step-variable-section\">\n <div class=\"cqa-api-edit-step-variable-input-wrap\">\n <cqa-custom-input [(value)]=\"variableName\" [label]=\"''\" placeholder=\"Variable Name\" [fullWidth]=\"true\"\n size=\"md\" (valueChange)=\"onVariableNameChange()\" aria-label=\"Variable Name\">\n </cqa-custom-input>\n </div>\n <p *ngIf=\"variableNameError\" class=\"cqa-api-edit-step-variable-error\" role=\"alert\">{{ variableNameError }}</p>\n </div>\n </div>\n <!-- Step 3: Response Body / Status tabs -->\n <div class=\"cqa-api-edit-step-panel\">\n <div class=\"cqa-api-edit-step-tabs-wrapper\">\n <div class=\"cqa-api-edit-step-tabs\">\n <cqa-button *ngFor=\"let tab of responseVerificationTabs\" type=\"button\" variant=\"text\" [text]=\"tab.label\"\n [customClass]=\"'cqa-api-edit-step-tab' + (activeResponseVerificationTab === tab.value ? ' cqa-api-edit-step-tab--active' : '')\"\n (clicked)=\"setResponseVerificationTab(tab.value)\">\n </cqa-button>\n </div>\n\n <!-- Response Body tab content: verification grid -->\n <div *ngIf=\"activeResponseVerificationTab === 'response-body'\" class=\"cqa-api-edit-step-step3-content\">\n <div class=\"cqa-api-edit-step-verification-header-row\">\n <span></span>\n <cqa-button type=\"button\" variant=\"text\" text=\"Add Verification\"\n customClass=\"cqa-api-edit-step-verification-add-link\" (clicked)=\"addVerificationRow()\">\n </cqa-button>\n </div>\n <div class=\"cqa-api-edit-step-verification\">\n <div class=\"cqa-api-edit-step-verification-grid cqa-api-edit-step-verification-grid-header\">\n <span class=\"cqa-api-edit-step-verification-label\">S.no</span>\n <span class=\"cqa-api-edit-step-verification-label\">JSON Path</span>\n <span class=\"cqa-api-edit-step-verification-label\">Verification</span>\n <span class=\"cqa-api-edit-step-verification-label\">Data Type</span>\n <span class=\"cqa-api-edit-step-verification-label\">Expected Value</span>\n <span class=\"cqa-api-edit-step-verification-label cqa-api-edit-step-verification-label--empty\"\n aria-hidden=\"true\"></span>\n </div>\n <div *ngFor=\"let row of verificationRows; let i = index; trackBy: trackByVerification\" [formGroup]=\"row\"\n class=\"cqa-api-edit-step-verification-row\">\n <span class=\"cqa-api-edit-step-verification-sno\">{{ i + 1 }}</span>\n <cqa-custom-input [value]=\"row.get('jsonPath')?.value ?? ''\"\n (valueChange)=\"row.get('jsonPath')?.setValue($event)\" placeholder=\"Json Path\" [fullWidth]=\"true\"\n size=\"sm\" class=\"cqa-api-edit-step-verification-input\" ariaLabel=\"JSON Path\">\n </cqa-custom-input>\n <cqa-dynamic-select [form]=\"row\" [config]=\"verificationSelectConfig\"\n class=\"cqa-api-edit-step-verification-select cqa-w-full\">\n </cqa-dynamic-select>\n <cqa-dynamic-select [form]=\"row\" [config]=\"verificationDataTypeSelectConfig\"\n class=\"cqa-api-edit-step-verification-select cqa-w-full\"\n (selectionChange)=\"onVerificationDataTypeChange(row, $event)\">\n </cqa-dynamic-select>\n <!-- Expected Value: text for String/Array/Object, number for Number, dropdown for Boolean -->\n <ng-container [ngSwitch]=\"row.get('dataType')?.value\">\n <cqa-dynamic-select *ngSwitchCase=\"'boolean'\" [form]=\"row\"\n [config]=\"verificationExpectedValueBooleanSelectConfig\"\n class=\"cqa-api-edit-step-verification-select cqa-api-edit-step-verification-expected-select cqa-w-full\">\n </cqa-dynamic-select>\n <cqa-custom-input *ngSwitchCase=\"'number'\" [value]=\"row.get('expectedValue')?.value ?? ''\"\n (valueChange)=\"row.get('expectedValue')?.setValue($event)\" placeholder=\"Expected Value (number)\"\n type=\"number\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-verification-input cqa-api-edit-step-verification-expected-number\"\n ariaLabel=\"Expected Value\">\n </cqa-custom-input>\n <cqa-custom-input *ngSwitchDefault [value]=\"row.get('expectedValue')?.value ?? ''\"\n (valueChange)=\"row.get('expectedValue')?.setValue($event)\"\n placeholder=\"Expected Value in String\" [fullWidth]=\"true\" size=\"sm\"\n class=\"cqa-api-edit-step-verification-input\" ariaLabel=\"Expected Value\">\n </cqa-custom-input>\n </ng-container>\n <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-verification-delete'\"\n [tooltip]=\"'Remove row'\" (clicked)=\"removeVerificationRow(i)\">\n <svg class=\"cqa-api-edit-step-verification-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#F9BFBF\" />\n </svg>\n </cqa-button>\n </div>\n </div>\n </div>\n\n <!-- Status tab content: S.no, Verification dropdown, Expected Value, add/remove rows -->\n <div *ngIf=\"activeResponseVerificationTab === 'status'\" class=\"cqa-api-edit-step-step3-content\">\n <div class=\"cqa-api-edit-step-verification-header-row\">\n <span></span>\n <cqa-button type=\"button\" variant=\"text\" text=\"Add Verification\"\n customClass=\"cqa-api-edit-step-verification-add-link\" (clicked)=\"addStatusVerificationRow()\">\n </cqa-button>\n </div>\n <div class=\"cqa-api-edit-step-status-verification\">\n <div class=\"cqa-api-edit-step-status-verification-header\">\n <span class=\"cqa-api-edit-step-verification-label\">S.no</span>\n <span class=\"cqa-api-edit-step-verification-label\">Verification</span>\n <span class=\"cqa-api-edit-step-verification-label\">Expected Value</span>\n <span class=\"cqa-api-edit-step-verification-label cqa-api-edit-step-verification-label--empty\"\n aria-hidden=\"true\"></span>\n </div>\n <div *ngFor=\"let row of statusVerificationRows; let i = index; trackBy: trackByStatusVerification\"\n [formGroup]=\"row\" class=\"cqa-api-edit-step-status-verification-row\">\n <span class=\"cqa-api-edit-step-verification-sno\">{{ i + 1 }}</span>\n <cqa-dynamic-select [form]=\"row\" [config]=\"statusVerificationSelectConfig\"\n class=\"cqa-api-edit-step-status-verification-select cqa-w-full\">\n </cqa-dynamic-select>\n <cqa-custom-input [value]=\"row.get('expectedValue')?.value ?? ''\"\n (valueChange)=\"row.get('expectedValue')?.setValue($event)\" placeholder=\"Expected Value\"\n type=\"number\" [fullWidth]=\"true\" size=\"sm\" class=\"cqa-api-edit-step-verification-input\"\n ariaLabel=\"Expected Value\">\n </cqa-custom-input>\n <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-verification-delete'\"\n [tooltip]=\"'Remove row'\" (clicked)=\"removeStatusVerificationRow(i)\">\n <svg class=\"cqa-api-edit-step-verification-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n <path\n d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n fill=\"#F9BFBF\" />\n </svg>\n </cqa-button>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Response Preview -->\n <div class=\"cqa-api-edit-step-response\">\n <h3 class=\"cqa-api-edit-step-response-title\">Response Preview</h3>\n <pre class=\"cqa-api-edit-step-response-content\">{{ responsePreview }}</pre>\n </div>\n\n <!-- Step actions: one row, full width. Step 1: Cancel + Next; Step 2: Back + Next; Step 3: Back + Create -->\n <div class=\"cqa-api-edit-step-actions\">\n <ng-container [ngSwitch]=\"currentStep\">\n <ng-container *ngSwitchCase=\"1\">\n <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Cancel\"\n customClass=\"cqa-api-edit-step-actions-btn-cancel\" (clicked)=\"onBack()\">\n </cqa-button>\n <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"Next\"\n customClass=\"cqa-api-edit-step-actions-btn-next\" (clicked)=\"onNext()\">\n </cqa-button>\n </ng-container>\n <ng-container *ngSwitchCase=\"2\">\n <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Back\"\n customClass=\"cqa-api-edit-step-actions-btn-back\" (clicked)=\"onBack()\">\n </cqa-button>\n <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"Next\"\n customClass=\"cqa-api-edit-step-actions-btn-next\" (clicked)=\"onNext()\">\n </cqa-button>\n </ng-container>\n <ng-container *ngSwitchCase=\"3\">\n <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Back\"\n customClass=\"cqa-api-edit-step-actions-btn-back\" (clicked)=\"onBack()\">\n </cqa-button>\n <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"Create\"\n customClass=\"cqa-api-edit-step-actions-btn-create\" (clicked)=\"onCreate()\">\n </cqa-button>\n </ng-container>\n </ng-container>\n </div>\n</div>" }]
|
|
1071
|
+
}], ctorParameters: function () { return [{ type: i1.FormBuilder }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { initialMethod: [{
|
|
1072
|
+
type: Input
|
|
1073
|
+
}], initialEnvironment: [{
|
|
1074
|
+
type: Input
|
|
1075
|
+
}], initialStep: [{
|
|
1076
|
+
type: Input
|
|
1077
|
+
}], initialUrl: [{
|
|
1078
|
+
type: Input
|
|
1079
|
+
}], initialPayloadTab: [{
|
|
1080
|
+
type: Input
|
|
1081
|
+
}], initialHeaders: [{
|
|
1082
|
+
type: Input
|
|
1083
|
+
}], initialResponsePreview: [{
|
|
1084
|
+
type: Input
|
|
1085
|
+
}], importCurl: [{
|
|
1086
|
+
type: Output
|
|
1087
|
+
}], importCurlCancel: [{
|
|
1088
|
+
type: Output
|
|
1089
|
+
}], sendRequest: [{
|
|
1090
|
+
type: Output
|
|
1091
|
+
}], back: [{
|
|
1092
|
+
type: Output
|
|
1093
|
+
}], next: [{
|
|
1094
|
+
type: Output
|
|
1095
|
+
}], create: [{
|
|
1096
|
+
type: Output
|
|
1097
|
+
}], headersChange: [{
|
|
1098
|
+
type: Output
|
|
1099
|
+
}], httpMethodOptions: [{
|
|
1100
|
+
type: Input
|
|
1101
|
+
}], environmentOptions: [{
|
|
1102
|
+
type: Input
|
|
1103
|
+
}], formatOptions: [{
|
|
1104
|
+
type: Input
|
|
1105
|
+
}], initialFormat: [{
|
|
1106
|
+
type: Input
|
|
1107
|
+
}], payloadEditorWithLinesRef: [{
|
|
1108
|
+
type: ViewChild,
|
|
1109
|
+
args: ['payloadEditorWithLinesRef']
|
|
1110
|
+
}], payloadTextareaRef: [{
|
|
1111
|
+
type: ViewChild,
|
|
1112
|
+
args: ['payloadTextareaRef']
|
|
1113
|
+
}], headerNameOptions: [{
|
|
1114
|
+
type: Input
|
|
1115
|
+
}], verificationOptions: [{
|
|
1116
|
+
type: Input
|
|
1117
|
+
}], verificationDataTypeOptions: [{
|
|
1118
|
+
type: Input
|
|
1119
|
+
}], statusVerificationOptions: [{
|
|
1120
|
+
type: Input
|
|
1121
|
+
}] } });
|
|
1122
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api-edit-step.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/api-edit-step/api-edit-step.component.ts","../../../../../../src/lib/test-case-details/api-edit-step/api-edit-step.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { FormArray, FormBuilder, FormControl, FormGroup } from '@angular/forms';\nimport { Subscription } from 'rxjs';\nimport { DynamicSelectFieldConfig, SelectOption } from '../../dynamic-select/dynamic-select-field.component';\n\nexport const API_EDIT_STEP_LABELS = {\n  REQUEST_DETAILS: 'Request Details',\n  STORE_RESPONSE: 'Store Response',\n  VALIDATION: 'Validation',\n} as const;\n\nexport type ApiEditPayloadTab = 'headers' | 'body' | 'params' | 'scripts';\n\n/** Which body content block is visible: headers (tabs) or import cURL textarea. */\nexport type ApiEditBodyView = 'headers' | 'import-curl';\n\nexport interface ApiEditHeaderRow {\n  name: string;\n  type: string;\n  value: string;\n}\n\n/** Payload emitted when user clicks Send Request (environment, method, url, headers). */\nexport interface ApiEditSendRequestPayload {\n  environment: string;\n  method: string;\n  url: string;\n  headers: ApiEditHeaderRow[];\n}\n\n/** Step 1 request + body payload. */\nexport interface ApiEditStep1Payload {\n  environment: string;\n  method: string;\n  url: string;\n  headers: ApiEditHeaderRow[];\n  activePayloadTab: ApiEditPayloadTab;\n  payloadType: 'raw' | 'x-www-form-urlencoded' | 'form-data';\n  payloadFormat: string;\n  payloadText: string;\n  keyValueRows: { key: string; value: string }[];\n  keyTypeValueRows: { key: string; type: string; value: string }[];\n}\n\n/** Step 2 store-response payload. */\nexport interface ApiEditStep2Payload {\n  variableName: string;\n}\n\n/** Response body verification row. */\nexport interface ApiEditResponseBodyVerificationRow {\n  jsonPath: string;\n  verification: string;\n  dataType: string;\n  expectedValue: string;\n}\n\n/** Status verification row. */\nexport interface ApiEditStatusVerificationRow {\n  verification: string;\n  expectedValue: string;\n}\n\n/** Step 3 validation payload. */\nexport interface ApiEditStep3Payload {\n  activeResponseVerificationTab: 'response-body' | 'status';\n  responseBodyVerifications: ApiEditResponseBodyVerificationRow[];\n  statusVerifications: ApiEditStatusVerificationRow[];\n}\n\n/** Full payload emitted when user clicks Create (all steps’ data). */\nexport interface ApiEditCreatePayload {\n  step1: ApiEditStep1Payload;\n  step2: ApiEditStep2Payload;\n  step3: ApiEditStep3Payload;\n}\n\n/** Environment option: either a string (display = value) or an object with id, name, value, label. */\nexport type EnvironmentOptionInput = string | SelectOption;\n\nconst METHODS_WITHOUT_BODY = ['GET', 'HEAD', 'DELETE'];\n\n/** Auto-close pairs for payload JSON editor: opening char -> closing char. */\nconst PAYLOAD_AUTO_CLOSE_PAIRS: Record<string, string> = { '{': '}', '[': ']', '\"': '\"', \"'\": \"'\" };\n\n@Component({\n  selector: 'cqa-api-edit-step',\n  templateUrl: './api-edit-step.component.html',\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.Default,\n})\nexport class ApiEditStepComponent implements OnChanges, OnInit, AfterViewInit, OnDestroy {\n  @Input() initialMethod?: string;\n  @Input() initialEnvironment?: string;\n  @Input() initialStep?: number;\n  @Input() initialUrl?: string;\n  @Input() initialPayloadTab?: ApiEditPayloadTab;\n  @Input() initialHeaders?: ApiEditHeaderRow[];\n  @Input() initialResponsePreview?: string;\n\n  /** Emits the cURL string when user clicks Import (value from the textarea control). */\n  @Output() importCurl = new EventEmitter<string>();\n  /** Emits when user cancels the Import cURL panel (clicks Cancel). */\n  @Output() importCurlCancel = new EventEmitter<void>();\n  /** Emits when user clicks Send Request, with environment, method, url, and headers. */\n  @Output() sendRequest = new EventEmitter<ApiEditSendRequestPayload>();\n  @Output() back = new EventEmitter<void>();\n  @Output() next = new EventEmitter<void>();\n  /** Emits when user clicks Create with all entered details: step1 (environment, HTTP method, URL, headers, body), step2 (variable name), step3 (verifications). */\n  @Output() create = new EventEmitter<ApiEditCreatePayload>();\n  /** Emits whenever headers change (add, remove, or edit) so the parent can reflect them in the canvas/controls. */\n  @Output() headersChange = new EventEmitter<ApiEditHeaderRow[]>();\n\n  /** Form control for Import cURL textarea; value is emitted when user clicks Import. */\n  readonly importCurlControl = new FormControl('');\n\n  variableName = '';\n  variableNameError = '';\n\n  /** Controls which body content is visible: headers (default) or import-curl. */\n  bodyView: ApiEditBodyView = 'headers';\n\n  readonly stepLabels = [\n    { index: 1, label: API_EDIT_STEP_LABELS.REQUEST_DETAILS },\n    { index: 2, label: API_EDIT_STEP_LABELS.STORE_RESPONSE },\n    { index: 3, label: API_EDIT_STEP_LABELS.VALIDATION },\n  ];\n  currentStep = 1;\n\n  /** The label for the current step only (used when showing a single step in the indicator). */\n  get activeStepLabel(): { index: number; label: string } | undefined {\n    return this.stepLabels[this.currentStep - 1];\n  }\n\n  private readonly minStep = 1;\n  private readonly maxStep = 3;\n\n  /** Percentage to translate the step strip (0, -33.333, -66.666) for CSS transition. */\n  get stripTranslatePercent(): number {\n    return (this.currentStep - 1) * -33.333;\n  }\n\n  setStep(step: number): void {\n    if (step >= this.minStep && step <= this.maxStep) {\n      this.currentStep = step;\n    }\n  }\n\n  private applyInitialStep(value: number | undefined): void {\n    if (value == null) return;\n    const clamped = Math.max(this.minStep, Math.min(this.maxStep, Math.floor(value)));\n    this.currentStep = clamped;\n  }\n\n  /** HTTP method options: array of strings or objects with id, name, value, label (passed from parent). */\n  @Input() httpMethodOptions: EnvironmentOptionInput[] = [];\n\n  /** Form for HTTP method select (cqa-dynamic-select) */\n  methodForm!: FormGroup;\n\n  /** Config for method dropdown (updated when httpMethodOptions changes to avoid new reference every CD) */\n  methodSelectConfig: DynamicSelectFieldConfig = {\n    key: 'method',\n    placeholder: 'Method',\n    searchable: false,\n    options: [],\n  };\n\n  private updateMethodSelectConfig(): void {\n    const options = (this.httpMethodOptions ?? []).map((m) => ApiEditStepComponent.toSelectOption(m));\n    this.methodSelectConfig = {\n      ...this.methodSelectConfig,\n      options,\n    };\n  }\n\n  private getMethodValues(): string[] {\n    return (this.httpMethodOptions ?? [])\n      .map((m) => ApiEditStepComponent.getOptionValue(m))\n      .filter((v): v is string => v != null);\n  }\n\n  /** Form for environment select (cqa-dynamic-select) */\n  environmentForm!: FormGroup;\n\n  /** Environment options: array of strings or objects with id, name, value, label (passed from parent). */\n  @Input() environmentOptions: EnvironmentOptionInput[] = [];\n\n  /** Config for environment dropdown (updated when environmentOptions changes to avoid new reference every CD) */\n  environmentSelectConfig: DynamicSelectFieldConfig = {\n    key: 'environment',\n    placeholder: 'Environment',\n    searchable: false,\n    options: [],\n  };\n\n  private static toSelectOption(item: EnvironmentOptionInput): SelectOption {\n    if (typeof item === 'string') {\n      return { id: item, value: item, name: item, label: item };\n    }\n    const o = item as SelectOption;\n    return {\n      id: o.id ?? o.value,\n      value: o.value ?? o.id,\n      name: o.name ?? o.label ?? String(o.value ?? o.id),\n      label: o.label ?? o.name ?? String(o.value ?? o.id),\n    };\n  }\n\n  private static getOptionValue(item: EnvironmentOptionInput): string | undefined {\n    if (typeof item === 'string') return item;\n    const o = item as SelectOption;\n    const v = o.value ?? o.id;\n    return v != null ? String(v) : undefined;\n  }\n\n  private updateEnvironmentSelectConfig(): void {\n    const options = (this.environmentOptions ?? []).map((e) => ApiEditStepComponent.toSelectOption(e));\n    this.environmentSelectConfig = {\n      ...this.environmentSelectConfig,\n      options,\n    };\n  }\n\n  private getEnvironmentValues(): string[] {\n    return (this.environmentOptions ?? [])\n      .map((e) => ApiEditStepComponent.getOptionValue(e))\n      .filter((v): v is string => v != null);\n  }\n\n  /** Current HTTP method (from form or default: first of httpMethodOptions). Treats empty string as unset. */\n  get selectedMethod(): string {\n    const value = this.methodForm?.get('method')?.value;\n    const firstValue = this.httpMethodOptions?.length\n      ? ApiEditStepComponent.getOptionValue(this.httpMethodOptions[0])\n      : undefined;\n    const resolved = value != null && value !== '' ? (typeof value === 'object' ? ApiEditStepComponent.getOptionValue(value as EnvironmentOptionInput) : String(value)) : undefined;\n    return resolved ?? firstValue ?? 'GET';\n  }\n  set selectedMethod(v: string) {\n    this.methodForm?.patchValue({ method: v });\n  }\n\n  /** True when the selected HTTP method is GET, HEAD, or DELETE (no request body). */\n  get isMethodWithoutBody(): boolean {\n    const method = this.selectedMethod != null ? String(this.selectedMethod).toUpperCase() : '';\n    return METHODS_WITHOUT_BODY.includes(method);\n  }\n\n  /** Cached environment value from last selection (so Create payload has the selected value). */\n  private currentEnvironmentValue = '';\n\n  /** Cached HTTP method from last selection (so Create payload has the selected value). */\n  private currentMethodValue = '';\n\n  /** Current environment (from form or default: first of environmentOptions). Treats empty string as unset. */\n  get selectedEnvironment(): string {\n    const value = this.environmentForm?.get('environment')?.value;\n    const firstValue = this.environmentOptions?.length\n      ? ApiEditStepComponent.getOptionValue(this.environmentOptions[0])\n      : undefined;\n    const resolved = value != null && value !== '' ? (typeof value === 'object' ? ApiEditStepComponent.getOptionValue(value as EnvironmentOptionInput) : String(value)) : undefined;\n    return resolved ?? firstValue ?? 'Development';\n  }\n  set selectedEnvironment(v: string) {\n    this.environmentForm?.patchValue({ environment: v });\n  }\n\n  /** Called when user selects an environment; keeps currentEnvironmentValue in sync for create payload. */\n  onEnvironmentSelectionChange(event: { key: string; value: unknown }): void {\n    const v = event.value;\n    this.currentEnvironmentValue =\n      v != null\n        ? typeof v === 'object'\n          ? (ApiEditStepComponent.getOptionValue(v as EnvironmentOptionInput) ?? '')\n          : String(v)\n        : '';\n  }\n\n  /** Called when user selects an HTTP method; keeps currentMethodValue in sync for create payload. */\n  onMethodSelectionChange(event: { key: string; value: unknown }): void {\n    const v = event.value;\n    this.currentMethodValue =\n      v != null\n        ? typeof v === 'object'\n          ? (ApiEditStepComponent.getOptionValue(v as EnvironmentOptionInput) ?? '')\n          : String(v)\n        : '';\n  }\n\n  url = '';\n\n  readonly payloadTabs: { value: ApiEditPayloadTab; label: string }[] = [\n    { value: 'headers', label: 'Headers' },\n    { value: 'body', label: 'Body' },\n    { value: 'params', label: 'Params' },\n    { value: 'scripts', label: 'Scripts' },\n  ];\n  activePayloadTab: ApiEditPayloadTab = 'headers';\n\n  /** Step 3: Response verification tabs */\n  readonly responseVerificationTabs: { value: 'response-body' | 'status'; label: string }[] = [\n    { value: 'response-body', label: 'Response Body' },\n    { value: 'status', label: 'Status' },\n  ];\n  activeResponseVerificationTab: 'response-body' | 'status' = 'response-body';\n\n  /** Payload type for Body/Params/Scripts: raw, x-www-form-urlencoded, form-data */\n  payloadType: 'raw' | 'x-www-form-urlencoded' | 'form-data' = 'raw';\n\n  /** Segment options for payload type (used by cqa-segment-control) */\n  readonly payloadTypeSegments = [\n    { label: 'Raw', value: 'raw' },\n    { label: 'x-www-form-urlencoded', value: 'x-www-form-urlencoded' },\n    { label: 'Form Data', value: 'form-data' },\n  ];\n\n  onPayloadTypeChange(val: string): void {\n    this.payloadType = val as 'raw' | 'x-www-form-urlencoded' | 'form-data';\n  }\n\n  /** Form for Format select (Body/Params/Scripts) */\n  payloadFormatForm!: FormGroup;\n\n  /** Format options: array of strings or objects (passed from parent). Falls back to JSON, XML, HTML, Text when empty. */\n  @Input() formatOptions: EnvironmentOptionInput[] = [];\n\n  @Input() initialFormat?: string;\n\n  private static readonly DEFAULT_FORMAT_OPTIONS: SelectOption[] = [\n    { id: 'json', value: 'json', name: 'JSON', label: 'JSON' },\n    { id: 'xml', value: 'xml', name: 'XML', label: 'XML' },\n    { id: 'html', value: 'html', name: 'HTML', label: 'HTML' },\n    { id: 'text', value: 'text', name: 'Text', label: 'Text' },\n  ];\n\n  /** Config for Format dropdown (updated when formatOptions changes, like Method dropdown) */\n  payloadFormatSelectConfig: DynamicSelectFieldConfig = {\n    key: 'format',\n    placeholder: 'Select',\n    searchable: false,\n    options: [...ApiEditStepComponent.DEFAULT_FORMAT_OPTIONS],\n  };\n\n  private updatePayloadFormatSelectConfig(): void {\n    const options =\n      this.formatOptions?.length > 0\n        ? this.formatOptions.map((f) => ApiEditStepComponent.toSelectOption(f))\n        : ApiEditStepComponent.DEFAULT_FORMAT_OPTIONS;\n    this.payloadFormatSelectConfig = {\n      ...this.payloadFormatSelectConfig,\n      options,\n    };\n  }\n\n  private getFormatValues(): string[] {\n    const source =\n      this.formatOptions?.length > 0\n        ? this.formatOptions\n        : (ApiEditStepComponent.DEFAULT_FORMAT_OPTIONS as EnvironmentOptionInput[]);\n    return source\n      .map((f) => ApiEditStepComponent.getOptionValue(f as EnvironmentOptionInput))\n      .filter((v): v is string => v != null);\n  }\n\n  /** Payload text area value for Body/Params/Scripts */\n  payloadText = '';\n\n  /** JSON parse error when format is JSON and payload is invalid; null when valid or not JSON. */\n  payloadJsonError: {\n    line: number;\n    column: number;\n    message: string;\n    excerpt: string;\n    caretOffset: number;\n  } | null = null;\n\n  /** Line numbers for payload textarea (1, 2, 3, ...) based on current line count. */\n  get payloadLineNumbers(): number[] {\n    const n = Math.max(1, (this.payloadText || '').split(/\\r?\\n/).length);\n    return Array.from({ length: n }, (_, i) => i + 1);\n  }\n\n  /** Whether current payload format is JSON (so we validate and show JSON editor behavior). */\n  get isPayloadFormatJson(): boolean {\n    const format = this.payloadFormatForm?.get('format')?.value;\n    return format === 'json';\n  }\n\n  /** Error messages for the payload textarea (red border + bottom line). */\n  get payloadJsonErrors(): string[] {\n    return this.payloadJsonError ? ['Invalid JSON format. Please check your syntax.'] : [];\n  }\n\n  /** Tooltip content for the line-level error icon (parse error on line X, excerpt, caret, message). */\n  get payloadJsonErrorTooltip(): string {\n    if (!this.payloadJsonError) return '';\n    const e = this.payloadJsonError;\n    const caret = ' '.repeat(e.caretOffset) + '^';\n    return `Parse error on line ${e.line}:\\n\\n${e.excerpt}\\n${caret}\\n\\n${e.message}`;\n  }\n\n  onPayloadTextChange(value: string): void {\n    this.payloadText = value;\n    this.validatePayloadJson();\n  }\n\n  onPayloadInput(event: Event): void {\n    const target = event.target as HTMLTextAreaElement | null;\n    this.onPayloadTextChange(target?.value ?? '');\n  }\n\n  onPayloadKeydown(event: KeyboardEvent): void {\n    if (!this.isPayloadFormatJson) return;\n    const key = event.key;\n    const close = PAYLOAD_AUTO_CLOSE_PAIRS[key];\n    if (!close) return;\n    const target = event.target as HTMLTextAreaElement | null;\n    if (!target) return;\n    event.preventDefault();\n    const start = target.selectionStart ?? 0;\n    const end = target.selectionEnd ?? start;\n    const before = this.payloadText.slice(0, start);\n    const after = this.payloadText.slice(end);\n    const insertPos = start + 1;\n    const newValue = before + key + close + after;\n    this.payloadText = newValue;\n    this.validatePayloadJson();\n    setTimeout(() => {\n      this.payloadTextareaRef?.nativeElement?.setSelectionRange(insertPos, insertPos);\n    }, 0);\n  }\n\n  /** Validate payload when format is JSON; sets payloadJsonError or clears it. */\n  validatePayloadJson(): void {\n    if (!this.isPayloadFormatJson) {\n      this.payloadJsonError = null;\n      return;\n    }\n    const text = this.payloadText ?? '';\n    if (text.trim() === '') {\n      this.payloadJsonError = null;\n      return;\n    }\n    try {\n      JSON.parse(text);\n      this.payloadJsonError = null;\n    } catch (err) {\n      const message = err instanceof Error ? err.message : String(err);\n      const position = this.extractPositionFromJsonError(message);\n      const { line, column } = this.positionToLineColumn(text, position);\n      const lines = text.split(/\\r?\\n/);\n      const lineIndex = Math.max(0, line - 1);\n      const lineContent = lines[lineIndex] ?? '';\n      const excerpt = lineContent || ' ';\n      const caretOffset = lineIndex === line - 1 ? Math.min(column - 1, lineContent.length) : 0;\n      this.payloadJsonError = {\n        line,\n        column,\n        message: this.normalizeJsonErrorMessage(message),\n        excerpt,\n        caretOffset,\n      };\n    }\n    this.cdr.markForCheck();\n  }\n\n  private extractPositionFromJsonError(message: string): number {\n    const match = message.match(/position\\s+(\\d+)/i) || message.match(/at\\s+position\\s+(\\d+)/i);\n    return match ? parseInt(match[1], 10) : 0;\n  }\n\n  private positionToLineColumn(text: string, position: number): { line: number; column: number } {\n    const before = text.slice(0, Math.min(position, text.length));\n    const lines = before.split(/\\r?\\n/);\n    const line = lines.length;\n    const lastLine = lines[lines.length - 1] ?? '';\n    return { line, column: lastLine.length + 1 };\n  }\n\n  private normalizeJsonErrorMessage(message: string): string {\n    if (message.includes('Expecting') || message.includes('expecting')) return message;\n    const tokenMatch = message.match(/Unexpected token\\s+['\"]?([^'\"]+)['\"]?/i)\n      || message.match(/unexpected character.*?([^\\s]+)/i);\n    const token = tokenMatch ? tokenMatch[1].trim() : 'undefined';\n    return `Expecting 'STRING', 'NUMBER', 'NULL', 'TRUE', 'FALSE', '{', '[', got '${token}'`;\n  }\n\n  @ViewChild('payloadEditorWithLinesRef') payloadEditorWithLinesRef?: ElementRef<HTMLElement>;\n  @ViewChild('payloadTextareaRef') payloadTextareaRef?: ElementRef<HTMLTextAreaElement>;\n\n  private static readonly DEFAULT_HEADER_NAME_OPTIONS: string[] = [\n    'Accept',\n    'Accept-Charset',\n    'Accept-Encoding',\n    'Accept-Language',\n    'Access-Control-Request-Headers',\n    'Access-Control-Request-Method',\n    'Authorization',\n    'Cache-Control',\n    'Content-MD5',\n    'Content-Length',\n    'Content-Transfer-Encoding',\n    'Content-Type',\n    'Cookie',\n    'Date',\n    'Expect',\n    'From',\n    'Host',\n    'If-Match',\n    'If-Modified-Since',\n    'If-None-Match',\n    'If-Unmodified-Since',\n    'If-Range',\n    'Keep-Alive',\n    'Max-Forwards',\n    'Origin',\n    'Pragma',\n    'Proxy-Authorization',\n    'Range',\n    'Referer',\n    'TE',\n    'Trailer',\n    'Transfer-Encoding',\n    'Upgrade',\n    'User-Agent',\n    'Via',\n    'Warning',\n    'X-Requested-With',\n    'X-Do-Not-Track',\n    'DNT',\n    'x-api-key',\n    'Connection',\n  ];\n\n  /** Header name options: array of strings or objects with id, name, value, label (passed from parent). Falls back to built-in list when empty. */\n  @Input() headerNameOptions: EnvironmentOptionInput[] = [];\n\n  /** Form array for header rows; each item is FormGroup({ name, value }) for cqa-dynamic-select and value input */\n  headersFormArray!: FormArray;\n\n  /** Typed accessor for template (FormArray.controls is AbstractControl[]) */\n  get headerRows(): FormGroup[] {\n    return (this.headersFormArray?.controls ?? []) as FormGroup[];\n  }\n\n  /** Config for header name dropdown (updated when headerNameOptions changes to avoid new reference every CD) */\n  headerNameSelectConfig: DynamicSelectFieldConfig = {\n    key: 'name',\n    placeholder: 'Select header',\n    searchable: true,\n    allowCustomValue: true,\n    options: [],\n  };\n\n  private updateHeaderNameSelectConfig(): void {\n    const source =\n      this.headerNameOptions?.length > 0\n        ? this.headerNameOptions\n        : (ApiEditStepComponent.DEFAULT_HEADER_NAME_OPTIONS as EnvironmentOptionInput[]);\n    const baseOptions = source.map((h) => ApiEditStepComponent.toSelectOption(h));\n    const baseValues = new Set(baseOptions.map((o) => String(o.value ?? o.id ?? '').trim()).filter(Boolean));\n    const currentNamesFromForm = this.getCurrentHeaderNamesFromForm();\n    const extraOptions: SelectOption[] = [];\n    for (const name of currentNamesFromForm) {\n      const key = name.trim();\n      if (key && !baseValues.has(key)) {\n        baseValues.add(key);\n        extraOptions.push({ id: key, value: key, name: key, label: key });\n      }\n    }\n    const options = [...baseOptions, ...extraOptions];\n    this.headerNameSelectConfig = {\n      ...this.headerNameSelectConfig,\n      options,\n    };\n  }\n\n  /** Collects distinct non-empty header names currently in the headers form (for dynamic options). */\n  private getCurrentHeaderNamesFromForm(): string[] {\n    const names: string[] = [];\n    const seen = new Set<string>();\n    if (!this.headersFormArray?.controls?.length) return names;\n    for (const control of this.headersFormArray.controls) {\n      const name = (control as FormGroup).get('name')?.value;\n      const str = name != null ? String(name).trim() : '';\n      if (str && !seen.has(str)) {\n        seen.add(str);\n        names.push(str);\n      }\n    }\n    return names;\n  }\n\n  private readonly defaultHeaders: ApiEditHeaderRow[] = [\n    { name: 'Content-Type', type: 'string', value: 'application/json' },\n  ];\n\n  /** Form array for key-value rows (Body payload); each item is FormGroup({ key, value }) */\n  keyValueFormArray!: FormArray;\n\n  get keyValueRows(): FormGroup[] {\n    return (this.keyValueFormArray?.controls ?? []) as FormGroup[];\n  }\n\n  /** Form array for key-type-value rows; each item is FormGroup({ key, type, value }) */\n  keyTypeValueFormArray!: FormArray;\n\n  get keyTypeValueRows(): FormGroup[] {\n    return (this.keyTypeValueFormArray?.controls ?? []) as FormGroup[];\n  }\n\n  /** Config for Type dropdown in key-type-value rows (Text, File) */\n  readonly keyTypeValueTypeSelectConfig: DynamicSelectFieldConfig = {\n    key: 'type',\n    placeholder: 'Text',\n    searchable: false,\n    options: [\n      { id: 'text', value: 'text', name: 'Text', label: 'Text' },\n      { id: 'file', value: 'file', name: 'File', label: 'File' },\n    ],\n  };\n\n  /** Form array for response body verification rows; each item is FormGroup({ jsonPath, verification, dataType, expectedValue }) */\n  verificationFormArray!: FormArray;\n\n  get verificationRows(): FormGroup[] {\n    return (this.verificationFormArray?.controls ?? []) as FormGroup[];\n  }\n\n  /** Verification options: array of strings or objects (passed from parent). Falls back to built-in list when empty. */\n  @Input() verificationOptions: EnvironmentOptionInput[] = [];\n\n  private static readonly DEFAULT_VERIFICATION_OPTIONS: SelectOption[] = [\n    { id: 'equals', value: 'equals', name: 'Equals', label: 'Equals' },\n    { id: 'not-equal', value: 'not-equal', name: 'Not equal', label: 'Not equal' },\n    { id: 'less-than', value: 'less-than', name: 'Less than', label: 'Less than' },\n    { id: 'greater-than', value: 'greater-than', name: 'Greater than', label: 'Greater than' },\n    { id: 'is-null', value: 'is-null', name: 'Is null', label: 'Is null' },\n    { id: 'is-not-null', value: 'is-not-null', name: 'Is not null', label: 'Is not null' },\n  ];\n\n  /** Config for Verification dropdown (updated when verificationOptions changes, like Method dropdown) */\n  verificationSelectConfig: DynamicSelectFieldConfig = {\n    key: 'verification',\n    placeholder: 'Equals',\n    searchable: false,\n    options: [...ApiEditStepComponent.DEFAULT_VERIFICATION_OPTIONS],\n  };\n\n  private updateVerificationSelectConfig(): void {\n    const options =\n      this.verificationOptions?.length > 0\n        ? this.verificationOptions.map((v) => ApiEditStepComponent.toSelectOption(v))\n        : ApiEditStepComponent.DEFAULT_VERIFICATION_OPTIONS;\n    this.verificationSelectConfig = {\n      ...this.verificationSelectConfig,\n      options,\n    };\n  }\n\n  private getVerificationValues(): string[] {\n    const source =\n      this.verificationOptions?.length > 0\n        ? this.verificationOptions\n        : (ApiEditStepComponent.DEFAULT_VERIFICATION_OPTIONS as EnvironmentOptionInput[]);\n    return source\n      .map((v) => ApiEditStepComponent.getOptionValue(v as EnvironmentOptionInput))\n      .filter((v): v is string => v != null);\n  }\n\n  private getDefaultVerification(): string {\n    const values = this.getVerificationValues();\n    return values.length > 0 ? values[0] : 'equals';\n  }\n\n  /** Data Type options: array of strings or objects (passed from parent). Falls back to built-in list when empty. */\n  @Input() verificationDataTypeOptions: EnvironmentOptionInput[] = [];\n\n  private static readonly DEFAULT_DATA_TYPE_OPTIONS: SelectOption[] = [\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: 'array', value: 'array', name: 'Array', label: 'Array' },\n    { id: 'object', value: 'object', name: 'Object', label: 'Object' },\n  ];\n\n  /** Config for Data Type dropdown (updated when verificationDataTypeOptions changes, like Method dropdown) */\n  verificationDataTypeSelectConfig: DynamicSelectFieldConfig = {\n    key: 'dataType',\n    placeholder: 'String',\n    searchable: false,\n    options: [...ApiEditStepComponent.DEFAULT_DATA_TYPE_OPTIONS],\n  };\n\n  private updateVerificationDataTypeSelectConfig(): void {\n    const options =\n      this.verificationDataTypeOptions?.length > 0\n        ? this.verificationDataTypeOptions.map((d) => ApiEditStepComponent.toSelectOption(d))\n        : ApiEditStepComponent.DEFAULT_DATA_TYPE_OPTIONS;\n    this.verificationDataTypeSelectConfig = {\n      ...this.verificationDataTypeSelectConfig,\n      options,\n    };\n  }\n\n  private getDataTypeValues(): string[] {\n    const source =\n      this.verificationDataTypeOptions?.length > 0\n        ? this.verificationDataTypeOptions\n        : (ApiEditStepComponent.DEFAULT_DATA_TYPE_OPTIONS as EnvironmentOptionInput[]);\n    return source\n      .map((d) => ApiEditStepComponent.getOptionValue(d as EnvironmentOptionInput))\n      .filter((v): v is string => v != null);\n  }\n\n  private getDefaultDataType(): string {\n    const values = this.getDataTypeValues();\n    return values.length > 0 ? values[0] : 'string';\n  }\n\n  /** Config for Expected Value when Data Type is Boolean (true/false dropdown) */\n  readonly verificationExpectedValueBooleanSelectConfig: DynamicSelectFieldConfig = {\n    key: 'expectedValue',\n    placeholder: 'Select',\n    searchable: false,\n    options: [\n      { id: 'true', value: 'true', name: 'true', label: 'true' },\n      { id: 'false', value: 'false', name: 'false', label: 'false' },\n    ],\n  };\n\n  /** Form array for Status tab verification rows; each item is FormGroup({ verification, expectedValue }) */\n  statusVerificationFormArray!: FormArray;\n\n  get statusVerificationRows(): FormGroup[] {\n    return (this.statusVerificationFormArray?.controls ?? []) as FormGroup[];\n  }\n\n  /** Config for Status tab Verification dropdown (Equals, Not equal, Less than, Greater than) */\n  /** Status tab Verification options (Equals, Not equal, etc.). Falls back to built-in list when empty. */\n  @Input() statusVerificationOptions: EnvironmentOptionInput[] = [];\n\n  private static readonly DEFAULT_STATUS_VERIFICATION_OPTIONS: SelectOption[] = [\n    { id: 'equals', value: 'equals', name: 'Equals', label: 'Equals' },\n    { id: 'not-equal', value: 'not-equal', name: 'Not equal', label: 'Not equal' },\n    { id: 'less-than', value: 'less-than', name: 'Less than', label: 'Less than' },\n    { id: 'greater-than', value: 'greater-than', name: 'Greater than', label: 'Greater than' },\n  ];\n\n  /** Config for Status tab Verification dropdown (updated when statusVerificationOptions changes). */\n  statusVerificationSelectConfig: DynamicSelectFieldConfig = {\n    key: 'verification',\n    placeholder: 'Equals',\n    searchable: false,\n    options: [...ApiEditStepComponent.DEFAULT_STATUS_VERIFICATION_OPTIONS],\n  };\n\n  private updateStatusVerificationSelectConfig(): void {\n    const options =\n      this.statusVerificationOptions?.length > 0\n        ? this.statusVerificationOptions.map((o) => ApiEditStepComponent.toSelectOption(o))\n        : ApiEditStepComponent.DEFAULT_STATUS_VERIFICATION_OPTIONS;\n    this.statusVerificationSelectConfig = {\n      ...this.statusVerificationSelectConfig,\n      options,\n    };\n  }\n\n  private getStatusVerificationValues(): string[] {\n    const source =\n      this.statusVerificationOptions?.length > 0\n        ? this.statusVerificationOptions\n        : (ApiEditStepComponent.DEFAULT_STATUS_VERIFICATION_OPTIONS as EnvironmentOptionInput[]);\n    return source\n      .map((o) => ApiEditStepComponent.getOptionValue(o as EnvironmentOptionInput))\n      .filter((v): v is string => v != null);\n  }\n\n  private getDefaultStatusVerification(): string {\n    const values = this.getStatusVerificationValues();\n    return values.length > 0 ? values[0] : 'equals';\n  }\n\n  responsePreview = ``;\n\n  private methodChangesSub?: Subscription;\n  private formatChangesSub?: Subscription;\n  private headerNameOptionsChangesSub?: Subscription;\n  private environmentFormChangesSub?: Subscription;\n  private methodFormChangesSub?: Subscription;\n\n  constructor(\n    private readonly fb: FormBuilder,\n    private readonly cdr: ChangeDetectorRef,\n  ) {}\n\n  ngOnInit(): void {\n    this.applyInitialStep(this.initialStep);\n    this.updateMethodSelectConfig();\n    this.updateHeaderNameSelectConfig();\n    const firstMethodValue = this.httpMethodOptions?.length\n      ? ApiEditStepComponent.getOptionValue(this.httpMethodOptions[0])\n      : undefined;\n    const defaultMethod = this.initialMethod ?? firstMethodValue ?? 'GET';\n    this.currentMethodValue = typeof defaultMethod === 'string' ? defaultMethod : (ApiEditStepComponent.getOptionValue(defaultMethod as EnvironmentOptionInput) ?? '');\n    this.methodForm = this.fb.group({\n      method: [defaultMethod],\n    });\n    const firstValue = this.environmentOptions?.length\n      ? ApiEditStepComponent.getOptionValue(this.environmentOptions[0])\n      : undefined;\n    const defaultEnv = this.initialEnvironment ?? firstValue ?? 'Development';\n    this.currentEnvironmentValue = typeof defaultEnv === 'string' ? defaultEnv : (ApiEditStepComponent.getOptionValue(defaultEnv as EnvironmentOptionInput) ?? '');\n    this.environmentForm = this.fb.group({\n      environment: [defaultEnv],\n    });\n    this.environmentFormChangesSub = this.environmentForm.get('environment')?.valueChanges?.subscribe((v: unknown) => {\n      const s = v != null && v !== '' ? (typeof v === 'object' ? ApiEditStepComponent.getOptionValue(v as EnvironmentOptionInput) : String(v)) : '';\n      this.currentEnvironmentValue = s ?? '';\n    });\n    this.methodFormChangesSub = this.methodForm.get('method')?.valueChanges?.subscribe((v: unknown) => {\n      const s = v != null && v !== '' ? (typeof v === 'object' ? ApiEditStepComponent.getOptionValue(v as EnvironmentOptionInput) : String(v)) : '';\n      this.currentMethodValue = s ?? '';\n    });\n    this.buildHeadersFormArray(\n      this.initialHeaders ?? [{ name: '', type: 'string', value: '' }]\n    );\n    this.headerNameOptionsChangesSub = this.headersFormArray?.valueChanges?.subscribe(() => {\n      this.updateHeaderNameSelectConfig();\n      this.headersChange.emit(this.headers);\n    });\n    this.updateHeaderNameSelectConfig();\n    this.headersChange.emit(this.headers);\n    this.updateEnvironmentSelectConfig();\n    this.updatePayloadFormatSelectConfig();\n    const firstFormatValue = this.getFormatValues()[0];\n    const defaultFormat = this.initialFormat ?? firstFormatValue ?? 'json';\n    this.payloadFormatForm = this.fb.group({ format: [defaultFormat] });\n    this.buildKeyValueFormArray();\n    this.buildKeyTypeValueFormArray();\n    this.updateVerificationSelectConfig();\n    this.updateVerificationDataTypeSelectConfig();\n    this.buildVerificationFormArray();\n    this.updateStatusVerificationSelectConfig();\n    this.buildStatusVerificationFormArray();\n    this.methodChangesSub = this.methodForm.get('method')?.valueChanges?.subscribe((value: string) => {\n      const method = value != null ? String(value).toUpperCase() : '';\n      if (METHODS_WITHOUT_BODY.includes(method) && this.activePayloadTab !== 'headers') {\n        this.activePayloadTab = 'headers';\n        this.cdr.detectChanges();\n      }\n    });\n    this.formatChangesSub = this.payloadFormatForm.get('format')?.valueChanges?.subscribe(() => {\n      this.validatePayloadJson();\n    });\n    this.validatePayloadJson();\n  }\n\n  ngAfterViewInit(): void {\n    this.setupPayloadTextareaScrollSync();\n  }\n\n  ngOnDestroy(): void {\n    this.methodChangesSub?.unsubscribe();\n    this.formatChangesSub?.unsubscribe();\n    this.headerNameOptionsChangesSub?.unsubscribe();\n    this.environmentFormChangesSub?.unsubscribe();\n    this.methodFormChangesSub?.unsubscribe();\n  }\n\n  /** Sync line numbers column scroll with payload textarea scroll (Postman-style). */\n  private setupPayloadTextareaScrollSync(): void {\n    setTimeout(() => {\n      const host = this.payloadEditorWithLinesRef?.nativeElement;\n      if (!host) return;\n      const lineNumbersEl = host.querySelector('.cqa-api-edit-step-payload-line-numbers') as HTMLElement | null;\n      const textarea = host.querySelector('textarea');\n      if (!lineNumbersEl || !textarea) return;\n      textarea.addEventListener('scroll', () => {\n        lineNumbersEl.scrollTop = textarea.scrollTop;\n      });\n    }, 0);\n  }\n\n  private buildKeyValueFormArray(rows?: { key: string; value: string }[]): void {\n    const initial = rows ?? [{ key: '', value: '' }];\n    const groups = initial.map((r) => this.fb.group({ key: [r.key], value: [r.value] }));\n    this.keyValueFormArray = this.fb.array(groups);\n  }\n\n  private buildKeyTypeValueFormArray(rows?: { key: string; type: string; value: string }[]): void {\n    const initial = rows ?? [{ key: '', type: 'text', value: '' }];\n    const groups = initial.map((r) => this.fb.group({ key: [r.key], type: [r.type], value: [r.value] }));\n    this.keyTypeValueFormArray = this.fb.array(groups);\n  }\n\n  private buildVerificationFormArray(rows?: { jsonPath: string; verification: string; dataType: string; expectedValue: string }[]): void {\n    const defaultVerification = this.getDefaultVerification();\n    const defaultDataType = this.getDefaultDataType();\n    const initial = rows ?? [{ jsonPath: '', verification: defaultVerification, dataType: defaultDataType, expectedValue: '' }];\n    const groups = initial.map((r) =>\n      this.fb.group({\n        jsonPath: [r.jsonPath],\n        verification: [r.verification],\n        dataType: [r.dataType],\n        expectedValue: [r.expectedValue],\n      })\n    );\n    this.verificationFormArray = this.fb.array(groups);\n  }\n\n  private buildStatusVerificationFormArray(rows?: { verification: string; expectedValue: string }[]): void {\n    const defaultVerification = this.getDefaultStatusVerification();\n    const initial = rows ?? [{ verification: defaultVerification, expectedValue: '' }];\n    const groups = initial.map((r) =>\n      this.fb.group({\n        verification: [r.verification],\n        expectedValue: [r.expectedValue],\n      })\n    );\n    this.statusVerificationFormArray = this.fb.array(groups);\n  }\n\n  private buildHeadersFormArray(rows: ApiEditHeaderRow[]): void {\n    const groups = rows.map((h) =>\n      this.fb.group({\n        name: [h.name],\n        value: [h.value],\n      })\n    );\n    if (this.headersFormArray) {\n      this.headersFormArray.clear();\n      groups.forEach((g) => this.headersFormArray.push(g));\n    } else {\n      this.headersFormArray = this.fb.array(groups);\n    }\n  }\n\n  /** Handler: show import cURL panel (called when user clicks \"Import API cURL\"). */\n  openImportCurlPanel(): void {\n    this.bodyView = 'import-curl';\n    this.cdr.detectChanges();\n  }\n\n  /** Handler: emit cURL value from control and close panel. Called when user clicks Import button. */\n  onImportCurlConfirm(): void {\n    const value = this.importCurlControl.value ?? '';\n    this.importCurl.emit(typeof value === 'string' ? value : '');\n    this.closeImportCurlPanel();\n  }\n\n  /** Handler: emit cancel and close panel. Called when user clicks Cancel in import cURL panel. */\n  onCancelImportCurl(): void {\n    this.importCurlCancel.emit();\n    this.closeImportCurlPanel();\n  }\n\n  /** Handler: show headers section (called from Cancel or after Import in import panel). */\n  closeImportCurlPanel(): void {\n    this.bodyView = 'headers';\n    this.cdr.detectChanges();\n  }\n\n  onSendRequest(): void {\n    const environment = this.selectedEnvironment != null ? String(this.selectedEnvironment) : '';\n    const method = this.selectedMethod != null ? String(this.selectedMethod) : '';\n    this.sendRequest.emit({\n      environment,\n      method,\n      url: this.url ?? '',\n      headers: this.headers,\n    });\n  }\n\n  onVariableNameChange(): void {\n    this.variableNameError = '';\n  }\n\n  onBack(): void {\n    if (this.currentStep > 1) {\n      this.setStep(this.currentStep - 1);\n    } else {\n      this.back.emit();\n    }\n  }\n\n  onNext(): void {\n    if (this.currentStep === 2 && !this.variableName?.trim()) {\n      this.variableNameError = 'Variable Name is required.';\n      return;\n    }\n    this.variableNameError = '';\n    if (this.currentStep < this.stepLabels.length) {\n      this.setStep(this.currentStep + 1);\n    } else {\n      this.next.emit();\n    }\n  }\n\n  setPayloadTab(value: ApiEditPayloadTab): void {\n    this.activePayloadTab = value;\n  }\n\n  setResponseVerificationTab(value: 'response-body' | 'status'): void {\n    this.activeResponseVerificationTab = value;\n  }\n\n  addHeader(): void {\n    this.headersFormArray.push(\n      this.fb.group({ name: [''], value: [''] })\n    );\n  }\n\n  removeHeader(index: number): void {\n    this.headersFormArray.removeAt(index);\n  }\n\n  /** When user adds a custom header name via the dropdown \"Add '…'\" option, set the control value. */\n  onHeaderNameAddCustomValue(event: { key: string; value: string }, row: FormGroup): void {\n    const control = row.get(event.key);\n    if (control && event.value != null) {\n      control.setValue(event.value.trim());\n    }\n  }\n\n  trackByHeader(index: number): number {\n    return index;\n  }\n\n  addKeyValueRow(): void {\n    this.keyValueFormArray.push(this.fb.group({ key: [''], value: [''] }));\n  }\n\n  removeKeyValueRow(index: number): void {\n    this.keyValueFormArray.removeAt(index);\n  }\n\n  trackByKeyValue(index: number): number {\n    return index;\n  }\n\n  addKeyTypeValueRow(): void {\n    this.keyTypeValueFormArray.push(this.fb.group({ key: [''], type: ['text'], value: [''] }));\n  }\n\n  removeKeyTypeValueRow(index: number): void {\n    this.keyTypeValueFormArray.removeAt(index);\n  }\n\n  trackByKeyTypeValue(index: number): number {\n    return index;\n  }\n\n  addVerificationRow(): void {\n    this.verificationFormArray.push(\n      this.fb.group({\n        jsonPath: [''],\n        verification: [this.getDefaultVerification()],\n        dataType: [this.getDefaultDataType()],\n        expectedValue: [''],\n      })\n    );\n  }\n\n  removeVerificationRow(index: number): void {\n    this.verificationFormArray.removeAt(index);\n  }\n\n  trackByVerification(index: number): number {\n    return index;\n  }\n\n  /** When Data Type changes in Response Body verification, reset Expected Value for that row. */\n  onVerificationDataTypeChange(row: FormGroup, event: { key: string; value: unknown }): void {\n    if (event.key === 'dataType') {\n      row.get('expectedValue')?.setValue('', { emitEvent: false });\n    }\n  }\n\n  addStatusVerificationRow(): void {\n    this.statusVerificationFormArray.push(\n      this.fb.group({ verification: [this.getDefaultStatusVerification()], expectedValue: [''] })\n    );\n  }\n\n  removeStatusVerificationRow(index: number): void {\n    this.statusVerificationFormArray.removeAt(index);\n  }\n\n  trackByStatusVerification(index: number): number {\n    return index;\n  }\n\n  /** Emit all entered details (environment, HTTP method, URL, headers, body, step2 variable, step3 verifications) when user clicks Create. */\n  onCreate(): void {\n    console.log('9999999');\n    \n    try {\n      const payload = this.getCreatePayload();\n      console.log('9999999 payload', payload);\n      this.create.emit(payload);\n    } catch (err) {\n      console.error('ApiEditStep getCreatePayload error:', err);\n      this.create.emit(this.getCreatePayloadFallback());\n    }\n  }\n\n  /** Minimal payload when getCreatePayload throws (so create always emits). */\n  private getCreatePayloadFallback(): ApiEditCreatePayload {\n    return {\n      step1: {\n        environment: '',\n        method: '',\n        url: this.url ?? '',\n        headers: [],\n        activePayloadTab: this.activePayloadTab,\n        payloadType: this.payloadType,\n        payloadFormat: '',\n        payloadText: this.payloadText ?? '',\n        keyValueRows: [],\n        keyTypeValueRows: [],\n      },\n      step2: { variableName: this.variableName ?? '' },\n      step3: {\n        activeResponseVerificationTab: this.activeResponseVerificationTab,\n        responseBodyVerifications: [],\n        statusVerifications: [],\n      },\n    };\n  }\n\n  /** Normalize form control value to string (handles object or string). */\n  private static getControlValue(control: { value: unknown } | null | undefined): string {\n    const v = control?.value;\n    if (v == null || v === '') return '';\n    return typeof v === 'object' ? (ApiEditStepComponent.getOptionValue(v as EnvironmentOptionInput) ?? '') : String(v);\n  }\n\n  /** Build full create payload: environment, HTTP method, URL, headers, body (step1), variable name (step2), verifications (step3). */\n  private getCreatePayload(): ApiEditCreatePayload {\n    const envFromForm = ApiEditStepComponent.getControlValue(this.environmentForm?.get('environment') as { value: unknown } | null);\n    const environment =\n      envFromForm !== '' ? envFromForm : (this.currentEnvironmentValue !== '' ? this.currentEnvironmentValue : this.selectedEnvironment);\n    const methodFromForm = ApiEditStepComponent.getControlValue(this.methodForm?.get('method') as { value: unknown } | null);\n    const method =\n      methodFromForm !== '' ? methodFromForm : (this.currentMethodValue !== '' ? this.currentMethodValue : this.selectedMethod);\n    const format = this.payloadFormatForm?.get('format')?.value ?? '';\n    const keyValueFormArray = this.keyValueFormArray;\n    const keyValueRows = (keyValueFormArray?.controls ?? []).map((c) => ({\n      key: (c as FormGroup).get('key')?.value ?? '',\n      value: (c as FormGroup).get('value')?.value ?? '',\n    }));\n    const keyTypeValueFormArray = this.keyTypeValueFormArray;\n    const keyTypeValueRows = (keyTypeValueFormArray?.controls ?? []).map((c) => ({\n      key: (c as FormGroup).get('key')?.value ?? '',\n      type: (c as FormGroup).get('type')?.value ?? 'text',\n      value: (c as FormGroup).get('value')?.value ?? '',\n    }));\n    const verificationFormArray = this.verificationFormArray;\n    const responseBodyVerifications = (verificationFormArray?.controls ?? []).map((c) => ({\n      jsonPath: (c as FormGroup).get('jsonPath')?.value ?? '',\n      verification: (c as FormGroup).get('verification')?.value ?? '',\n      dataType: (c as FormGroup).get('dataType')?.value ?? '',\n      expectedValue: (c as FormGroup).get('expectedValue')?.value ?? '',\n    }));\n    const statusVerificationFormArray = this.statusVerificationFormArray;\n    const statusVerifications = (statusVerificationFormArray?.controls ?? []).map((c) => ({\n      verification: (c as FormGroup).get('verification')?.value ?? '',\n      expectedValue: (c as FormGroup).get('expectedValue')?.value ?? '',\n    }));\n    return {\n      step1: {\n        environment,\n        method,\n        url: this.url ?? '',\n        headers: this.headers,\n        activePayloadTab: this.activePayloadTab,\n        payloadType: this.payloadType,\n        payloadFormat: format,\n        payloadText: this.payloadText ?? '',\n        keyValueRows,\n        keyTypeValueRows,\n      },\n      step2: {\n        variableName: this.variableName ?? '',\n      },\n      step3: {\n        activeResponseVerificationTab: this.activeResponseVerificationTab,\n        responseBodyVerifications,\n        statusVerifications,\n      },\n    };\n  }\n\n  /** Current header rows from form (for consumers that read headers) */\n  get headers(): ApiEditHeaderRow[] {\n    if (!this.headersFormArray) return [];\n    return this.headersFormArray.controls.map((c) => ({\n      name: (c as FormGroup).get('name')?.value ?? '',\n      type: 'string',\n      value: (c as FormGroup).get('value')?.value ?? '',\n    }));\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['initialMethod']?.currentValue != null && this.methodForm) {\n      const v = changes['initialMethod'].currentValue;\n      this.currentMethodValue = typeof v === 'string' ? v : (ApiEditStepComponent.getOptionValue(v as EnvironmentOptionInput) ?? '');\n      this.methodForm.patchValue({ method: v });\n    }\n    if (changes['httpMethodOptions']) {\n      this.updateMethodSelectConfig();\n      if (this.methodForm && this.httpMethodOptions?.length) {\n        const current = this.methodForm.get('method')?.value;\n        const values = this.getMethodValues();\n        const valid = values.includes(current);\n        if (!valid) {\n          const firstValue = ApiEditStepComponent.getOptionValue(this.httpMethodOptions[0]);\n          if (firstValue != null) {\n            this.currentMethodValue = firstValue;\n            this.methodForm.patchValue({ method: firstValue });\n          }\n        }\n      }\n    }\n    if (changes['initialEnvironment']?.currentValue != null && this.environmentForm) {\n      const v = changes['initialEnvironment'].currentValue;\n      this.currentEnvironmentValue = typeof v === 'string' ? v : (ApiEditStepComponent.getOptionValue(v as EnvironmentOptionInput) ?? '');\n      this.environmentForm.patchValue({ environment: v });\n    }\n    if (changes['environmentOptions']) {\n      this.updateEnvironmentSelectConfig();\n      if (this.environmentForm && this.environmentOptions?.length) {\n        const current = this.environmentForm.get('environment')?.value;\n        const values = this.getEnvironmentValues();\n        const valid = values.includes(current);\n        if (!valid) {\n          const firstValue = ApiEditStepComponent.getOptionValue(this.environmentOptions[0]);\n          if (firstValue != null) {\n            this.currentEnvironmentValue = firstValue;\n            this.environmentForm.patchValue({ environment: firstValue });\n          }\n        }\n      }\n    }\n    if (changes['initialStep']) {\n      this.applyInitialStep(changes['initialStep'].currentValue);\n    }\n    if (changes['initialUrl']?.currentValue != null) {\n      this.url = changes['initialUrl'].currentValue;\n    }\n    if (changes['initialPayloadTab']?.currentValue != null) {\n      this.activePayloadTab = changes['initialPayloadTab'].currentValue;\n    }\n    if (changes['headerNameOptions']) {\n      this.updateHeaderNameSelectConfig();\n    }\n    if (changes['initialHeaders']?.currentValue != null) {\n      this.buildHeadersFormArray(changes['initialHeaders'].currentValue as ApiEditHeaderRow[]);\n      this.updateHeaderNameSelectConfig();\n    }\n    if (changes['initialResponsePreview']?.currentValue != null) {\n      this.responsePreview = changes['initialResponsePreview'].currentValue;\n    }\n    if (changes['initialFormat']?.currentValue != null && this.payloadFormatForm) {\n      this.payloadFormatForm.patchValue({ format: changes['initialFormat'].currentValue });\n    }\n    if (changes['formatOptions']) {\n      this.updatePayloadFormatSelectConfig();\n      if (this.payloadFormatForm) {\n        const current = this.payloadFormatForm.get('format')?.value;\n        const values = this.getFormatValues();\n        const valid = values.includes(current);\n        if (!valid && values.length > 0) {\n          const firstValue = values[0];\n          this.payloadFormatForm.patchValue({ format: firstValue });\n        }\n      }\n    }\n    if (changes['verificationOptions']) {\n      this.updateVerificationSelectConfig();\n      const values = this.getVerificationValues();\n      const defaultVerification = this.getDefaultVerification();\n      if (this.verificationFormArray && values.length > 0) {\n        this.verificationFormArray.controls.forEach((control) => {\n          const current = (control as FormGroup).get('verification')?.value;\n          if (current && !values.includes(current)) {\n            (control as FormGroup).patchValue({ verification: defaultVerification });\n          }\n        });\n      }\n    }\n    if (changes['verificationDataTypeOptions']) {\n      this.updateVerificationDataTypeSelectConfig();\n      const values = this.getDataTypeValues();\n      const defaultDataType = this.getDefaultDataType();\n      if (this.verificationFormArray && values.length > 0) {\n        this.verificationFormArray.controls.forEach((control) => {\n          const current = (control as FormGroup).get('dataType')?.value;\n          if (current && !values.includes(current)) {\n            (control as FormGroup).patchValue({ dataType: defaultDataType });\n          }\n        });\n      }\n    }\n    if (changes['statusVerificationOptions']) {\n      this.updateStatusVerificationSelectConfig();\n      const values = this.getStatusVerificationValues();\n      const defaultVerification = this.getDefaultStatusVerification();\n      if (this.statusVerificationFormArray && values.length > 0) {\n        this.statusVerificationFormArray.controls.forEach((control) => {\n          const current = (control as FormGroup).get('verification')?.value;\n          if (current && !values.includes(current)) {\n            (control as FormGroup).patchValue({ verification: defaultVerification });\n          }\n        });\n      }\n    }\n  }\n}\n","<div class=\"cqa-api-edit-step-container\">\n  <!-- Title -->\n  <h2\n    class=\"cqa-api-edit-step-title cqa-font-inter cqa-text-[12px] cqa-font-semibold cqa-leading-[100%] cqa-tracking-normal cqa-text-[#000000]\">\n    Create API Test Step</h2>\n\n  <!-- Step indicator: all three steps visible, active step highlighted (Postman-style) -->\n  <div class=\"cqa-api-edit-step-indicator\">\n    <ng-container *ngFor=\"let step of stepLabels; let i = index\">\n      <cqa-button type=\"button\" variant=\"text\"\n        customClass=\"cqa-api-edit-step-indicator-item cqa-api-edit-step-indicator-item--clickable\"\n        [attr.aria-label]=\"'Step ' + step.index + ': ' + step.label\" [attr.aria-pressed]=\"step.index === currentStep\"\n        (clicked)=\"setStep(step.index)\">\n        <span class=\"cqa-api-edit-step-indicator-circle\"\n          [class.cqa-api-edit-step-indicator-circle--active]=\"step.index === currentStep\">\n          {{ step.index }}\n        </span>\n        <span class=\"cqa-api-edit-step-indicator-label\"\n          [class.cqa-api-edit-step-indicator-label--active]=\"step.index === currentStep\">\n          {{ step.label }}\n        </span>\n        <div *ngIf=\"i < stepLabels.length - 1\" class=\"cqa-api-edit-step-indicator-line\" aria-hidden=\"true\"></div>\n      </cqa-button>\n    </ng-container>\n  </div>\n\n  <!-- Step content viewport: smooth slide between steps -->\n  <div class=\"cqa-api-edit-step-viewport\">\n    <div class=\"cqa-api-edit-step-strip\" [style.transform]=\"'translateX(' + stripTranslatePercent + '%)'\">\n      <!-- Step 1: Environment, request, body, response -->\n      <div class=\"cqa-api-edit-step-panel\">\n        <!-- Environment row: new line, select aligned right -->\n        <div class=\"cqa-api-edit-step-environment-row\">\n          <cqa-dynamic-select *ngIf=\"environmentForm\" [form]=\"environmentForm\" [config]=\"environmentSelectConfig\"\n            class=\"cqa-api-edit-step-environment-select\" aria-label=\"Environment\"\n            (selectionChange)=\"onEnvironmentSelectionChange($event)\">\n          </cqa-dynamic-select>\n        </div>\n\n        <!-- Request row: method, URL, buttons -->\n        <div class=\"cqa-api-edit-step-request-row\">\n          <cqa-dynamic-select *ngIf=\"methodForm\" [form]=\"methodForm\" [config]=\"methodSelectConfig\"\n            class=\"cqa-api-edit-step-method-select\" aria-label=\"HTTP method\"\n            (selectionChange)=\"onMethodSelectionChange($event)\">\n          </cqa-dynamic-select>\n          <div class=\"cqa-api-edit-step-url-wrap\">\n            <cqa-custom-input [(value)]=\"url\" [label]=\"''\" placeholder=\"\" [fullWidth]=\"true\" size=\"md\">\n            </cqa-custom-input>\n          </div>\n          <div class=\"cqa-api-edit-step-import-curl-trigger\" style=\"display: contents\" (click)=\"openImportCurlPanel()\">\n            <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Import API cURL\"\n              customClass=\"cqa-api-edit-step-btn-outline cqa-font-inter cqa-text-[14px] cqa-font-semibold cqa-leading-[100%] cqa-tracking-normal cqa-text-[#414146]\">\n            </cqa-button>\n          </div>\n          <cqa-button variant=\"filled\" btnSize=\"lg\" text=\"Send Request\" (clicked)=\"onSendRequest()\"\n            customClass=\"cqa-api-edit-step-btn-primary cqa-font-inter cqa-text-[14px] cqa-font-semibold cqa-leading-[100%] cqa-tracking-normal cqa-text-[#FBFCFF]\">\n          </cqa-button>\n        </div>\n\n\n\n        <!-- Body content: header section (default) OR import cURL section -->\n        <ng-container *ngIf=\"bodyView === 'import-curl'\">\n          <div class=\"cqa-api-edit-step-import-curl-panel\">\n            <div class=\"cqa-api-edit-step-import-curl-header\">\n              <h3 class=\"cqa-api-edit-step-import-curl-title\">Import API cURL</h3>\n            </div>\n            <div class=\"cqa-api-edit-step-import-curl-separator\"></div>\n            <div class=\"cqa-api-edit-step-import-curl-body\">\n              <cqa-custom-textarea [value]=\"importCurlControl.value\"\n                (valueChange)=\"importCurlControl.setValue($event)\"\n                placeholder=\"Paste your cURL command here (e.g., curl -X POST https://api.example.com/users)\"\n                [fullWidth]=\"true\" [rows]=\"8\" resize=\"vertical\" size=\"md\"\n                class=\"cqa-api-edit-step-import-curl-textarea\">\n              </cqa-custom-textarea>\n            </div>\n            <div class=\"cqa-api-edit-step-import-curl-footer\">\n              <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Cancel\"\n                customClass=\"cqa-api-edit-step-import-curl-btn-cancel\" (clicked)=\"onCancelImportCurl()\"></cqa-button>\n              <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"Import\"\n                customClass=\"cqa-api-edit-step-import-curl-btn-import\" (clicked)=\"onImportCurlConfirm()\"></cqa-button>\n            </div>\n          </div>\n        </ng-container>\n\n        <ng-container *ngIf=\"bodyView === 'headers'\">\n          <div class=\"cqa-api-edit-step-tabs-wrapper\">\n            <div class=\"cqa-api-edit-step-tabs\">\n              <cqa-button *ngFor=\"let tab of payloadTabs\" type=\"button\" variant=\"text\" [text]=\"tab.label\"\n                [customClass]=\"'cqa-api-edit-step-tab' + (activePayloadTab === tab.value ? ' cqa-api-edit-step-tab--active' : '') + (isMethodWithoutBody && tab.value !== 'headers' ? ' cqa-api-edit-step-tab--disabled' : '')\"\n                [disabled]=\"isMethodWithoutBody && tab.value !== 'headers'\"\n                (clicked)=\"setPayloadTab(tab.value)\">\n              </cqa-button>\n            </div>\n\n            <!-- Payload content (Headers) -->\n            <div *ngIf=\"activePayloadTab === 'headers'\" class=\"cqa-api-edit-step-payload\">\n              <div class=\"cqa-api-edit-step-headers-grid\">\n                <span class=\"cqa-api-edit-step-headers-label\">Header Name</span>\n                <span class=\"cqa-api-edit-step-headers-label\">Header Value</span>\n                <span class=\"cqa-api-edit-step-headers-label cqa-api-edit-step-headers-label--empty\"\n                  aria-hidden=\"true\"></span>\n              </div>\n              <div *ngFor=\"let row of headerRows; let i = index; trackBy: trackByHeader\" [formGroup]=\"row\"\n                class=\"cqa-api-edit-step-header-row\">\n                <cqa-dynamic-select [form]=\"row\" [config]=\"headerNameSelectConfig\"\n                  (addCustomValue)=\"onHeaderNameAddCustomValue($event, row)\"\n                  class=\"cqa-api-edit-step-header-type-select cqa-w-full\">\n                </cqa-dynamic-select>\n                <cqa-custom-input [value]=\"row.get('value')?.value ?? ''\"\n                  (valueChange)=\"row.get('value')?.setValue($event)\" placeholder=\"Value\" [fullWidth]=\"true\" size=\"sm\"\n                  class=\"cqa-api-edit-step-header-value-input\" ariaLabel=\"Header value\">\n                </cqa-custom-input>\n                <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-header-delete'\"\n                  [tooltip]=\"'Remove header'\" (clicked)=\"removeHeader(i)\">\n                  <svg class=\"cqa-api-edit-step-header-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n                    fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n                    <path\n                      d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n                      fill=\"#F9BFBF\" />\n                  </svg>\n                </cqa-button>\n              </div>\n              <div class=\"cqa-api-edit-step-add-header-wrap\">\n                <cqa-button type=\"button\" variant=\"text\" text=\"+ Add Header\"\n                  [customClass]=\"'cqa-api-edit-step-add-header-link'\" (clicked)=\"addHeader()\">\n                </cqa-button>\n              </div>\n            </div>\n\n            <!-- Payload content (Body only): type, format, text area, Back/Next -->\n            <div *ngIf=\"activePayloadTab === 'body'\"\n              class=\"cqa-api-edit-step-payload cqa-api-edit-step-payload-editor\">\n              <div class=\"cqa-api-edit-step-payload-type-row\">\n                <span class=\"cqa-api-edit-step-payload-type-label\">Type</span>\n                <div class=\"cqa-api-edit-step-payload-type-radios\">\n                  <cqa-segment-control [value]=\"payloadType\" [segments]=\"payloadTypeSegments\"\n                    (valueChange)=\"onPayloadTypeChange($event)\"\n                    class=\"cqa-api-edit-step-payload-type-segment\">\n                  </cqa-segment-control>\n                </div>\n                <div *ngIf=\"payloadType === 'raw'\" class=\"cqa-api-edit-step-payload-format-wrap\">\n                  <span class=\"cqa-api-edit-step-payload-format-label\">Format:</span>\n                  <cqa-dynamic-select *ngIf=\"payloadFormatForm\" [form]=\"payloadFormatForm\"\n                    [config]=\"payloadFormatSelectConfig\" class=\"cqa-api-edit-step-payload-format-select\"\n                    aria-label=\"Format\">\n                  </cqa-dynamic-select>\n                </div>\n              </div>\n              <!-- Raw: text area with line numbers (Postman-style payload body) -->\n              <div *ngIf=\"payloadType === 'raw'\" class=\"cqa-api-edit-step-payload-body\" #payloadEditorWithLinesRef>\n                <div class=\"cqa-api-edit-step-payload-editor-with-lines\">\n                  <div class=\"cqa-api-edit-step-payload-line-numbers\" aria-hidden=\"true\">\n                    <span *ngFor=\"let n of payloadLineNumbers\" class=\"cqa-api-edit-step-payload-line-num\">\n                      <span *ngIf=\"payloadJsonError && payloadJsonError.line === n\"\n                        class=\"cqa-api-edit-step-payload-line-error-icon\" [title]=\"payloadJsonErrorTooltip\"\n                        aria-label=\"Parse error on this line\">\n                        <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"14\" height=\"14\" viewBox=\"0 0 14 14\" fill=\"none\" aria-hidden=\"true\">\n                          <circle cx=\"7\" cy=\"7\" r=\"6\" fill=\"#EF4444\"/>\n                          <path d=\"M4 4l6 6M10 4l-6 6\" stroke=\"white\" stroke-width=\"1.5\" stroke-linecap=\"round\"/>\n                        </svg>\n                      </span>\n                      <span class=\"cqa-api-edit-step-payload-line-num-text\">{{ n }}</span>\n                    </span>\n                  </div>\n                  <div class=\"cqa-api-edit-step-payload-textarea-cell\">\n                    <div class=\"cqa-api-edit-step-payload-textarea cqa-w-full cqa-flex cqa-flex-col cqa-min-h-0\"\n                      [ngClass]=\"{'cqa-api-edit-step-payload-textarea--error': payloadJsonError}\">\n                      <textarea #payloadTextareaRef\n                        class=\"cqa-api-edit-step-payload-textarea-input cqa-w-full cqa-resize-y cqa-outline-none cqa-box-border\"\n                        [ngClass]=\"{'cqa-api-edit-step-payload-textarea-input--error': payloadJsonError}\"\n                        [value]=\"payloadText\"\n                        [attr.rows]=\"10\"\n                        placeholder=\"\"\n                        [attr.aria-label]=\"'Payload'\"\n                        [attr.aria-invalid]=\"!!payloadJsonError\"\n                        (input)=\"onPayloadInput($event)\"\n                        (keydown)=\"onPayloadKeydown($event)\">\n                      </textarea>\n                    </div>\n                  </div>\n                </div>\n                <p *ngIf=\"payloadJsonError\" class=\"cqa-api-edit-step-payload-json-error-msg\">\n                  Invalid JSON format. Please check your syntax.\n                </p>\n              </div>\n\n              <!-- x-www-form-urlencoded: Key–Value rows, add/remove dynamically -->\n              <div *ngIf=\"payloadType === 'x-www-form-urlencoded'\" class=\"cqa-api-edit-step-key-value\">\n                <div class=\"cqa-api-edit-step-key-value-grid cqa-api-edit-step-key-value-header\">\n                  <span class=\"cqa-api-edit-step-key-value-label\">Key</span>\n                  <span class=\"cqa-api-edit-step-key-value-label\">Value</span>\n                  <span class=\"cqa-api-edit-step-key-value-label cqa-api-edit-step-key-value-label--empty\"\n                    aria-hidden=\"true\"></span>\n                </div>\n                <div *ngFor=\"let row of keyValueRows; let i = index; trackBy: trackByKeyValue\" [formGroup]=\"row\"\n                  class=\"cqa-api-edit-step-key-value-row\">\n                  <cqa-custom-input [value]=\"row.get('key')?.value ?? ''\"\n                    (valueChange)=\"row.get('key')?.setValue($event)\" placeholder=\"Key\" [fullWidth]=\"true\" size=\"sm\"\n                    class=\"cqa-api-edit-step-key-value-input\" ariaLabel=\"Key\">\n                  </cqa-custom-input>\n                  <cqa-custom-input [value]=\"row.get('value')?.value ?? ''\"\n                    (valueChange)=\"row.get('value')?.setValue($event)\" placeholder=\"Value\" [fullWidth]=\"true\" size=\"sm\"\n                    class=\"cqa-api-edit-step-key-value-input\" ariaLabel=\"Value\">\n                  </cqa-custom-input>\n                  <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-key-value-delete'\"\n                    [tooltip]=\"'Remove row'\" (clicked)=\"removeKeyValueRow(i)\">\n                    <svg class=\"cqa-api-edit-step-key-value-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n                      fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n                      <path\n                        d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n                        fill=\"#F9BFBF\" />\n                    </svg>\n                  </cqa-button>\n                </div>\n                <div class=\"cqa-api-edit-step-key-value-add-wrap\">\n                  <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"+ Add Row\"\n                    customClass=\"cqa-api-edit-step-key-value-add-btn\" (clicked)=\"addKeyValueRow()\">\n                  </cqa-button>\n                </div>\n              </div>\n\n              <!-- Form Data: Key–Type–Value rows; Type is a dropdown (Text/File), add/remove dynamically -->\n              <div *ngIf=\"payloadType === 'form-data'\" class=\"cqa-api-edit-step-key-type-value\">\n                <div class=\"cqa-api-edit-step-key-type-value-header\">\n                  <span class=\"cqa-api-edit-step-key-type-value-label\">Key</span>\n                  <span class=\"cqa-api-edit-step-key-type-value-label\">Type</span>\n                  <span class=\"cqa-api-edit-step-key-type-value-label\">Value</span>\n                  <span class=\"cqa-api-edit-step-key-type-value-label cqa-api-edit-step-key-type-value-label--empty\"\n                    aria-hidden=\"true\"></span>\n                </div>\n                <div *ngFor=\"let row of keyTypeValueRows; let i = index; trackBy: trackByKeyTypeValue\" [formGroup]=\"row\"\n                  class=\"cqa-api-edit-step-key-type-value-row\">\n                  <cqa-custom-input [value]=\"row.get('key')?.value ?? ''\"\n                    (valueChange)=\"row.get('key')?.setValue($event)\" placeholder=\"Key\" [fullWidth]=\"true\" size=\"sm\"\n                    class=\"cqa-api-edit-step-key-type-value-input\" ariaLabel=\"Key\">\n                  </cqa-custom-input>\n                  <cqa-dynamic-select [form]=\"row\" [config]=\"keyTypeValueTypeSelectConfig\"\n                    class=\"cqa-api-edit-step-key-type-value-type-select cqa-w-full\">\n                  </cqa-dynamic-select>\n                  <cqa-custom-input [value]=\"row.get('value')?.value ?? ''\"\n                    (valueChange)=\"row.get('value')?.setValue($event)\" placeholder=\"Value\" [fullWidth]=\"true\" size=\"sm\"\n                    class=\"cqa-api-edit-step-key-type-value-input\" ariaLabel=\"Value\">\n                  </cqa-custom-input>\n                  <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-key-type-value-delete'\"\n                    [tooltip]=\"'Remove row'\" (clicked)=\"removeKeyTypeValueRow(i)\">\n                    <svg class=\"cqa-api-edit-step-key-type-value-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n                      fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n                      <path\n                        d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n                        fill=\"#F9BFBF\" />\n                    </svg>\n                  </cqa-button>\n                </div>\n                <div class=\"cqa-api-edit-step-key-type-value-add-wrap\">\n                  <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"+ Add Row\"\n                    customClass=\"cqa-api-edit-step-key-type-value-add-btn\" (clicked)=\"addKeyTypeValueRow()\">\n                  </cqa-button>\n                </div>\n              </div>\n            </div>\n          </div>\n        </ng-container>\n\n      </div>\n      <!-- Step 2: Variable Name: input, validation, Back / Next -->\n      <div class=\"cqa-api-edit-step-panel\">\n        <div class=\"cqa-api-edit-step-variable-section\">\n          <div class=\"cqa-api-edit-step-variable-input-wrap\">\n            <cqa-custom-input [(value)]=\"variableName\" [label]=\"''\" placeholder=\"Variable Name\" [fullWidth]=\"true\"\n              size=\"md\" (valueChange)=\"onVariableNameChange()\" aria-label=\"Variable Name\">\n            </cqa-custom-input>\n          </div>\n          <p *ngIf=\"variableNameError\" class=\"cqa-api-edit-step-variable-error\" role=\"alert\">{{ variableNameError }}</p>\n        </div>\n      </div>\n      <!-- Step 3: Response Body / Status tabs -->\n      <div class=\"cqa-api-edit-step-panel\">\n        <div class=\"cqa-api-edit-step-tabs-wrapper\">\n          <div class=\"cqa-api-edit-step-tabs\">\n            <cqa-button *ngFor=\"let tab of responseVerificationTabs\" type=\"button\" variant=\"text\" [text]=\"tab.label\"\n              [customClass]=\"'cqa-api-edit-step-tab' + (activeResponseVerificationTab === tab.value ? ' cqa-api-edit-step-tab--active' : '')\"\n              (clicked)=\"setResponseVerificationTab(tab.value)\">\n            </cqa-button>\n          </div>\n\n          <!-- Response Body tab content: verification grid -->\n          <div *ngIf=\"activeResponseVerificationTab === 'response-body'\" class=\"cqa-api-edit-step-step3-content\">\n            <div class=\"cqa-api-edit-step-verification-header-row\">\n              <span></span>\n              <cqa-button type=\"button\" variant=\"text\" text=\"Add Verification\"\n                customClass=\"cqa-api-edit-step-verification-add-link\" (clicked)=\"addVerificationRow()\">\n              </cqa-button>\n            </div>\n            <div class=\"cqa-api-edit-step-verification\">\n              <div class=\"cqa-api-edit-step-verification-grid cqa-api-edit-step-verification-grid-header\">\n                <span class=\"cqa-api-edit-step-verification-label\">S.no</span>\n                <span class=\"cqa-api-edit-step-verification-label\">JSON Path</span>\n                <span class=\"cqa-api-edit-step-verification-label\">Verification</span>\n                <span class=\"cqa-api-edit-step-verification-label\">Data Type</span>\n                <span class=\"cqa-api-edit-step-verification-label\">Expected Value</span>\n                <span class=\"cqa-api-edit-step-verification-label cqa-api-edit-step-verification-label--empty\"\n                  aria-hidden=\"true\"></span>\n              </div>\n              <div *ngFor=\"let row of verificationRows; let i = index; trackBy: trackByVerification\" [formGroup]=\"row\"\n                class=\"cqa-api-edit-step-verification-row\">\n                <span class=\"cqa-api-edit-step-verification-sno\">{{ i + 1 }}</span>\n                <cqa-custom-input [value]=\"row.get('jsonPath')?.value ?? ''\"\n                  (valueChange)=\"row.get('jsonPath')?.setValue($event)\" placeholder=\"Json Path\" [fullWidth]=\"true\"\n                  size=\"sm\" class=\"cqa-api-edit-step-verification-input\" ariaLabel=\"JSON Path\">\n                </cqa-custom-input>\n                <cqa-dynamic-select [form]=\"row\" [config]=\"verificationSelectConfig\"\n                  class=\"cqa-api-edit-step-verification-select cqa-w-full\">\n                </cqa-dynamic-select>\n                <cqa-dynamic-select [form]=\"row\" [config]=\"verificationDataTypeSelectConfig\"\n                  class=\"cqa-api-edit-step-verification-select cqa-w-full\"\n                  (selectionChange)=\"onVerificationDataTypeChange(row, $event)\">\n                </cqa-dynamic-select>\n                <!-- Expected Value: text for String/Array/Object, number for Number, dropdown for Boolean -->\n                <ng-container [ngSwitch]=\"row.get('dataType')?.value\">\n                  <cqa-dynamic-select *ngSwitchCase=\"'boolean'\" [form]=\"row\"\n                    [config]=\"verificationExpectedValueBooleanSelectConfig\"\n                    class=\"cqa-api-edit-step-verification-select cqa-api-edit-step-verification-expected-select cqa-w-full\">\n                  </cqa-dynamic-select>\n                  <cqa-custom-input *ngSwitchCase=\"'number'\" [value]=\"row.get('expectedValue')?.value ?? ''\"\n                    (valueChange)=\"row.get('expectedValue')?.setValue($event)\" placeholder=\"Expected Value (number)\"\n                    type=\"number\" [fullWidth]=\"true\" size=\"sm\"\n                    class=\"cqa-api-edit-step-verification-input cqa-api-edit-step-verification-expected-number\"\n                    ariaLabel=\"Expected Value\">\n                  </cqa-custom-input>\n                  <cqa-custom-input *ngSwitchDefault [value]=\"row.get('expectedValue')?.value ?? ''\"\n                    (valueChange)=\"row.get('expectedValue')?.setValue($event)\"\n                    placeholder=\"Expected Value in String\" [fullWidth]=\"true\" size=\"sm\"\n                    class=\"cqa-api-edit-step-verification-input\" ariaLabel=\"Expected Value\">\n                  </cqa-custom-input>\n                </ng-container>\n                <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-verification-delete'\"\n                  [tooltip]=\"'Remove row'\" (clicked)=\"removeVerificationRow(i)\">\n                  <svg class=\"cqa-api-edit-step-verification-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n                    fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n                    <path\n                      d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n                      fill=\"#F9BFBF\" />\n                  </svg>\n                </cqa-button>\n              </div>\n            </div>\n          </div>\n\n          <!-- Status tab content: S.no, Verification dropdown, Expected Value, add/remove rows -->\n          <div *ngIf=\"activeResponseVerificationTab === 'status'\" class=\"cqa-api-edit-step-step3-content\">\n            <div class=\"cqa-api-edit-step-verification-header-row\">\n              <span></span>\n              <cqa-button type=\"button\" variant=\"text\" text=\"Add Verification\"\n                customClass=\"cqa-api-edit-step-verification-add-link\" (clicked)=\"addStatusVerificationRow()\">\n              </cqa-button>\n            </div>\n            <div class=\"cqa-api-edit-step-status-verification\">\n              <div class=\"cqa-api-edit-step-status-verification-header\">\n                <span class=\"cqa-api-edit-step-verification-label\">S.no</span>\n                <span class=\"cqa-api-edit-step-verification-label\">Verification</span>\n                <span class=\"cqa-api-edit-step-verification-label\">Expected Value</span>\n                <span class=\"cqa-api-edit-step-verification-label cqa-api-edit-step-verification-label--empty\"\n                  aria-hidden=\"true\"></span>\n              </div>\n              <div *ngFor=\"let row of statusVerificationRows; let i = index; trackBy: trackByStatusVerification\"\n                [formGroup]=\"row\" class=\"cqa-api-edit-step-status-verification-row\">\n                <span class=\"cqa-api-edit-step-verification-sno\">{{ i + 1 }}</span>\n                <cqa-dynamic-select [form]=\"row\" [config]=\"statusVerificationSelectConfig\"\n                  class=\"cqa-api-edit-step-status-verification-select cqa-w-full\">\n                </cqa-dynamic-select>\n                <cqa-custom-input [value]=\"row.get('expectedValue')?.value ?? ''\"\n                  (valueChange)=\"row.get('expectedValue')?.setValue($event)\" placeholder=\"Expected Value\"\n                  type=\"number\" [fullWidth]=\"true\" size=\"sm\" class=\"cqa-api-edit-step-verification-input\"\n                  ariaLabel=\"Expected Value\">\n                </cqa-custom-input>\n                <cqa-button type=\"button\" variant=\"text\" [customClass]=\"'cqa-api-edit-step-verification-delete'\"\n                  [tooltip]=\"'Remove row'\" (clicked)=\"removeStatusVerificationRow(i)\">\n                  <svg class=\"cqa-api-edit-step-verification-delete-icon\" width=\"16\" height=\"16\" viewBox=\"0 0 16 16\"\n                    fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\">\n                    <path\n                      d=\"M10.6663 6V12.6667H5.33301V6H10.6663ZM9.66634 2H6.33301L5.66634 2.66667H3.33301V4H12.6663V2.66667H10.333L9.66634 2ZM11.9997 4.66667H3.99967V12.6667C3.99967 13.4 4.59967 14 5.33301 14H10.6663C11.3997 14 11.9997 13.4 11.9997 12.6667V4.66667Z\"\n                      fill=\"#F9BFBF\" />\n                  </svg>\n                </cqa-button>\n              </div>\n            </div>\n          </div>\n        </div>\n      </div>\n    </div>\n  </div>\n\n  <!-- Response Preview -->\n  <div class=\"cqa-api-edit-step-response\">\n    <h3 class=\"cqa-api-edit-step-response-title\">Response Preview</h3>\n    <pre class=\"cqa-api-edit-step-response-content\">{{ responsePreview }}</pre>\n  </div>\n\n  <!-- Step actions: one row, full width. Step 1: Cancel + Next; Step 2: Back + Next; Step 3: Back + Create -->\n  <div class=\"cqa-api-edit-step-actions\">\n    <ng-container [ngSwitch]=\"currentStep\">\n      <ng-container *ngSwitchCase=\"1\">\n        <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Cancel\"\n          customClass=\"cqa-api-edit-step-actions-btn-cancel\" (clicked)=\"onBack()\">\n        </cqa-button>\n        <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"Next\"\n          customClass=\"cqa-api-edit-step-actions-btn-next\" (clicked)=\"onNext()\">\n        </cqa-button>\n      </ng-container>\n      <ng-container *ngSwitchCase=\"2\">\n        <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Back\"\n          customClass=\"cqa-api-edit-step-actions-btn-back\" (clicked)=\"onBack()\">\n        </cqa-button>\n        <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"Next\"\n          customClass=\"cqa-api-edit-step-actions-btn-next\" (clicked)=\"onNext()\">\n        </cqa-button>\n      </ng-container>\n      <ng-container *ngSwitchCase=\"3\">\n        <cqa-button type=\"button\" variant=\"outlined\" btnSize=\"lg\" text=\"Back\"\n          customClass=\"cqa-api-edit-step-actions-btn-back\" (clicked)=\"onBack()\">\n        </cqa-button>\n        <cqa-button type=\"button\" variant=\"filled\" btnSize=\"lg\" text=\"Create\"\n          customClass=\"cqa-api-edit-step-actions-btn-create\" (clicked)=\"onCreate()\">\n        </cqa-button>\n      </ng-container>\n    </ng-container>\n  </div>\n</div>"]}
|