@cqa-lib/cqa-ui 1.1.183 → 1.1.185
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 +21 -9
- package/esm2020/lib/execution-screen/main-step-collapse/main-step-collapse.component.mjs +11 -2
- package/esm2020/lib/step-builder/step-builder-api/step-builder-api.component.mjs +510 -0
- package/esm2020/lib/step-builder/step-builder-custom-code/step-builder-custom-code.component.mjs +92 -7
- package/esm2020/lib/test-case-details/create-step-group/create-step-group.component.mjs +114 -0
- package/esm2020/lib/test-case-details/delete-steps/delete-steps.component.mjs +104 -0
- package/esm2020/lib/ui-kit.module.mjs +17 -2
- package/esm2020/public-api.mjs +4 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +847 -15
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +835 -14
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/add-prerequisite-cases-section/add-prerequisite-cases-section.component.d.ts +11 -3
- package/lib/execution-screen/main-step-collapse/main-step-collapse.component.d.ts +5 -3
- package/lib/step-builder/step-builder-api/step-builder-api.component.d.ts +115 -0
- package/lib/step-builder/step-builder-custom-code/step-builder-custom-code.component.d.ts +17 -3
- package/lib/test-case-details/create-step-group/create-step-group.component.d.ts +30 -0
- package/lib/test-case-details/delete-steps/delete-steps.component.d.ts +26 -0
- package/lib/ui-kit.module.d.ts +35 -32
- package/package.json +1 -1
- package/public-api.d.ts +3 -0
- package/styles.css +1 -1
package/esm2020/lib/step-builder/step-builder-custom-code/step-builder-custom-code.component.mjs
CHANGED
|
@@ -1,29 +1,109 @@
|
|
|
1
1
|
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
2
|
-
import { Validators } from '@angular/forms';
|
|
2
|
+
import { FormControl, Validators } from '@angular/forms';
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
4
|
import * as i1 from "@angular/forms";
|
|
5
5
|
import * as i2 from "../../dynamic-select/dynamic-select-field.component";
|
|
6
6
|
import * as i3 from "../../custom-textarea/custom-textarea.component";
|
|
7
7
|
import * as i4 from "../../custom-input/custom-input.component";
|
|
8
|
-
import * as i5 from "
|
|
8
|
+
import * as i5 from "@angular/material/slide-toggle";
|
|
9
|
+
import * as i6 from "../../button/button.component";
|
|
10
|
+
import * as i7 from "@angular/common";
|
|
9
11
|
export class StepBuilderCustomCodeComponent {
|
|
10
12
|
constructor(fb) {
|
|
11
13
|
this.fb = fb;
|
|
12
14
|
/** Options for language dropdown */
|
|
13
15
|
this.languageOptions = [];
|
|
16
|
+
/** Template to get variables from */
|
|
17
|
+
this.template = null;
|
|
18
|
+
/** Function to handle variable processing or custom logic. Can be passed from parent component. */
|
|
19
|
+
this.setTemplateVariables = () => { return []; };
|
|
14
20
|
/** Emit when step is created */
|
|
15
21
|
this.createStep = new EventEmitter();
|
|
16
22
|
/** Emit when cancelled */
|
|
17
23
|
this.cancelled = new EventEmitter();
|
|
24
|
+
this.templateVariables = [];
|
|
18
25
|
this.customCodeForm = this.fb.group({
|
|
19
26
|
language: ['', Validators.required],
|
|
20
27
|
code: ['', Validators.required],
|
|
21
28
|
metadata: [''],
|
|
22
29
|
description: ['']
|
|
23
30
|
});
|
|
31
|
+
this.variablesForm = this.fb.group({});
|
|
24
32
|
}
|
|
25
33
|
ngOnInit() {
|
|
26
|
-
//
|
|
34
|
+
// Initialize template variables if template is provided
|
|
35
|
+
if (this.template) {
|
|
36
|
+
this.loadTemplateVariables();
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
ngOnChanges(changes) {
|
|
40
|
+
if (changes['template'] && this.template) {
|
|
41
|
+
this.loadTemplateVariables();
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
loadTemplateVariables() {
|
|
45
|
+
if (this.template && this.setTemplateVariables) {
|
|
46
|
+
this.templateVariables = this.setTemplateVariables(this.template);
|
|
47
|
+
this.buildVariablesForm();
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
buildVariablesForm() {
|
|
51
|
+
// Clear existing form controls
|
|
52
|
+
Object.keys(this.variablesForm.controls).forEach(key => {
|
|
53
|
+
this.variablesForm.removeControl(key);
|
|
54
|
+
});
|
|
55
|
+
// Add form controls for each variable
|
|
56
|
+
this.templateVariables.forEach(variable => {
|
|
57
|
+
// Handle boolean variables - use boolean value, others use string
|
|
58
|
+
const defaultValue = variable.type === 'boolean'
|
|
59
|
+
? (variable.value === true || variable.value === 'true' || variable.value === 1)
|
|
60
|
+
: (variable.value || '');
|
|
61
|
+
this.variablesForm.addControl(variable.name, new FormControl(defaultValue));
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
getSelectConfig(variable) {
|
|
65
|
+
const options = (variable.options || []).map((opt) => ({
|
|
66
|
+
id: opt,
|
|
67
|
+
value: opt,
|
|
68
|
+
name: opt,
|
|
69
|
+
label: opt
|
|
70
|
+
}));
|
|
71
|
+
return {
|
|
72
|
+
key: variable.name,
|
|
73
|
+
placeholder: `Select ${variable.label}`,
|
|
74
|
+
multiple: false,
|
|
75
|
+
searchable: false,
|
|
76
|
+
options: options,
|
|
77
|
+
onChange: (value) => {
|
|
78
|
+
this.onVariableValueChange(variable.name, value);
|
|
79
|
+
}
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
onVariableValueChange(variableName, value) {
|
|
83
|
+
// Update the variable in templateVariables array
|
|
84
|
+
const variable = this.templateVariables.find(v => v.name === variableName);
|
|
85
|
+
if (variable) {
|
|
86
|
+
variable.value = value;
|
|
87
|
+
}
|
|
88
|
+
// Also update form control
|
|
89
|
+
if (this.variablesForm.get(variableName)) {
|
|
90
|
+
this.variablesForm.get(variableName)?.setValue(value);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
onVariableBooleanChange(variableName, value) {
|
|
94
|
+
// Update the variable in templateVariables array
|
|
95
|
+
const variable = this.templateVariables.find(v => v.name === variableName);
|
|
96
|
+
if (variable) {
|
|
97
|
+
variable.value = value;
|
|
98
|
+
}
|
|
99
|
+
// Also update form control
|
|
100
|
+
if (this.variablesForm.get(variableName)) {
|
|
101
|
+
this.variablesForm.get(variableName)?.setValue(value);
|
|
102
|
+
}
|
|
103
|
+
else {
|
|
104
|
+
// Create form control if it doesn't exist
|
|
105
|
+
this.variablesForm.addControl(variableName, new FormControl(value));
|
|
106
|
+
}
|
|
27
107
|
}
|
|
28
108
|
getLanguageConfig() {
|
|
29
109
|
return {
|
|
@@ -44,22 +124,27 @@ export class StepBuilderCustomCodeComponent {
|
|
|
44
124
|
language: formValue.language || '',
|
|
45
125
|
code: formValue.code || '',
|
|
46
126
|
metadata: formValue.metadata || '',
|
|
47
|
-
description: formValue.description || ''
|
|
127
|
+
description: formValue.description || '',
|
|
128
|
+
templateVariables: this.templateVariables.length > 0 ? this.templateVariables : undefined
|
|
48
129
|
};
|
|
49
130
|
this.createStep.emit(stepData);
|
|
50
131
|
}
|
|
51
132
|
}
|
|
52
133
|
}
|
|
53
134
|
StepBuilderCustomCodeComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepBuilderCustomCodeComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component });
|
|
54
|
-
StepBuilderCustomCodeComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: StepBuilderCustomCodeComponent, selector: "cqa-step-builder-custom-code", inputs: { languageOptions: "languageOptions" }, outputs: { createStep: "createStep", cancelled: "cancelled" }, host: { classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n Custom Code Step\n </h2>\n\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-max-h-[500px] cqa-overflow-y-auto\">\n <!-- Language Dropdown -->\n <div class=\"cqa-mb-3\">\n <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n Language\n </label>\n <cqa-dynamic-select class=\"cqa-w-full\" [form]=\"customCodeForm\" [config]=\"getLanguageConfig()\">\n </cqa-dynamic-select>\n </div>\n\n <!-- Code Textarea -->\n <div class=\"cqa-mb-3\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1 cqa-block\">\n Code\n </label>\n <cqa-custom-textarea\n class=\"cqa-step-builder-custom-code-textarea\"\n [placeholder]=\"'// Write your code here...'\"\n [value]=\"customCodeForm.get('code')?.value\"\n [fullWidth]=\"true\"\n [rows]=\"4\"\n (valueChange)=\"customCodeForm.get('code')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n\n <div class=\"cqa-flex cqa-flex-wrap cqa-custom-form-fields\">\n <!-- Metadata Input -->\n <div class=\"cqa-mb-2 cqa-w-1/2 cqa-pr-2\">\n <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n Metadata\n </label>\n <cqa-custom-input\n [placeholder]=\"'Text Input'\"\n [value]=\"customCodeForm.get('metadata')?.value\"\n [fullWidth]=\"true\"\n (valueChange)=\"customCodeForm.get('metadata')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- Description Input -->\n <div class=\"cqa-w-1/2 cqa-pl-2\">\n <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n Description\n </label>\n <cqa-custom-input\n [placeholder]=\"'Text Input'\"\n [value]=\"customCodeForm.get('description')?.value\"\n [fullWidth]=\"true\"\n (valueChange)=\"customCodeForm.get('description')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n </div>\n </div>\n\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n <cqa-button class=\"cqa-w-1/2 cqa-rounded-[10px]\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button class=\"cqa-border-solid cqa-rounded-[9px] cqa-w-1/2 cqa-border cqa-border-[#3F43EE]\" variant=\"filled\" text=\"Create Step\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>\n\n", components: [{ type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore"] }, { type: i3.CustomTextareaComponent, selector: "cqa-custom-textarea", inputs: ["label", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "rows", "cols", "resize", "textareaInlineStyle", "labelInlineStyle", "customClass"], outputs: ["valueChange", "blurred", "focused"] }, { type: 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.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }] });
|
|
135
|
+
StepBuilderCustomCodeComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: StepBuilderCustomCodeComponent, selector: "cqa-step-builder-custom-code", inputs: { languageOptions: "languageOptions", template: "template", setTemplateVariables: "setTemplateVariables" }, outputs: { createStep: "createStep", cancelled: "cancelled" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n Custom Code Step\n </h2>\n\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-max-h-[500px] cqa-overflow-y-auto\">\n \n\n <!-- Language Dropdown -->\n <div class=\"cqa-mb-3\">\n <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n Language\n </label>\n <cqa-dynamic-select class=\"cqa-w-full\" [form]=\"customCodeForm\" [config]=\"getLanguageConfig()\">\n </cqa-dynamic-select>\n </div>\n\n <!-- Code Textarea -->\n <div class=\"cqa-mb-3\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1 cqa-block\">\n Code\n </label>\n <cqa-custom-textarea\n class=\"cqa-step-builder-custom-code-textarea\"\n [placeholder]=\"'// Write your code here...'\"\n [value]=\"customCodeForm.get('code')?.value\"\n [fullWidth]=\"true\"\n [rows]=\"4\"\n (valueChange)=\"customCodeForm.get('code')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n\n <div class=\"cqa-flex cqa-flex-wrap cqa-custom-form-fields\">\n <!-- Metadata Input -->\n <div class=\"cqa-mb-2 cqa-w-1/2 cqa-pr-2\">\n <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n Metadata\n </label>\n <cqa-custom-input\n [placeholder]=\"'Text Input'\"\n [value]=\"customCodeForm.get('metadata')?.value\"\n [fullWidth]=\"true\"\n (valueChange)=\"customCodeForm.get('metadata')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- Description Input -->\n <div class=\"cqa-w-1/2 cqa-pl-2\">\n <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n Description\n </label>\n <cqa-custom-input\n [placeholder]=\"'Text Input'\"\n [value]=\"customCodeForm.get('description')?.value\"\n [fullWidth]=\"true\"\n (valueChange)=\"customCodeForm.get('description')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n </div>\n\n <!-- Template Variables Section -->\n <div *ngIf=\"templateVariables && templateVariables.length > 0\" class=\"cqa-mb-4\">\n <!-- Template Variables Form Fields -->\n <div class=\"cqa-flex cqa-gap-x-6 cqa-flex-wrap cqa-mb-4\">\n <ng-container *ngFor=\"let variable of templateVariables\">\n <!-- Boolean variables with mat-slide-toggle -->\n <ng-container *ngIf=\"variable.type === 'boolean'\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\" style=\"width: calc(50% - 12px);\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">\n {{ variable.label }}\n </label>\n <mat-slide-toggle\n [checked]=\"variablesForm.get(variable.name)?.value || variable.value || false\"\n (change)=\"onVariableBooleanChange(variable.name, $event.checked)\"\n color=\"primary\">\n </mat-slide-toggle>\n </div>\n </ng-container>\n \n <!-- Non-boolean, non-custom_code variables -->\n <ng-container *ngIf=\"variable.name !== 'custom_code' && variable.type !== 'boolean'\">\n <ng-container *ngIf=\"(variable.name === 'type' || variable.name === 'scrollTo'); else defaultInput\">\n <div class=\"cqa-flex cqa-flex-col\" style=\"width: calc(50% - 12px);\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1\">\n {{ variable.label }}\n </label>\n <cqa-dynamic-select [form]=\"variablesForm\" [config]=\"getSelectConfig(variable)\">\n </cqa-dynamic-select>\n </div>\n </ng-container>\n <ng-template #defaultInput>\n <div class=\"cqa-flex cqa-flex-col\" style=\"width: calc(50% - 12px);\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1\">\n {{ variable.label }}\n </label>\n <cqa-custom-input [placeholder]=\"'Text Input'\" [value]=\"variable.value\" [fullWidth]=\"true\"\n (valueChange)=\"onVariableValueChange(variable.name, $event)\">\n </cqa-custom-input>\n </div>\n </ng-template>\n </ng-container>\n </ng-container>\n </div>\n </div>\n </div>\n\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n <cqa-button class=\"cqa-w-1/2 cqa-rounded-[10px]\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button class=\"cqa-border-solid cqa-rounded-[9px] cqa-w-1/2 cqa-border cqa-border-[#3F43EE]\" variant=\"filled\" text=\"Create Step\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>\n\n", components: [{ type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore"] }, { type: i3.CustomTextareaComponent, selector: "cqa-custom-textarea", inputs: ["label", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "rows", "cols", "resize", "textareaInlineStyle", "labelInlineStyle", "customClass"], outputs: ["valueChange", "blurred", "focused"] }, { type: 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.MatSlideToggle, selector: "mat-slide-toggle", inputs: ["disabled", "disableRipple", "color", "tabIndex", "name", "id", "labelPosition", "aria-label", "aria-labelledby", "aria-describedby", "required", "checked"], outputs: ["change", "toggleChange"], exportAs: ["matSlideToggle"] }, { type: i6.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }], directives: [{ type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
|
|
55
136
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepBuilderCustomCodeComponent, decorators: [{
|
|
56
137
|
type: Component,
|
|
57
|
-
args: [{ selector: 'cqa-step-builder-custom-code', host: { class: 'cqa-ui-root' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n Custom Code Step\n </h2>\n\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-max-h-[500px] cqa-overflow-y-auto\">\n <!-- Language Dropdown -->\n <div class=\"cqa-mb-3\">\n <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n Language\n </label>\n <cqa-dynamic-select class=\"cqa-w-full\" [form]=\"customCodeForm\" [config]=\"getLanguageConfig()\">\n </cqa-dynamic-select>\n </div>\n\n <!-- Code Textarea -->\n <div class=\"cqa-mb-3\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1 cqa-block\">\n Code\n </label>\n <cqa-custom-textarea\n class=\"cqa-step-builder-custom-code-textarea\"\n [placeholder]=\"'// Write your code here...'\"\n [value]=\"customCodeForm.get('code')?.value\"\n [fullWidth]=\"true\"\n [rows]=\"4\"\n (valueChange)=\"customCodeForm.get('code')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n\n <div class=\"cqa-flex cqa-flex-wrap cqa-custom-form-fields\">\n <!-- Metadata Input -->\n <div class=\"cqa-mb-2 cqa-w-1/2 cqa-pr-2\">\n <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n Metadata\n </label>\n <cqa-custom-input\n [placeholder]=\"'Text Input'\"\n [value]=\"customCodeForm.get('metadata')?.value\"\n [fullWidth]=\"true\"\n (valueChange)=\"customCodeForm.get('metadata')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- Description Input -->\n <div class=\"cqa-w-1/2 cqa-pl-2\">\n <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n Description\n </label>\n <cqa-custom-input\n [placeholder]=\"'Text Input'\"\n [value]=\"customCodeForm.get('description')?.value\"\n [fullWidth]=\"true\"\n (valueChange)=\"customCodeForm.get('description')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n </div>\n </div>\n\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n <cqa-button class=\"cqa-w-1/2 cqa-rounded-[10px]\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button class=\"cqa-border-solid cqa-rounded-[9px] cqa-w-1/2 cqa-border cqa-border-[#3F43EE]\" variant=\"filled\" text=\"Create Step\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>\n\n", styles: [] }]
|
|
138
|
+
args: [{ selector: 'cqa-step-builder-custom-code', host: { class: 'cqa-ui-root' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n Custom Code Step\n </h2>\n\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-max-h-[500px] cqa-overflow-y-auto\">\n \n\n <!-- Language Dropdown -->\n <div class=\"cqa-mb-3\">\n <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n Language\n </label>\n <cqa-dynamic-select class=\"cqa-w-full\" [form]=\"customCodeForm\" [config]=\"getLanguageConfig()\">\n </cqa-dynamic-select>\n </div>\n\n <!-- Code Textarea -->\n <div class=\"cqa-mb-3\">\n <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1 cqa-block\">\n Code\n </label>\n <cqa-custom-textarea\n class=\"cqa-step-builder-custom-code-textarea\"\n [placeholder]=\"'// Write your code here...'\"\n [value]=\"customCodeForm.get('code')?.value\"\n [fullWidth]=\"true\"\n [rows]=\"4\"\n (valueChange)=\"customCodeForm.get('code')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n\n <div class=\"cqa-flex cqa-flex-wrap cqa-custom-form-fields\">\n <!-- Metadata Input -->\n <div class=\"cqa-mb-2 cqa-w-1/2 cqa-pr-2\">\n <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n Metadata\n </label>\n <cqa-custom-input\n [placeholder]=\"'Text Input'\"\n [value]=\"customCodeForm.get('metadata')?.value\"\n [fullWidth]=\"true\"\n (valueChange)=\"customCodeForm.get('metadata')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- Description Input -->\n <div class=\"cqa-w-1/2 cqa-pl-2\">\n <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n Description\n </label>\n <cqa-custom-input\n [placeholder]=\"'Text Input'\"\n [value]=\"customCodeForm.get('description')?.value\"\n [fullWidth]=\"true\"\n (valueChange)=\"customCodeForm.get('description')?.setValue($event)\">\n </cqa-custom-input>\n </div>\n </div>\n\n <!-- Template Variables Section -->\n <div *ngIf=\"templateVariables && templateVariables.length > 0\" class=\"cqa-mb-4\">\n <!-- Template Variables Form Fields -->\n <div class=\"cqa-flex cqa-gap-x-6 cqa-flex-wrap cqa-mb-4\">\n <ng-container *ngFor=\"let variable of templateVariables\">\n <!-- Boolean variables with mat-slide-toggle -->\n <ng-container *ngIf=\"variable.type === 'boolean'\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\" style=\"width: calc(50% - 12px);\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">\n {{ variable.label }}\n </label>\n <mat-slide-toggle\n [checked]=\"variablesForm.get(variable.name)?.value || variable.value || false\"\n (change)=\"onVariableBooleanChange(variable.name, $event.checked)\"\n color=\"primary\">\n </mat-slide-toggle>\n </div>\n </ng-container>\n \n <!-- Non-boolean, non-custom_code variables -->\n <ng-container *ngIf=\"variable.name !== 'custom_code' && variable.type !== 'boolean'\">\n <ng-container *ngIf=\"(variable.name === 'type' || variable.name === 'scrollTo'); else defaultInput\">\n <div class=\"cqa-flex cqa-flex-col\" style=\"width: calc(50% - 12px);\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1\">\n {{ variable.label }}\n </label>\n <cqa-dynamic-select [form]=\"variablesForm\" [config]=\"getSelectConfig(variable)\">\n </cqa-dynamic-select>\n </div>\n </ng-container>\n <ng-template #defaultInput>\n <div class=\"cqa-flex cqa-flex-col\" style=\"width: calc(50% - 12px);\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1\">\n {{ variable.label }}\n </label>\n <cqa-custom-input [placeholder]=\"'Text Input'\" [value]=\"variable.value\" [fullWidth]=\"true\"\n (valueChange)=\"onVariableValueChange(variable.name, $event)\">\n </cqa-custom-input>\n </div>\n </ng-template>\n </ng-container>\n </ng-container>\n </div>\n </div>\n </div>\n\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n <cqa-button class=\"cqa-w-1/2 cqa-rounded-[10px]\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button class=\"cqa-border-solid cqa-rounded-[9px] cqa-w-1/2 cqa-border cqa-border-[#3F43EE]\" variant=\"filled\" text=\"Create Step\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>\n\n", styles: [] }]
|
|
58
139
|
}], ctorParameters: function () { return [{ type: i1.FormBuilder }]; }, propDecorators: { languageOptions: [{
|
|
59
140
|
type: Input
|
|
141
|
+
}], template: [{
|
|
142
|
+
type: Input
|
|
143
|
+
}], setTemplateVariables: [{
|
|
144
|
+
type: Input
|
|
60
145
|
}], createStep: [{
|
|
61
146
|
type: Output
|
|
62
147
|
}], cancelled: [{
|
|
63
148
|
type: Output
|
|
64
149
|
}] } });
|
|
65
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"step-builder-custom-code.component.js","sourceRoot":"","sources":["../../../../../../src/lib/step-builder/step-builder-custom-code/step-builder-custom-code.component.ts","../../../../../../src/lib/step-builder/step-builder-custom-code/step-builder-custom-code.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,EAAE,YAAY,EAAU,MAAM,eAAe,CAAC;AAC/E,OAAO,EAA0B,UAAU,EAAE,MAAM,gBAAgB,CAAC;;;;;;;AAgBpE,MAAM,OAAO,8BAA8B;IAYzC,YAAoB,EAAe;QAAf,OAAE,GAAF,EAAE,CAAa;QAXnC,oCAAoC;QAC3B,oBAAe,GAAmB,EAAE,CAAC;QAE9C,gCAAgC;QACtB,eAAU,GAAG,IAAI,YAAY,EAAsB,CAAC;QAE9D,0BAA0B;QAChB,cAAS,GAAG,IAAI,YAAY,EAAQ,CAAC;QAK7C,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,EAAE,CAAC,KAAK,CAAC;YAClC,QAAQ,EAAE,CAAC,EAAE,EAAE,UAAU,CAAC,QAAQ,CAAC;YACnC,IAAI,EAAE,CAAC,EAAE,EAAE,UAAU,CAAC,QAAQ,CAAC;YAC/B,QAAQ,EAAE,CAAC,EAAE,CAAC;YACd,WAAW,EAAE,CAAC,EAAE,CAAC;SAClB,CAAC,CAAC;IACL,CAAC;IAED,QAAQ;QACN,2BAA2B;IAC7B,CAAC;IAED,iBAAiB;QACf,OAAO;YACL,GAAG,EAAE,UAAU;YACf,WAAW,EAAE,KAAK;YAClB,QAAQ,EAAE,KAAK;YACf,UAAU,EAAE,KAAK;YACjB,OAAO,EAAE,IAAI,CAAC,eAAe;SAC9B,CAAC;IACJ,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;IACxB,CAAC;IAED,YAAY;QACV,IAAI,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE;YAC7B,MAAM,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC;YAC5C,MAAM,QAAQ,GAAuB;gBACnC,QAAQ,EAAE,SAAS,CAAC,QAAQ,IAAI,EAAE;gBAClC,IAAI,EAAE,SAAS,CAAC,IAAI,IAAI,EAAE;gBAC1B,QAAQ,EAAE,SAAS,CAAC,QAAQ,IAAI,EAAE;gBAClC,WAAW,EAAE,SAAS,CAAC,WAAW,IAAI,EAAE;aACzC,CAAC;YACF,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SAChC;IACH,CAAC;;2HAlDU,8BAA8B;+GAA9B,8BAA8B,4NCjB3C,s2FAuEA;2FDtDa,8BAA8B;kBAN1C,SAAS;+BACE,8BAA8B,QAGlC,EAAE,KAAK,EAAE,aAAa,EAAE;kGAIrB,eAAe;sBAAvB,KAAK;gBAGI,UAAU;sBAAnB,MAAM;gBAGG,SAAS;sBAAlB,MAAM","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit } from '@angular/core';\nimport { FormBuilder, FormGroup, Validators } from '@angular/forms';\nimport { DynamicSelectFieldConfig, SelectOption } from '../../dynamic-select/dynamic-select-field.component';\n\nexport interface CustomCodeFormData {\n  language: string;\n  code: string;\n  metadata?: string;\n  description?: string;\n}\n\n@Component({\n  selector: 'cqa-step-builder-custom-code',\n  templateUrl: './step-builder-custom-code.component.html',\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' }\n})\nexport class StepBuilderCustomCodeComponent implements OnInit {\n  /** Options for language dropdown */\n  @Input() languageOptions: SelectOption[] = [];\n\n  /** Emit when step is created */\n  @Output() createStep = new EventEmitter<CustomCodeFormData>();\n\n  /** Emit when cancelled */\n  @Output() cancelled = new EventEmitter<void>();\n\n  customCodeForm: FormGroup;\n\n  constructor(private fb: FormBuilder) {\n    this.customCodeForm = this.fb.group({\n      language: ['', Validators.required],\n      code: ['', Validators.required],\n      metadata: [''],\n      description: ['']\n    });\n  }\n\n  ngOnInit(): void {\n    // Component initialization\n  }\n\n  getLanguageConfig(): DynamicSelectFieldConfig {\n    return {\n      key: 'language',\n      placeholder: '...',\n      multiple: false,\n      searchable: false,\n      options: this.languageOptions\n    };\n  }\n\n  onCancel(): void {\n    this.cancelled.emit();\n  }\n\n  onCreateStep(): void {\n    if (this.customCodeForm.valid) {\n      const formValue = this.customCodeForm.value;\n      const stepData: CustomCodeFormData = {\n        language: formValue.language || '',\n        code: formValue.code || '',\n        metadata: formValue.metadata || '',\n        description: formValue.description || ''\n      };\n      this.createStep.emit(stepData);\n    }\n  }\n}\n\n","<div class=\"cqa-flex cqa-flex-col cqa-bg-white cqa-px-4 cqa-py-2\">\n  <!-- Header -->\n  <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n    Custom Code Step\n  </h2>\n\n  <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-max-h-[500px] cqa-overflow-y-auto\">\n    <!-- Language Dropdown -->\n    <div class=\"cqa-mb-3\">\n      <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n        Language\n      </label>\n      <cqa-dynamic-select class=\"cqa-w-full\" [form]=\"customCodeForm\" [config]=\"getLanguageConfig()\">\n      </cqa-dynamic-select>\n    </div>\n\n    <!-- Code Textarea -->\n    <div class=\"cqa-mb-3\">\n      <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1 cqa-block\">\n        Code\n      </label>\n      <cqa-custom-textarea\n        class=\"cqa-step-builder-custom-code-textarea\"\n        [placeholder]=\"'// Write your code here...'\"\n        [value]=\"customCodeForm.get('code')?.value\"\n        [fullWidth]=\"true\"\n        [rows]=\"4\"\n        (valueChange)=\"customCodeForm.get('code')?.setValue($event)\">\n      </cqa-custom-textarea>\n    </div>\n\n    <div class=\"cqa-flex cqa-flex-wrap cqa-custom-form-fields\">\n      <!-- Metadata Input -->\n      <div class=\"cqa-mb-2 cqa-w-1/2 cqa-pr-2\">\n        <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n          Metadata\n        </label>\n        <cqa-custom-input\n          [placeholder]=\"'Text Input'\"\n          [value]=\"customCodeForm.get('metadata')?.value\"\n          [fullWidth]=\"true\"\n          (valueChange)=\"customCodeForm.get('metadata')?.setValue($event)\">\n        </cqa-custom-input>\n      </div>\n\n      <!-- Description Input -->\n      <div class=\"cqa-w-1/2 cqa-pl-2\">\n        <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n          Description\n        </label>\n        <cqa-custom-input\n          [placeholder]=\"'Text Input'\"\n          [value]=\"customCodeForm.get('description')?.value\"\n          [fullWidth]=\"true\"\n          (valueChange)=\"customCodeForm.get('description')?.setValue($event)\">\n        </cqa-custom-input>\n      </div>\n    </div>\n  </div>\n\n  <!-- Action Buttons -->\n  <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n    <cqa-button class=\"cqa-w-1/2 cqa-rounded-[10px]\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCancel()\">\n    </cqa-button>\n    <cqa-button class=\"cqa-border-solid cqa-rounded-[9px] cqa-w-1/2 cqa-border cqa-border-[#3F43EE]\" variant=\"filled\" text=\"Create Step\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCreateStep()\">\n    </cqa-button>\n  </div>\n</div>\n\n"]}
|
|
150
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"step-builder-custom-code.component.js","sourceRoot":"","sources":["../../../../../../src/lib/step-builder/step-builder-custom-code/step-builder-custom-code.component.ts","../../../../../../src/lib/step-builder/step-builder-custom-code/step-builder-custom-code.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, OnChanges, SimpleChanges } from '@angular/core';\nimport { FormBuilder, FormGroup, FormControl, Validators } from '@angular/forms';\nimport { DynamicSelectFieldConfig, SelectOption } from '../../dynamic-select/dynamic-select-field.component';\nimport { ActionTemplate } from '../step-builder-action/step-builder-action.component';\n\nexport interface CustomCodeFormData {\n  language: string;\n  code: string;\n  metadata?: string;\n  description?: string;\n  templateVariables?: any[];\n}\n\n@Component({\n  selector: 'cqa-step-builder-custom-code',\n  templateUrl: './step-builder-custom-code.component.html',\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' }\n})\nexport class StepBuilderCustomCodeComponent implements OnInit, OnChanges {\n  /** Options for language dropdown */\n  @Input() languageOptions: SelectOption[] = [];\n\n  /** Template to get variables from */\n  @Input() template: ActionTemplate | null = null;\n  \n  /** Function to handle variable processing or custom logic. Can be passed from parent component. */\n  @Input() setTemplateVariables: (variables: ActionTemplate) => any = () => { return []};\n  \n  /** Emit when step is created */\n  @Output() createStep = new EventEmitter<CustomCodeFormData>();\n\n  /** Emit when cancelled */\n  @Output() cancelled = new EventEmitter<void>();\n\n  customCodeForm: FormGroup;\n  templateVariables: any[] = [];\n  variablesForm: FormGroup;\n\n  constructor(private fb: FormBuilder) {\n    this.customCodeForm = this.fb.group({\n      language: ['', Validators.required],\n      code: ['', Validators.required],\n      metadata: [''],\n      description: ['']\n    });\n    this.variablesForm = this.fb.group({});\n  }\n\n  ngOnInit(): void {\n    // Initialize template variables if template is provided\n    if (this.template) {\n      this.loadTemplateVariables();\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['template'] && this.template) {\n      this.loadTemplateVariables();\n    }\n  }\n\n  private loadTemplateVariables(): void {\n    if (this.template && this.setTemplateVariables) {\n      this.templateVariables = this.setTemplateVariables(this.template);\n      this.buildVariablesForm();\n    }\n  }\n\n  private buildVariablesForm(): void {\n    // Clear existing form controls\n    Object.keys(this.variablesForm.controls).forEach(key => {\n      this.variablesForm.removeControl(key);\n    });\n\n    // Add form controls for each variable\n    this.templateVariables.forEach(variable => {\n      // Handle boolean variables - use boolean value, others use string\n      const defaultValue = variable.type === 'boolean' \n        ? (variable.value === true || variable.value === 'true' || variable.value === 1)\n        : (variable.value || '');\n      this.variablesForm.addControl(variable.name, new FormControl(defaultValue));\n    });\n  }\n\n  getSelectConfig(variable: any): DynamicSelectFieldConfig {\n    const options: SelectOption[] = (variable.options || []).map((opt: string) => ({\n      id: opt,\n      value: opt,\n      name: opt,\n      label: opt\n    }));\n\n    return {\n      key: variable.name,\n      placeholder: `Select ${variable.label}`,\n      multiple: false,\n      searchable: false,\n      options: options,\n      onChange: (value: any) => {\n        this.onVariableValueChange(variable.name, value);\n      }\n    };\n  }\n\n  onVariableValueChange(variableName: string, value: any): void {\n    // Update the variable in templateVariables array\n    const variable = this.templateVariables.find(v => v.name === variableName);\n    if (variable) {\n      variable.value = value;\n    }\n    // Also update form control\n    if (this.variablesForm.get(variableName)) {\n      this.variablesForm.get(variableName)?.setValue(value);\n    }\n  }\n\n  onVariableBooleanChange(variableName: string, value: boolean): void {\n    // Update the variable in templateVariables array\n    const variable = this.templateVariables.find(v => v.name === variableName);\n    if (variable) {\n      variable.value = value;\n    }\n    // Also update form control\n    if (this.variablesForm.get(variableName)) {\n      this.variablesForm.get(variableName)?.setValue(value);\n    } else {\n      // Create form control if it doesn't exist\n      this.variablesForm.addControl(variableName, new FormControl(value));\n    }\n  }\n\n  getLanguageConfig(): DynamicSelectFieldConfig {\n    return {\n      key: 'language',\n      placeholder: '...',\n      multiple: false,\n      searchable: false,\n      options: this.languageOptions\n    };\n  }\n\n  onCancel(): void {\n    this.cancelled.emit();\n  }\n\n  onCreateStep(): void {\n    if (this.customCodeForm.valid) {\n      const formValue = this.customCodeForm.value;\n      const stepData: CustomCodeFormData = {\n        language: formValue.language || '',\n        code: formValue.code || '',\n        metadata: formValue.metadata || '',\n        description: formValue.description || '',\n        templateVariables: this.templateVariables.length > 0 ? this.templateVariables : undefined\n      };\n      this.createStep.emit(stepData);\n    }\n  }\n}\n\n","<div class=\"cqa-flex cqa-flex-col cqa-bg-white cqa-px-4 cqa-py-2\">\n  <!-- Header -->\n  <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n    Custom Code Step\n  </h2>\n\n  <div class=\"cqa-flex cqa-flex-col cqa-flex-1 cqa-max-h-[500px] cqa-overflow-y-auto\">\n   \n\n    <!-- Language Dropdown -->\n    <div class=\"cqa-mb-3\">\n      <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n        Language\n      </label>\n      <cqa-dynamic-select class=\"cqa-w-full\" [form]=\"customCodeForm\" [config]=\"getLanguageConfig()\">\n      </cqa-dynamic-select>\n    </div>\n\n    <!-- Code Textarea -->\n    <div class=\"cqa-mb-3\">\n      <label class=\"cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1 cqa-block\">\n        Code\n      </label>\n      <cqa-custom-textarea\n        class=\"cqa-step-builder-custom-code-textarea\"\n        [placeholder]=\"'// Write your code here...'\"\n        [value]=\"customCodeForm.get('code')?.value\"\n        [fullWidth]=\"true\"\n        [rows]=\"4\"\n        (valueChange)=\"customCodeForm.get('code')?.setValue($event)\">\n      </cqa-custom-textarea>\n    </div>\n\n    <div class=\"cqa-flex cqa-flex-wrap cqa-custom-form-fields\">\n      <!-- Metadata Input -->\n      <div class=\"cqa-mb-2 cqa-w-1/2 cqa-pr-2\">\n        <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n          Metadata\n        </label>\n        <cqa-custom-input\n          [placeholder]=\"'Text Input'\"\n          [value]=\"customCodeForm.get('metadata')?.value\"\n          [fullWidth]=\"true\"\n          (valueChange)=\"customCodeForm.get('metadata')?.setValue($event)\">\n        </cqa-custom-input>\n      </div>\n\n      <!-- Description Input -->\n      <div class=\"cqa-w-1/2 cqa-pl-2\">\n        <label class=\"cqa-leading-[100%] cqa-block cqa-text-[12px] cqa-font-medium cqa-text-[#161617] cqa-mb-1\">\n          Description\n        </label>\n        <cqa-custom-input\n          [placeholder]=\"'Text Input'\"\n          [value]=\"customCodeForm.get('description')?.value\"\n          [fullWidth]=\"true\"\n          (valueChange)=\"customCodeForm.get('description')?.setValue($event)\">\n        </cqa-custom-input>\n      </div>\n    </div>\n\n     <!-- Template Variables Section -->\n     <div *ngIf=\"templateVariables && templateVariables.length > 0\" class=\"cqa-mb-4\">\n      <!-- Template Variables Form Fields -->\n      <div class=\"cqa-flex cqa-gap-x-6 cqa-flex-wrap cqa-mb-4\">\n        <ng-container *ngFor=\"let variable of templateVariables\">\n          <!-- Boolean variables with mat-slide-toggle -->\n          <ng-container *ngIf=\"variable.type === 'boolean'\">\n            <div class=\"cqa-flex cqa-items-center cqa-gap-2\" style=\"width: calc(50% - 12px);\">\n              <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">\n                {{ variable.label }}\n              </label>\n              <mat-slide-toggle\n                [checked]=\"variablesForm.get(variable.name)?.value || variable.value || false\"\n                (change)=\"onVariableBooleanChange(variable.name, $event.checked)\"\n                color=\"primary\">\n              </mat-slide-toggle>\n            </div>\n          </ng-container>\n          \n          <!-- Non-boolean, non-custom_code variables -->\n          <ng-container *ngIf=\"variable.name !== 'custom_code' && variable.type !== 'boolean'\">\n            <ng-container *ngIf=\"(variable.name === 'type' || variable.name === 'scrollTo'); else defaultInput\">\n              <div class=\"cqa-flex cqa-flex-col\" style=\"width: calc(50% - 12px);\">\n                <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1\">\n                  {{ variable.label }}\n                </label>\n                <cqa-dynamic-select [form]=\"variablesForm\" [config]=\"getSelectConfig(variable)\">\n                </cqa-dynamic-select>\n              </div>\n            </ng-container>\n            <ng-template #defaultInput>\n              <div class=\"cqa-flex cqa-flex-col\" style=\"width: calc(50% - 12px);\">\n                <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1\">\n                  {{ variable.label }}\n                </label>\n                <cqa-custom-input [placeholder]=\"'Text Input'\" [value]=\"variable.value\" [fullWidth]=\"true\"\n                  (valueChange)=\"onVariableValueChange(variable.name, $event)\">\n                </cqa-custom-input>\n              </div>\n            </ng-template>\n          </ng-container>\n        </ng-container>\n      </div>\n    </div>\n  </div>\n\n  <!-- Action Buttons -->\n  <div class=\"cqa-flex cqa-w-full cqa-gap-2 cqa-mt-auto cqa-pt-4 cqa-border-t cqa-border-gray-200\">\n    <cqa-button class=\"cqa-w-1/2 cqa-rounded-[10px]\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCancel()\">\n    </cqa-button>\n    <cqa-button class=\"cqa-border-solid cqa-rounded-[9px] cqa-w-1/2 cqa-border cqa-border-[#3F43EE]\" variant=\"filled\" text=\"Create Step\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCreateStep()\">\n    </cqa-button>\n  </div>\n</div>\n\n"]}
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
2
|
+
import { Validators } from '@angular/forms';
|
|
3
|
+
import { isNormalStepConfig, } from '../test-case-step.models';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@angular/forms";
|
|
6
|
+
import * as i2 from "../../custom-input/custom-input.component";
|
|
7
|
+
import * as i3 from "../../button/button.component";
|
|
8
|
+
import * as i4 from "@angular/common";
|
|
9
|
+
/**
|
|
10
|
+
* Create Step Group panel/modal for Test Case Details.
|
|
11
|
+
* Follows the same structure and validation pattern as Test Data Modal and Loop Step.
|
|
12
|
+
* Displays selected steps and allows naming the new group; emits createGroup with group name
|
|
13
|
+
* so the host can create the step group and replace selected steps with it.
|
|
14
|
+
*/
|
|
15
|
+
export class CreateStepGroupComponent {
|
|
16
|
+
constructor(fb) {
|
|
17
|
+
this.fb = fb;
|
|
18
|
+
this.stepsToGroup = [];
|
|
19
|
+
this.createGroup = new EventEmitter();
|
|
20
|
+
this.cancelled = new EventEmitter();
|
|
21
|
+
this.form = this.fb.group({
|
|
22
|
+
groupName: ['', [Validators.required, Validators.minLength(1)]],
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
get stepsCount() {
|
|
26
|
+
return this.stepsToGroup?.length ?? 0;
|
|
27
|
+
}
|
|
28
|
+
get subtitleText() {
|
|
29
|
+
const n = this.stepsCount;
|
|
30
|
+
return n === 1
|
|
31
|
+
? 'Group 1 selected step into a container'
|
|
32
|
+
: `Group ${n} selected steps into a container`;
|
|
33
|
+
}
|
|
34
|
+
/** Display label for a step in the "Steps to group" list (same pattern as normal-step display). */
|
|
35
|
+
getStepDisplayLabel(step, index) {
|
|
36
|
+
if (isNormalStepConfig(step)) {
|
|
37
|
+
return this.getNormalStepLabel(step);
|
|
38
|
+
}
|
|
39
|
+
if ('groupName' in step && step.groupName) {
|
|
40
|
+
return step.groupName;
|
|
41
|
+
}
|
|
42
|
+
if ('condition' in step && step.condition) {
|
|
43
|
+
return step.condition;
|
|
44
|
+
}
|
|
45
|
+
if ('loopType' in step) {
|
|
46
|
+
const loop = step;
|
|
47
|
+
if (loop.loopType === 'for' && loop.testDataProfile) {
|
|
48
|
+
return `For loop: ${loop.testDataProfile}`;
|
|
49
|
+
}
|
|
50
|
+
if (loop.loopType === 'while' && loop.condition) {
|
|
51
|
+
return `While: ${loop.condition}`;
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
if (step.description) {
|
|
55
|
+
return step.description;
|
|
56
|
+
}
|
|
57
|
+
return `Step ${index + 1}`;
|
|
58
|
+
}
|
|
59
|
+
getNormalStepLabel(step) {
|
|
60
|
+
const params = step.parameters ?? [];
|
|
61
|
+
const getParam = (name) => params.find((p) => p.name?.toLowerCase() === name.toLowerCase());
|
|
62
|
+
const val = (p) => p?.displayValue ?? p?.value ?? '';
|
|
63
|
+
switch (step.eventType) {
|
|
64
|
+
case 'navigate': {
|
|
65
|
+
const url = getParam('url');
|
|
66
|
+
return url ? `Navigate to ${val(url)}` : 'Navigate';
|
|
67
|
+
}
|
|
68
|
+
case 'ai-agent': {
|
|
69
|
+
const instructions = getParam('instructions') ?? getParam('description');
|
|
70
|
+
return val(instructions) || 'AI Agent step';
|
|
71
|
+
}
|
|
72
|
+
case 'type':
|
|
73
|
+
return val(getParam('text')) || 'Type';
|
|
74
|
+
case 'click':
|
|
75
|
+
return val(getParam('selector')) ? `Click ${val(getParam('selector'))}` : 'Click';
|
|
76
|
+
case 'verify':
|
|
77
|
+
return val(getParam('description')) || 'Verify';
|
|
78
|
+
case 'custom': {
|
|
79
|
+
const desc = getParam('description') ?? getParam('action');
|
|
80
|
+
return val(desc) || 'Custom step';
|
|
81
|
+
}
|
|
82
|
+
default:
|
|
83
|
+
return val(getParam('description')) || val(params[0]) || 'Step';
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
onCancel() {
|
|
87
|
+
this.cancelled.emit();
|
|
88
|
+
}
|
|
89
|
+
onCreateGroup() {
|
|
90
|
+
if (this.form.invalid || this.stepsCount === 0) {
|
|
91
|
+
this.form.markAllAsTouched();
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
const groupName = (this.form.get('groupName')?.value ?? '').trim();
|
|
95
|
+
if (!groupName) {
|
|
96
|
+
this.form.get('groupName')?.setErrors({ required: true });
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
99
|
+
this.createGroup.emit({ groupName });
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
CreateStepGroupComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: CreateStepGroupComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component });
|
|
103
|
+
CreateStepGroupComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: CreateStepGroupComponent, selector: "cqa-create-step-group", inputs: { stepsToGroup: "stepsToGroup" }, outputs: { createGroup: "createGroup", cancelled: "cancelled" }, host: { classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div\n class=\"cqa-bg-white cqa-rounded-[12px] cqa-shadow-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-w-full cqa-max-w-[500px] cqa-flex cqa-flex-col cqa-gap-4 cqa-p-6 cqa-box-border cqa-min-h-0\">\n <!-- Header: title + close (X) -->\n <div class=\"cqa-flex cqa-items-start cqa-justify-between cqa-gap-2\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n <h2 class=\"cqa-text-[16px] cqa-leading-[24px] cqa-font-bold cqa-text-[#111827] cqa-m-0\">\n Create step group\n </h2>\n <p class=\"cqa-text-[14px] cqa-leading-[20px] cqa-text-[#64748B] cqa-m-0\">\n {{ subtitleText }}\n </p>\n </div>\n <button\n type=\"button\"\n (click)=\"onCancel()\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-7 cqa-min-w-7 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0 cqa-flex-shrink-0\"\n title=\"Close\"\n aria-label=\"Close\">\n <svg width=\"20\" height=\"20\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" aria-hidden=\"true\">\n <path d=\"M18 6L6 18M6 6l12 12\" />\n </svg>\n </button>\n </div>\n\n <!-- Group name * -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-[#161617]\">\n Group name <span class=\"cqa-text-red-500\">*</span>\n </label>\n <cqa-custom-input\n placeholder=\"e.g., Authentication flow\"\n [value]=\"form.get('groupName')?.value\"\n [fullWidth]=\"true\"\n size=\"md\"\n (valueChange)=\"form.get('groupName')?.setValue($event); form.get('groupName')?.updateValueAndValidity()\">\n </cqa-custom-input>\n <p *ngIf=\"form.get('groupName')?.invalid && form.get('groupName')?.touched\" class=\"cqa-text-xs cqa-text-red-500 cqa-m-0\">\n Group name is required.\n </p>\n </div>\n\n <!-- Steps to group (N) -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <label class=\"cqa-font-semibold cqa-text-[12px] cqa-leading-[100%] cqa-tracking-normal cqa-text-[#0A0A0A] cqa-align-middle\">\n Steps to group ({{ stepsCount }})\n </label>\n <div class=\"cqa-scrollbar-hide cqa-flex cqa-flex-col cqa-gap-2 cqa-max-h-[200px] cqa-overflow-y-auto cqa-rounded-[8px] cqa-p-[6px] cqa-border cqa-border-solid cqa-border-[rgba(0,0,0,0.1)]\">\n <div\n *ngFor=\"let step of stepsToGroup; let i = index\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-rounded-[4px] cqa-py-[4px] cqa-px-[8px] cqa-bg-[rgba(216,217,252,0.3)]\">\n <!-- Numbered badge -->\n <span\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-rounded-full cqa-bg-[#3F43EE] cqa-text-white cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex-shrink-0\">\n {{ i + 1 }}\n </span>\n <span class=\"cqa-font-medium cqa-text-[10px] cqa-leading-[15px] cqa-tracking-[0px] cqa-text-[#0B0B0C] cqa-flex-1 cqa-min-w-0 cqa-truncate\">\n {{ getStepDisplayLabel(step, i) }}\n </span>\n </div>\n </div>\n </div>\n\n <!-- Actions: Cancel | Create group -->\n <div class=\"cqa-flex cqa-items-stretch cqa-w-full cqa-gap-3\">\n <div class=\"cqa-flex-1 cqa-min-w-0\">\n <cqa-button\n variant=\"outlined\"\n btnSize=\"lg\"\n text=\"Cancel\"\n [fullWidth]=\"true\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n </div>\n <div class=\"cqa-flex-1 cqa-min-w-0\">\n <cqa-button\n variant=\"filled\"\n btnSize=\"lg\"\n text=\"Create group\"\n [fullWidth]=\"true\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#3F43EE] cqa-bg-[#3F43EE]'\"\n (clicked)=\"onCreateGroup()\">\n </cqa-button>\n </div>\n </div>\n</div>\n", components: [{ type: i2.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: i3.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
|
|
104
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: CreateStepGroupComponent, decorators: [{
|
|
105
|
+
type: Component,
|
|
106
|
+
args: [{ selector: 'cqa-create-step-group', host: { class: 'cqa-ui-root' }, template: "<div\n class=\"cqa-bg-white cqa-rounded-[12px] cqa-shadow-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-w-full cqa-max-w-[500px] cqa-flex cqa-flex-col cqa-gap-4 cqa-p-6 cqa-box-border cqa-min-h-0\">\n <!-- Header: title + close (X) -->\n <div class=\"cqa-flex cqa-items-start cqa-justify-between cqa-gap-2\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n <h2 class=\"cqa-text-[16px] cqa-leading-[24px] cqa-font-bold cqa-text-[#111827] cqa-m-0\">\n Create step group\n </h2>\n <p class=\"cqa-text-[14px] cqa-leading-[20px] cqa-text-[#64748B] cqa-m-0\">\n {{ subtitleText }}\n </p>\n </div>\n <button\n type=\"button\"\n (click)=\"onCancel()\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-7 cqa-min-w-7 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0 cqa-flex-shrink-0\"\n title=\"Close\"\n aria-label=\"Close\">\n <svg width=\"20\" height=\"20\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" aria-hidden=\"true\">\n <path d=\"M18 6L6 18M6 6l12 12\" />\n </svg>\n </button>\n </div>\n\n <!-- Group name * -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-[#161617]\">\n Group name <span class=\"cqa-text-red-500\">*</span>\n </label>\n <cqa-custom-input\n placeholder=\"e.g., Authentication flow\"\n [value]=\"form.get('groupName')?.value\"\n [fullWidth]=\"true\"\n size=\"md\"\n (valueChange)=\"form.get('groupName')?.setValue($event); form.get('groupName')?.updateValueAndValidity()\">\n </cqa-custom-input>\n <p *ngIf=\"form.get('groupName')?.invalid && form.get('groupName')?.touched\" class=\"cqa-text-xs cqa-text-red-500 cqa-m-0\">\n Group name is required.\n </p>\n </div>\n\n <!-- Steps to group (N) -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <label class=\"cqa-font-semibold cqa-text-[12px] cqa-leading-[100%] cqa-tracking-normal cqa-text-[#0A0A0A] cqa-align-middle\">\n Steps to group ({{ stepsCount }})\n </label>\n <div class=\"cqa-scrollbar-hide cqa-flex cqa-flex-col cqa-gap-2 cqa-max-h-[200px] cqa-overflow-y-auto cqa-rounded-[8px] cqa-p-[6px] cqa-border cqa-border-solid cqa-border-[rgba(0,0,0,0.1)]\">\n <div\n *ngFor=\"let step of stepsToGroup; let i = index\"\n class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-rounded-[4px] cqa-py-[4px] cqa-px-[8px] cqa-bg-[rgba(216,217,252,0.3)]\">\n <!-- Numbered badge -->\n <span\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-rounded-full cqa-bg-[#3F43EE] cqa-text-white cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex-shrink-0\">\n {{ i + 1 }}\n </span>\n <span class=\"cqa-font-medium cqa-text-[10px] cqa-leading-[15px] cqa-tracking-[0px] cqa-text-[#0B0B0C] cqa-flex-1 cqa-min-w-0 cqa-truncate\">\n {{ getStepDisplayLabel(step, i) }}\n </span>\n </div>\n </div>\n </div>\n\n <!-- Actions: Cancel | Create group -->\n <div class=\"cqa-flex cqa-items-stretch cqa-w-full cqa-gap-3\">\n <div class=\"cqa-flex-1 cqa-min-w-0\">\n <cqa-button\n variant=\"outlined\"\n btnSize=\"lg\"\n text=\"Cancel\"\n [fullWidth]=\"true\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n </div>\n <div class=\"cqa-flex-1 cqa-min-w-0\">\n <cqa-button\n variant=\"filled\"\n btnSize=\"lg\"\n text=\"Create group\"\n [fullWidth]=\"true\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#3F43EE] cqa-bg-[#3F43EE]'\"\n (clicked)=\"onCreateGroup()\">\n </cqa-button>\n </div>\n </div>\n</div>\n" }]
|
|
107
|
+
}], ctorParameters: function () { return [{ type: i1.FormBuilder }]; }, propDecorators: { stepsToGroup: [{
|
|
108
|
+
type: Input
|
|
109
|
+
}], createGroup: [{
|
|
110
|
+
type: Output
|
|
111
|
+
}], cancelled: [{
|
|
112
|
+
type: Output
|
|
113
|
+
}] } });
|
|
114
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"create-step-group.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/create-step-group/create-step-group.component.ts","../../../../../../src/lib/test-case-details/create-step-group/create-step-group.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { FormBuilder, FormGroup, Validators } from '@angular/forms';\nimport {\n  TestCaseStepConfig,\n  NormalStepConfig,\n  TestCaseEventType,\n  isNormalStepConfig,\n} from '../test-case-step.models';\n\nexport interface CreateStepGroupFormData {\n  groupName: string;\n}\n\n/**\n * Create Step Group panel/modal for Test Case Details.\n * Follows the same structure and validation pattern as Test Data Modal and Loop Step.\n * Displays selected steps and allows naming the new group; emits createGroup with group name\n * so the host can create the step group and replace selected steps with it.\n */\n@Component({\n  selector: 'cqa-create-step-group',\n  templateUrl: './create-step-group.component.html',\n  host: { class: 'cqa-ui-root' },\n})\nexport class CreateStepGroupComponent {\n  @Input() stepsToGroup: TestCaseStepConfig[] = [];\n\n  @Output() createGroup = new EventEmitter<CreateStepGroupFormData>();\n  @Output() cancelled = new EventEmitter<void>();\n\n  form: FormGroup;\n\n  constructor(private readonly fb: FormBuilder) {\n    this.form = this.fb.group({\n      groupName: ['', [Validators.required, Validators.minLength(1)]],\n    });\n  }\n\n  get stepsCount(): number {\n    return this.stepsToGroup?.length ?? 0;\n  }\n\n  get subtitleText(): string {\n    const n = this.stepsCount;\n    return n === 1\n      ? 'Group 1 selected step into a container'\n      : `Group ${n} selected steps into a container`;\n  }\n\n  /** Display label for a step in the \"Steps to group\" list (same pattern as normal-step display). */\n  getStepDisplayLabel(step: TestCaseStepConfig, index: number): string {\n    if (isNormalStepConfig(step)) {\n      return this.getNormalStepLabel(step);\n    }\n    if ('groupName' in step && step.groupName) {\n      return step.groupName;\n    }\n    if ('condition' in step && step.condition) {\n      return step.condition;\n    }\n    if ('loopType' in step) {\n      const loop = step as { loopType: string; testDataProfile?: string; condition?: string };\n      if (loop.loopType === 'for' && loop.testDataProfile) {\n        return `For loop: ${loop.testDataProfile}`;\n      }\n      if (loop.loopType === 'while' && loop.condition) {\n        return `While: ${loop.condition}`;\n      }\n    }\n    if (step.description) {\n      return step.description;\n    }\n    return `Step ${index + 1}`;\n  }\n\n  private getNormalStepLabel(step: NormalStepConfig): string {\n    const params = step.parameters ?? [];\n    const getParam = (name: string) =>\n      params.find((p) => p.name?.toLowerCase() === name.toLowerCase());\n    const val = (p: { value?: string; displayValue?: string } | undefined) =>\n      p?.displayValue ?? p?.value ?? '';\n\n    switch (step.eventType as TestCaseEventType) {\n      case 'navigate': {\n        const url = getParam('url');\n        return url ? `Navigate to ${val(url)}` : 'Navigate';\n      }\n      case 'ai-agent': {\n        const instructions = getParam('instructions') ?? getParam('description');\n        return val(instructions) || 'AI Agent step';\n      }\n      case 'type':\n        return val(getParam('text')) || 'Type';\n      case 'click':\n        return val(getParam('selector')) ? `Click ${val(getParam('selector'))}` : 'Click';\n      case 'verify':\n        return val(getParam('description')) || 'Verify';\n      case 'custom': {\n        const desc = getParam('description') ?? getParam('action');\n        return val(desc) || 'Custom step';\n      }\n      default:\n        return val(getParam('description')) || val(params[0]) || 'Step';\n    }\n  }\n\n  onCancel(): void {\n    this.cancelled.emit();\n  }\n\n  onCreateGroup(): void {\n    if (this.form.invalid || this.stepsCount === 0) {\n      this.form.markAllAsTouched();\n      return;\n    }\n    const groupName = (this.form.get('groupName')?.value ?? '').trim();\n    if (!groupName) {\n      this.form.get('groupName')?.setErrors({ required: true });\n      return;\n    }\n    this.createGroup.emit({ groupName });\n  }\n}\n","<div\n  class=\"cqa-bg-white cqa-rounded-[12px] cqa-shadow-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-w-full cqa-max-w-[500px] cqa-flex cqa-flex-col cqa-gap-4 cqa-p-6 cqa-box-border cqa-min-h-0\">\n  <!-- Header: title + close (X) -->\n  <div class=\"cqa-flex cqa-items-start cqa-justify-between cqa-gap-2\">\n    <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n      <h2 class=\"cqa-text-[16px] cqa-leading-[24px] cqa-font-bold cqa-text-[#111827] cqa-m-0\">\n        Create step group\n      </h2>\n      <p class=\"cqa-text-[14px] cqa-leading-[20px] cqa-text-[#64748B] cqa-m-0\">\n        {{ subtitleText }}\n      </p>\n    </div>\n    <button\n      type=\"button\"\n      (click)=\"onCancel()\"\n      class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-7 cqa-min-w-7 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0 cqa-flex-shrink-0\"\n      title=\"Close\"\n      aria-label=\"Close\">\n      <svg width=\"20\" height=\"20\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" aria-hidden=\"true\">\n        <path d=\"M18 6L6 18M6 6l12 12\" />\n      </svg>\n    </button>\n  </div>\n\n  <!-- Group name * -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n    <label class=\"cqa-text-sm cqa-font-medium cqa-text-[#161617]\">\n      Group name <span class=\"cqa-text-red-500\">*</span>\n    </label>\n    <cqa-custom-input\n      placeholder=\"e.g., Authentication flow\"\n      [value]=\"form.get('groupName')?.value\"\n      [fullWidth]=\"true\"\n      size=\"md\"\n      (valueChange)=\"form.get('groupName')?.setValue($event); form.get('groupName')?.updateValueAndValidity()\">\n    </cqa-custom-input>\n    <p *ngIf=\"form.get('groupName')?.invalid && form.get('groupName')?.touched\" class=\"cqa-text-xs cqa-text-red-500 cqa-m-0\">\n      Group name is required.\n    </p>\n  </div>\n\n  <!-- Steps to group (N) -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n    <label class=\"cqa-font-semibold cqa-text-[12px] cqa-leading-[100%] cqa-tracking-normal cqa-text-[#0A0A0A] cqa-align-middle\">\n      Steps to group ({{ stepsCount }})\n    </label>\n    <div class=\"cqa-scrollbar-hide cqa-flex cqa-flex-col cqa-gap-2 cqa-max-h-[200px] cqa-overflow-y-auto cqa-rounded-[8px] cqa-p-[6px] cqa-border cqa-border-solid cqa-border-[rgba(0,0,0,0.1)]\">\n      <div\n        *ngFor=\"let step of stepsToGroup; let i = index\"\n        class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-rounded-[4px] cqa-py-[4px] cqa-px-[8px] cqa-bg-[rgba(216,217,252,0.3)]\">\n        <!-- Numbered badge -->\n        <span\n          class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-rounded-full cqa-bg-[#3F43EE] cqa-text-white cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex-shrink-0\">\n          {{ i + 1 }}\n        </span>\n        <span class=\"cqa-font-medium cqa-text-[10px] cqa-leading-[15px] cqa-tracking-[0px] cqa-text-[#0B0B0C] cqa-flex-1 cqa-min-w-0 cqa-truncate\">\n          {{ getStepDisplayLabel(step, i) }}\n        </span>\n      </div>\n    </div>\n  </div>\n\n  <!-- Actions: Cancel | Create group -->\n  <div class=\"cqa-flex cqa-items-stretch cqa-w-full cqa-gap-3\">\n    <div class=\"cqa-flex-1 cqa-min-w-0\">\n      <cqa-button\n        variant=\"outlined\"\n        btnSize=\"lg\"\n        text=\"Cancel\"\n        [fullWidth]=\"true\"\n        [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"\n        (clicked)=\"onCancel()\">\n      </cqa-button>\n    </div>\n    <div class=\"cqa-flex-1 cqa-min-w-0\">\n      <cqa-button\n        variant=\"filled\"\n        btnSize=\"lg\"\n        text=\"Create group\"\n        [fullWidth]=\"true\"\n        [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#3F43EE] cqa-bg-[#3F43EE]'\"\n        (clicked)=\"onCreateGroup()\">\n      </cqa-button>\n    </div>\n  </div>\n</div>\n"]}
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
2
|
+
import { isNormalStepConfig, } from '../test-case-step.models';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "../../button/button.component";
|
|
5
|
+
import * as i2 from "@angular/common";
|
|
6
|
+
/**
|
|
7
|
+
* Delete Steps confirmation modal for Test Case Details.
|
|
8
|
+
* Matches the design of Create Step Group: shows selected steps with context (e.g. "IF lane"),
|
|
9
|
+
* warning text, and Cancel / Delete actions. Steps are shown with red/danger styling.
|
|
10
|
+
*/
|
|
11
|
+
export class DeleteStepsComponent {
|
|
12
|
+
constructor() {
|
|
13
|
+
/** Steps that will be deleted (shown in the list). */
|
|
14
|
+
this.stepsToDelete = [];
|
|
15
|
+
/** Context label for the sub-heading, e.g. "IF lane", "ELSE", "For loop". */
|
|
16
|
+
this.contextLabel = '';
|
|
17
|
+
this.confirmDelete = new EventEmitter();
|
|
18
|
+
this.cancelled = new EventEmitter();
|
|
19
|
+
}
|
|
20
|
+
get stepsCount() {
|
|
21
|
+
return this.stepsToDelete?.length ?? 0;
|
|
22
|
+
}
|
|
23
|
+
get titleText() {
|
|
24
|
+
const n = this.stepsCount;
|
|
25
|
+
return n === 1 ? 'Delete 1 step?' : `Delete ${n} steps?`;
|
|
26
|
+
}
|
|
27
|
+
get selectedStepsSubtitle() {
|
|
28
|
+
const ctx = this.contextLabel?.trim() || 'selected';
|
|
29
|
+
return `Selected steps for ${ctx} (${this.stepsCount})`;
|
|
30
|
+
}
|
|
31
|
+
/** Display label for a step in the list (same pattern as create-step-group). */
|
|
32
|
+
getStepDisplayLabel(step, index) {
|
|
33
|
+
if (isNormalStepConfig(step)) {
|
|
34
|
+
return this.getNormalStepLabel(step);
|
|
35
|
+
}
|
|
36
|
+
if ('groupName' in step && step.groupName) {
|
|
37
|
+
return step.groupName;
|
|
38
|
+
}
|
|
39
|
+
if ('condition' in step && step.condition) {
|
|
40
|
+
return step.condition;
|
|
41
|
+
}
|
|
42
|
+
if ('loopType' in step) {
|
|
43
|
+
const loop = step;
|
|
44
|
+
if (loop.loopType === 'for' && loop.testDataProfile) {
|
|
45
|
+
return `For loop: ${loop.testDataProfile}`;
|
|
46
|
+
}
|
|
47
|
+
if (loop.loopType === 'while' && loop.condition) {
|
|
48
|
+
return `While: ${loop.condition}`;
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
if (step.description) {
|
|
52
|
+
return step.description;
|
|
53
|
+
}
|
|
54
|
+
return `Step ${index + 1}`;
|
|
55
|
+
}
|
|
56
|
+
getNormalStepLabel(step) {
|
|
57
|
+
const params = step.parameters ?? [];
|
|
58
|
+
const getParam = (name) => params.find((p) => p.name?.toLowerCase() === name.toLowerCase());
|
|
59
|
+
const val = (p) => p?.displayValue ?? p?.value ?? '';
|
|
60
|
+
switch (step.eventType) {
|
|
61
|
+
case 'navigate': {
|
|
62
|
+
const url = getParam('url');
|
|
63
|
+
return url ? `Navigate to ${val(url)}` : 'Navigate to';
|
|
64
|
+
}
|
|
65
|
+
case 'ai-agent': {
|
|
66
|
+
const instructions = getParam('instructions') ?? getParam('description');
|
|
67
|
+
return val(instructions) || 'AI Agent step';
|
|
68
|
+
}
|
|
69
|
+
case 'type':
|
|
70
|
+
return val(getParam('text')) || 'Type';
|
|
71
|
+
case 'click':
|
|
72
|
+
return val(getParam('selector')) ? `Click ${val(getParam('selector'))}` : 'Click';
|
|
73
|
+
case 'verify':
|
|
74
|
+
return val(getParam('description')) || 'Verify';
|
|
75
|
+
case 'custom': {
|
|
76
|
+
const desc = getParam('description') ?? getParam('action');
|
|
77
|
+
return val(desc) || 'Custom step';
|
|
78
|
+
}
|
|
79
|
+
default:
|
|
80
|
+
return val(getParam('description')) || val(params[0]) || 'Step';
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
onCancel() {
|
|
84
|
+
this.cancelled.emit();
|
|
85
|
+
}
|
|
86
|
+
onDelete() {
|
|
87
|
+
this.confirmDelete.emit();
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
DeleteStepsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DeleteStepsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
91
|
+
DeleteStepsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DeleteStepsComponent, selector: "cqa-delete-steps", inputs: { stepsToDelete: "stepsToDelete", contextLabel: "contextLabel" }, outputs: { confirmDelete: "confirmDelete", cancelled: "cancelled" }, host: { classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div\n class=\"cqa-bg-white cqa-rounded-[12px] cqa-shadow-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-w-full cqa-max-w-[500px] cqa-flex cqa-flex-col cqa-gap-4 cqa-p-6 cqa-box-border cqa-min-h-0\">\n <!-- Header: title + close (X) -->\n <div class=\"cqa-flex cqa-items-start cqa-justify-between cqa-gap-2\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n <h2 class=\"cqa-font-medium cqa-text-[22px] cqa-leading-[28px] cqa-tracking-normal cqa-text-[#101828] cqa-m-0\">\n {{ titleText }}\n </h2>\n <p class=\"cqa-font-normal cqa-text-[14px] cqa-leading-[21px] cqa-tracking-[-0.15px] cqa-text-[#4A5565] cqa-m-0\">\n This action cannot be undone, but you can use the undo function in the toolbar.\n </p>\n </div>\n <button\n type=\"button\"\n (click)=\"onCancel()\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-7 cqa-min-w-7 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0 cqa-flex-shrink-0\"\n title=\"Close\"\n aria-label=\"Close\">\n <svg width=\"20\" height=\"20\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" aria-hidden=\"true\">\n <path d=\"M18 6L6 18M6 6l12 12\" />\n </svg>\n </button>\n </div>\n\n <!-- Selected steps for [context] (N) \u2013 scroll when many steps, scrollbar hidden (same as create-step-group) -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-min-h-0\">\n <label class=\"cqa-font-semibold cqa-text-[12px] cqa-leading-[100%] cqa-tracking-normal cqa-text-[#0A0A0A] cqa-align-middle\">\n {{ selectedStepsSubtitle }}\n </label>\n <div class=\"cqa-scrollbar-hide cqa-flex cqa-flex-col cqa-gap-2 cqa-max-h-[200px] cqa-min-h-0 cqa-overflow-y-auto cqa-overflow-x-hidden cqa-rounded-[8px] cqa-p-[6px] cqa-border cqa-border-solid cqa-border-[rgba(0,0,0,0.1)]\">\n <div\n *ngFor=\"let step of stepsToDelete; let i = index\"\n class=\"cqa-flex cqa-items-center cqa-gap-[8px] cqa-rounded-[4px] cqa-pt-[4px] cqa-pb-[4px] cqa-pl-[8px] cqa-pr-[8px] cqa-bg-[rgba(238,63,63,0.1)]\">\n <!-- Numbered badge (red) -->\n <span\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-rounded-full cqa-bg-[#C63535] cqa-text-white cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex-shrink-0\">\n {{ i + 1 }}\n </span>\n <span class=\"cqa-font-medium cqa-text-[10px] cqa-leading-[15px] cqa-tracking-[0px] cqa-text-[#DC2626] cqa-flex-1 cqa-min-w-0 cqa-truncate\">\n {{ getStepDisplayLabel(step, i) }}\n </span>\n </div>\n </div>\n </div> \n\n <!-- Actions: Cancel | Delete (right-aligned) -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-end cqa-w-full cqa-gap-3\">\n <cqa-button\n variant=\"outlined\"\n btnSize=\"lg\"\n text=\"Cancel\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-bg-white cqa-border cqa-border-solid cqa-border-[#414146] cqa-text-[#414146]'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button\n variant=\"filled\"\n btnSize=\"lg\"\n text=\"Delete\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-rounded-[8px] cqa-border-0 cqa-bg-[#EE3F3F] cqa-text-white'\"\n (clicked)=\"onDelete()\">\n </cqa-button>\n </div>\n</div>\n", components: [{ type: i1.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }], directives: [{ type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
|
|
92
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DeleteStepsComponent, decorators: [{
|
|
93
|
+
type: Component,
|
|
94
|
+
args: [{ selector: 'cqa-delete-steps', host: { class: 'cqa-ui-root' }, template: "<div\n class=\"cqa-bg-white cqa-rounded-[12px] cqa-shadow-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-w-full cqa-max-w-[500px] cqa-flex cqa-flex-col cqa-gap-4 cqa-p-6 cqa-box-border cqa-min-h-0\">\n <!-- Header: title + close (X) -->\n <div class=\"cqa-flex cqa-items-start cqa-justify-between cqa-gap-2\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n <h2 class=\"cqa-font-medium cqa-text-[22px] cqa-leading-[28px] cqa-tracking-normal cqa-text-[#101828] cqa-m-0\">\n {{ titleText }}\n </h2>\n <p class=\"cqa-font-normal cqa-text-[14px] cqa-leading-[21px] cqa-tracking-[-0.15px] cqa-text-[#4A5565] cqa-m-0\">\n This action cannot be undone, but you can use the undo function in the toolbar.\n </p>\n </div>\n <button\n type=\"button\"\n (click)=\"onCancel()\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-7 cqa-min-w-7 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0 cqa-flex-shrink-0\"\n title=\"Close\"\n aria-label=\"Close\">\n <svg width=\"20\" height=\"20\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" aria-hidden=\"true\">\n <path d=\"M18 6L6 18M6 6l12 12\" />\n </svg>\n </button>\n </div>\n\n <!-- Selected steps for [context] (N) \u2013 scroll when many steps, scrollbar hidden (same as create-step-group) -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-min-h-0\">\n <label class=\"cqa-font-semibold cqa-text-[12px] cqa-leading-[100%] cqa-tracking-normal cqa-text-[#0A0A0A] cqa-align-middle\">\n {{ selectedStepsSubtitle }}\n </label>\n <div class=\"cqa-scrollbar-hide cqa-flex cqa-flex-col cqa-gap-2 cqa-max-h-[200px] cqa-min-h-0 cqa-overflow-y-auto cqa-overflow-x-hidden cqa-rounded-[8px] cqa-p-[6px] cqa-border cqa-border-solid cqa-border-[rgba(0,0,0,0.1)]\">\n <div\n *ngFor=\"let step of stepsToDelete; let i = index\"\n class=\"cqa-flex cqa-items-center cqa-gap-[8px] cqa-rounded-[4px] cqa-pt-[4px] cqa-pb-[4px] cqa-pl-[8px] cqa-pr-[8px] cqa-bg-[rgba(238,63,63,0.1)]\">\n <!-- Numbered badge (red) -->\n <span\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-rounded-full cqa-bg-[#C63535] cqa-text-white cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex-shrink-0\">\n {{ i + 1 }}\n </span>\n <span class=\"cqa-font-medium cqa-text-[10px] cqa-leading-[15px] cqa-tracking-[0px] cqa-text-[#DC2626] cqa-flex-1 cqa-min-w-0 cqa-truncate\">\n {{ getStepDisplayLabel(step, i) }}\n </span>\n </div>\n </div>\n </div> \n\n <!-- Actions: Cancel | Delete (right-aligned) -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-end cqa-w-full cqa-gap-3\">\n <cqa-button\n variant=\"outlined\"\n btnSize=\"lg\"\n text=\"Cancel\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-bg-white cqa-border cqa-border-solid cqa-border-[#414146] cqa-text-[#414146]'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button\n variant=\"filled\"\n btnSize=\"lg\"\n text=\"Delete\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-rounded-[8px] cqa-border-0 cqa-bg-[#EE3F3F] cqa-text-white'\"\n (clicked)=\"onDelete()\">\n </cqa-button>\n </div>\n</div>\n" }]
|
|
95
|
+
}], propDecorators: { stepsToDelete: [{
|
|
96
|
+
type: Input
|
|
97
|
+
}], contextLabel: [{
|
|
98
|
+
type: Input
|
|
99
|
+
}], confirmDelete: [{
|
|
100
|
+
type: Output
|
|
101
|
+
}], cancelled: [{
|
|
102
|
+
type: Output
|
|
103
|
+
}] } });
|
|
104
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"delete-steps.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/delete-steps/delete-steps.component.ts","../../../../../../src/lib/test-case-details/delete-steps/delete-steps.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\nimport {\n  TestCaseStepConfig,\n  NormalStepConfig,\n  TestCaseEventType,\n  isNormalStepConfig,\n} from '../test-case-step.models';\n\n/**\n * Delete Steps confirmation modal for Test Case Details.\n * Matches the design of Create Step Group: shows selected steps with context (e.g. \"IF lane\"),\n * warning text, and Cancel / Delete actions. Steps are shown with red/danger styling.\n */\n@Component({\n  selector: 'cqa-delete-steps',\n  templateUrl: './delete-steps.component.html',\n  host: { class: 'cqa-ui-root' },\n})\nexport class DeleteStepsComponent {\n  /** Steps that will be deleted (shown in the list). */\n  @Input() stepsToDelete: TestCaseStepConfig[] = [];\n\n  /** Context label for the sub-heading, e.g. \"IF lane\", \"ELSE\", \"For loop\". */\n  @Input() contextLabel = '';\n\n  @Output() confirmDelete = new EventEmitter<void>();\n  @Output() cancelled = new EventEmitter<void>();\n\n  get stepsCount(): number {\n    return this.stepsToDelete?.length ?? 0;\n  }\n\n  get titleText(): string {\n    const n = this.stepsCount;\n    return n === 1 ? 'Delete 1 step?' : `Delete ${n} steps?`;\n  }\n\n  get selectedStepsSubtitle(): string {\n    const ctx = this.contextLabel?.trim() || 'selected';\n    return `Selected steps for ${ctx} (${this.stepsCount})`;\n  }\n\n  /** Display label for a step in the list (same pattern as create-step-group). */\n  getStepDisplayLabel(step: TestCaseStepConfig, index: number): string {\n    if (isNormalStepConfig(step)) {\n      return this.getNormalStepLabel(step);\n    }\n    if ('groupName' in step && step.groupName) {\n      return step.groupName;\n    }\n    if ('condition' in step && step.condition) {\n      return step.condition;\n    }\n    if ('loopType' in step) {\n      const loop = step as { loopType: string; testDataProfile?: string; condition?: string };\n      if (loop.loopType === 'for' && loop.testDataProfile) {\n        return `For loop: ${loop.testDataProfile}`;\n      }\n      if (loop.loopType === 'while' && loop.condition) {\n        return `While: ${loop.condition}`;\n      }\n    }\n    if (step.description) {\n      return step.description;\n    }\n    return `Step ${index + 1}`;\n  }\n\n  private getNormalStepLabel(step: NormalStepConfig): string {\n    const params = step.parameters ?? [];\n    const getParam = (name: string) =>\n      params.find((p) => p.name?.toLowerCase() === name.toLowerCase());\n    const val = (p: { value?: string; displayValue?: string } | undefined) =>\n      p?.displayValue ?? p?.value ?? '';\n\n    switch (step.eventType as TestCaseEventType) {\n      case 'navigate': {\n        const url = getParam('url');\n        return url ? `Navigate to ${val(url)}` : 'Navigate to';\n      }\n      case 'ai-agent': {\n        const instructions = getParam('instructions') ?? getParam('description');\n        return val(instructions) || 'AI Agent step';\n      }\n      case 'type':\n        return val(getParam('text')) || 'Type';\n      case 'click':\n        return val(getParam('selector')) ? `Click ${val(getParam('selector'))}` : 'Click';\n      case 'verify':\n        return val(getParam('description')) || 'Verify';\n      case 'custom': {\n        const desc = getParam('description') ?? getParam('action');\n        return val(desc) || 'Custom step';\n      }\n      default:\n        return val(getParam('description')) || val(params[0]) || 'Step';\n    }\n  }\n\n  onCancel(): void {\n    this.cancelled.emit();\n  }\n\n  onDelete(): void {\n    this.confirmDelete.emit();\n  }\n}\n","<div\n  class=\"cqa-bg-white cqa-rounded-[12px] cqa-shadow-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-w-full cqa-max-w-[500px] cqa-flex cqa-flex-col cqa-gap-4 cqa-p-6 cqa-box-border cqa-min-h-0\">\n  <!-- Header: title + close (X) -->\n  <div class=\"cqa-flex cqa-items-start cqa-justify-between cqa-gap-2\">\n    <div class=\"cqa-flex cqa-flex-col cqa-gap-1 cqa-min-w-0\">\n      <h2 class=\"cqa-font-medium cqa-text-[22px] cqa-leading-[28px] cqa-tracking-normal cqa-text-[#101828] cqa-m-0\">\n        {{ titleText }}\n      </h2>\n      <p class=\"cqa-font-normal cqa-text-[14px] cqa-leading-[21px] cqa-tracking-[-0.15px] cqa-text-[#4A5565] cqa-m-0\">\n        This action cannot be undone, but you can use the undo function in the toolbar.\n      </p>\n    </div>\n    <button\n      type=\"button\"\n      (click)=\"onCancel()\"\n      class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-7 cqa-min-w-7 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0 cqa-flex-shrink-0\"\n      title=\"Close\"\n      aria-label=\"Close\">\n      <svg width=\"20\" height=\"20\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" aria-hidden=\"true\">\n        <path d=\"M18 6L6 18M6 6l12 12\" />\n      </svg>\n    </button>\n  </div>\n\n  <!-- Selected steps for [context] (N) – scroll when many steps, scrollbar hidden (same as create-step-group) -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-min-h-0\">\n    <label class=\"cqa-font-semibold cqa-text-[12px] cqa-leading-[100%] cqa-tracking-normal cqa-text-[#0A0A0A] cqa-align-middle\">\n      {{ selectedStepsSubtitle }}\n    </label>\n    <div class=\"cqa-scrollbar-hide cqa-flex cqa-flex-col cqa-gap-2 cqa-max-h-[200px] cqa-min-h-0 cqa-overflow-y-auto cqa-overflow-x-hidden cqa-rounded-[8px] cqa-p-[6px] cqa-border cqa-border-solid cqa-border-[rgba(0,0,0,0.1)]\">\n      <div\n        *ngFor=\"let step of stepsToDelete; let i = index\"\n        class=\"cqa-flex cqa-items-center cqa-gap-[8px] cqa-rounded-[4px] cqa-pt-[4px] cqa-pb-[4px] cqa-pl-[8px] cqa-pr-[8px] cqa-bg-[rgba(238,63,63,0.1)]\">\n        <!-- Numbered badge (red) -->\n        <span\n          class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-7 cqa-h-7 cqa-rounded-full cqa-bg-[#C63535] cqa-text-white cqa-text-[10px] cqa-leading-[15px] cqa-font-medium cqa-flex-shrink-0\">\n          {{ i + 1 }}\n        </span>\n        <span class=\"cqa-font-medium cqa-text-[10px] cqa-leading-[15px] cqa-tracking-[0px] cqa-text-[#DC2626] cqa-flex-1 cqa-min-w-0 cqa-truncate\">\n          {{ getStepDisplayLabel(step, i) }}\n        </span>\n      </div>\n    </div>\n  </div>  \n\n  <!-- Actions: Cancel | Delete (right-aligned) -->\n  <div class=\"cqa-flex cqa-items-center cqa-justify-end cqa-w-full cqa-gap-3\">\n    <cqa-button\n      variant=\"outlined\"\n      btnSize=\"lg\"\n      text=\"Cancel\"\n      [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-bg-white cqa-border cqa-border-solid cqa-border-[#414146] cqa-text-[#414146]'\"\n      (clicked)=\"onCancel()\">\n    </cqa-button>\n    <cqa-button\n      variant=\"filled\"\n      btnSize=\"lg\"\n      text=\"Delete\"\n      [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-rounded-[8px] cqa-border-0 cqa-bg-[#EE3F3F] cqa-text-white'\"\n      (clicked)=\"onDelete()\">\n    </cqa-button>\n  </div>\n</div>\n"]}
|