@cqa-lib/cqa-ui 1.1.192 → 1.1.194
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/autocomplete/autocomplete.component.mjs +156 -0
- package/esm2020/lib/autocomplete/autocomplete.model.mjs +2 -0
- package/esm2020/lib/step-builder/step-builder-database/step-builder-database.component.mjs +29 -15
- package/esm2020/lib/step-builder/step-builder-document/step-builder-document.component.mjs +70 -7
- package/esm2020/lib/test-case-details/api-edit-step/api-edit-step.component.mjs +3 -3
- package/esm2020/lib/test-case-details/condition-step/condition-step.component.mjs +505 -17
- package/esm2020/lib/test-case-details/delete-steps/delete-steps.component.mjs +32 -11
- package/esm2020/lib/test-case-details/loop-step/loop-step.component.mjs +131 -9
- package/esm2020/lib/test-case-details/step-details-drawer/step-details-drawer-data.mjs +67 -1
- package/esm2020/lib/test-case-details/step-details-drawer/step-details-drawer-field.config.mjs +66 -1
- package/esm2020/lib/test-case-details/step-details-drawer/step-details-drawer.component.mjs +112 -191
- package/esm2020/lib/test-case-details/step-group/step-group.component.mjs +31 -3
- package/esm2020/lib/test-case-details/test-case-details-renderer/test-case-details-renderer.component.mjs +114 -5
- package/esm2020/lib/ui-kit.module.mjs +6 -1
- package/esm2020/public-api.mjs +3 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +1523 -466
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +1514 -462
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/autocomplete/autocomplete.component.d.ts +48 -0
- package/lib/autocomplete/autocomplete.model.d.ts +10 -0
- package/lib/step-builder/step-builder-database/step-builder-database.component.d.ts +6 -1
- package/lib/step-builder/step-builder-document/step-builder-document.component.d.ts +18 -1
- package/lib/test-case-details/condition-step/condition-step.component.d.ts +73 -4
- package/lib/test-case-details/delete-steps/delete-steps.component.d.ts +18 -2
- package/lib/test-case-details/loop-step/loop-step.component.d.ts +34 -4
- package/lib/test-case-details/step-details-drawer/step-details-drawer-data.d.ts +56 -0
- package/lib/test-case-details/step-details-drawer/step-details-drawer-field.config.d.ts +6 -0
- package/lib/test-case-details/step-details-drawer/step-details-drawer.component.d.ts +33 -27
- package/lib/test-case-details/step-group/step-group.component.d.ts +15 -1
- package/lib/test-case-details/test-case-details-renderer/test-case-details-renderer.component.d.ts +27 -4
- package/lib/ui-kit.module.d.ts +127 -126
- package/package.json +1 -1
- package/public-api.d.ts +2 -0
- package/styles.css +1 -1
|
@@ -0,0 +1,156 @@
|
|
|
1
|
+
import { Component, EventEmitter, HostListener, Input, Output, } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/material/icon";
|
|
4
|
+
import * as i2 from "@angular/common";
|
|
5
|
+
export class AutocompleteComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
/** Placeholder text for the input */
|
|
8
|
+
this.placeholder = 'Search or select...';
|
|
9
|
+
/** Options shown in the dropdown (filtered by input when open) */
|
|
10
|
+
this.options = [];
|
|
11
|
+
/** Initial or controlled value (displayed in the input) */
|
|
12
|
+
this.value = '';
|
|
13
|
+
/** Disable the input */
|
|
14
|
+
this.disabled = false;
|
|
15
|
+
/** Whether to show the clear button when there is text */
|
|
16
|
+
this.showClear = true;
|
|
17
|
+
/** Accessible label for the input */
|
|
18
|
+
this.ariaLabel = 'Auto-complete';
|
|
19
|
+
/** Auto focus the input when rendered */
|
|
20
|
+
this.autoFocus = false;
|
|
21
|
+
/** Size variant (matches search bar widths) */
|
|
22
|
+
this.size = 'md';
|
|
23
|
+
/** Stretch to full width of container */
|
|
24
|
+
this.fullWidth = false;
|
|
25
|
+
/** Emit when the input value changes (e.g. for two-way binding) */
|
|
26
|
+
this.valueChange = new EventEmitter();
|
|
27
|
+
/** Emit when an option is selected from the list (value from option) */
|
|
28
|
+
this.optionSelect = new EventEmitter();
|
|
29
|
+
/** Emit when the value is cleared via the clear button */
|
|
30
|
+
this.cleared = new EventEmitter();
|
|
31
|
+
this.inputValue = '';
|
|
32
|
+
this.panelOpen = false;
|
|
33
|
+
this.blurTimeout = null;
|
|
34
|
+
this.widthClasses = {
|
|
35
|
+
sm: 'cqa-w-[295px]',
|
|
36
|
+
md: 'cqa-w-[395px]',
|
|
37
|
+
lg: 'cqa-w-[495px]',
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
ngOnInit() {
|
|
41
|
+
// Initialize inputValue from value input
|
|
42
|
+
if (this.value !== undefined && this.value !== null && this.value !== '') {
|
|
43
|
+
this.inputValue = String(this.value);
|
|
44
|
+
}
|
|
45
|
+
else {
|
|
46
|
+
this.inputValue = '';
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
ngOnChanges(changes) {
|
|
50
|
+
if (changes['value']) {
|
|
51
|
+
const newValue = changes['value'].currentValue;
|
|
52
|
+
if (newValue !== undefined && newValue !== null && newValue !== '') {
|
|
53
|
+
const stringValue = String(newValue);
|
|
54
|
+
if (stringValue !== this.inputValue) {
|
|
55
|
+
this.inputValue = stringValue;
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
else {
|
|
59
|
+
// Clear input if value is empty/null/undefined
|
|
60
|
+
if (this.inputValue !== '') {
|
|
61
|
+
this.inputValue = '';
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
get displayOptions() {
|
|
67
|
+
const list = this.options ?? [];
|
|
68
|
+
const q = (this.inputValue ?? '').toLowerCase().trim();
|
|
69
|
+
if (!q) {
|
|
70
|
+
return list;
|
|
71
|
+
}
|
|
72
|
+
return list.filter((opt) => {
|
|
73
|
+
const label = (opt.label ?? opt.value ?? '').toLowerCase();
|
|
74
|
+
return label.includes(q);
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
onInput(event) {
|
|
78
|
+
const target = event.target;
|
|
79
|
+
const nextValue = target?.value ?? '';
|
|
80
|
+
this.inputValue = nextValue;
|
|
81
|
+
this.valueChange.emit(this.inputValue);
|
|
82
|
+
this.panelOpen = true;
|
|
83
|
+
}
|
|
84
|
+
onFocus() {
|
|
85
|
+
if (this.blurTimeout) {
|
|
86
|
+
clearTimeout(this.blurTimeout);
|
|
87
|
+
this.blurTimeout = null;
|
|
88
|
+
}
|
|
89
|
+
this.panelOpen = true;
|
|
90
|
+
}
|
|
91
|
+
onBlur() {
|
|
92
|
+
this.blurTimeout = setTimeout(() => {
|
|
93
|
+
this.panelOpen = false;
|
|
94
|
+
this.blurTimeout = null;
|
|
95
|
+
}, 150);
|
|
96
|
+
}
|
|
97
|
+
selectOption(option) {
|
|
98
|
+
const displayLabel = option.label ?? option.value;
|
|
99
|
+
this.inputValue = displayLabel;
|
|
100
|
+
this.valueChange.emit(this.inputValue);
|
|
101
|
+
this.optionSelect.emit(option);
|
|
102
|
+
this.panelOpen = false;
|
|
103
|
+
}
|
|
104
|
+
clear() {
|
|
105
|
+
if (this.disabled || this.inputValue === '') {
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
108
|
+
this.inputValue = '';
|
|
109
|
+
this.valueChange.emit('');
|
|
110
|
+
this.cleared.emit();
|
|
111
|
+
this.panelOpen = false;
|
|
112
|
+
}
|
|
113
|
+
getOptionLabel(option) {
|
|
114
|
+
return option.label ?? option.value ?? '';
|
|
115
|
+
}
|
|
116
|
+
trackByValue(_index, option) {
|
|
117
|
+
return option.value;
|
|
118
|
+
}
|
|
119
|
+
onEscape() {
|
|
120
|
+
this.panelOpen = false;
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
AutocompleteComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AutocompleteComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
124
|
+
AutocompleteComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AutocompleteComponent, selector: "cqa-autocomplete", inputs: { placeholder: "placeholder", options: "options", value: "value", disabled: "disabled", showClear: "showClear", ariaLabel: "ariaLabel", autoFocus: "autoFocus", size: "size", fullWidth: "fullWidth" }, outputs: { valueChange: "valueChange", optionSelect: "optionSelect", cleared: "cleared" }, host: { listeners: { "document:keydown.escape": "onEscape()" }, classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: "<div\n class=\"cqa-autocomplete-root cqa-relative\"\n [style.display]=\"fullWidth ? 'block' : 'inline-block'\"\n [style.width]=\"fullWidth ? '100%' : 'auto'\">\n <!-- Field: same styling as search bar -->\n <div\n class=\"cqa-inline-flex cqa-items-center cqa-gap-2 cqa-px-6 cqa-py-3 cqa-text-[14px] cqa-border cqa-border-solid cqa-border-gray-200 cqa-rounded-md cqa-bg-white cqa-shadow-sm cqa-transition-colors\"\n [ngClass]=\"[fullWidth ? 'cqa-w-full' : widthClasses[size]]\">\n <input\n type=\"text\"\n class=\"cqa-flex-1 cqa-min-w-0 cqa-border-none cqa-outline-none cqa-bg-transparent placeholder:cqa-text-gray-300 disabled:cqa-text-gray-400 disabled:cqa-cursor-not-allowed cqa-font-['SF_Pro_Text'] cqa-font-normal cqa-text-[12.3px] cqa-leading-none cqa-tracking-normal cqa-align-middle cqa-text-black-100\"\n [placeholder]=\"placeholder\"\n [value]=\"inputValue\"\n (input)=\"onInput($event)\"\n (focus)=\"onFocus()\"\n (blur)=\"onBlur()\"\n [disabled]=\"disabled\"\n [attr.aria-label]=\"ariaLabel\"\n [attr.aria-expanded]=\"panelOpen\"\n aria-autocomplete=\"list\"\n aria-haspopup=\"listbox\"\n autocomplete=\"off\"\n autocapitalize=\"none\"\n spellcheck=\"false\"\n [attr.autofocus]=\"autoFocus ? '' : null\"\n />\n <button\n *ngIf=\"showClear && inputValue\"\n type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-p-0 cqa-w-4 cqa-h-4 cqa-border-0 cqa-bg-transparent cqa-cursor-pointer cqa-text-gray-500 cqa-hover:cqa-text-gray-700 disabled:cqa-text-gray-300 cqa-transition-colors cqa-leading-none\"\n (click)=\"clear()\"\n [disabled]=\"disabled\"\n aria-label=\"Clear\">\n <mat-icon\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-leading-none cqa-p-0\"\n [style.width.px]=\"16\"\n [style.height.px]=\"16\"\n [style.fontSize.px]=\"16\"\n [ngClass]=\"{ 'cqa-opacity-[0.38]': disabled }\">\n close\n </mat-icon>\n </button>\n </div>\n <!-- List box: below field, same width as field -->\n <ul\n *ngIf=\"panelOpen\"\n role=\"listbox\"\n class=\"cqa-autocomplete-list cqa-absolute cqa-left-0 cqa-right-0 cqa-top-full cqa-mt-1 cqa-py-1 cqa-border cqa-border-solid cqa-border-gray-200 cqa-rounded-md cqa-bg-white cqa-shadow-sm cqa-max-h-[240px] cqa-overflow-y-auto cqa-scrollbar-thin cqa-z-10 cqa-list-none cqa-m-0 cqa-p-0\"\n [class.cqa-w-full]=\"fullWidth\">\n <li\n *ngFor=\"let option of displayOptions; trackBy: trackByValue\"\n role=\"option\"\n (mousedown)=\"selectOption(option); $event.preventDefault()\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-[12.3px] cqa-leading-none cqa-tracking-normal cqa-text-black-100 cqa-cursor-pointer cqa-font-['SF_Pro_Text'] hover:cqa-bg-gray-100 cqa-transition-colors\">\n {{ getOptionLabel(option) }}\n </li>\n <li\n *ngIf=\"displayOptions.length === 0\"\n class=\"cqa-px-4 cqa-py-3 cqa-text-[12.3px] cqa-text-gray-400 cqa-cursor-default\">\n No matches\n </li>\n </ul>\n</div>\n", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }], directives: [{ type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }] });
|
|
125
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AutocompleteComponent, decorators: [{
|
|
126
|
+
type: Component,
|
|
127
|
+
args: [{ selector: 'cqa-autocomplete', host: { class: 'cqa-ui-root' }, template: "<div\n class=\"cqa-autocomplete-root cqa-relative\"\n [style.display]=\"fullWidth ? 'block' : 'inline-block'\"\n [style.width]=\"fullWidth ? '100%' : 'auto'\">\n <!-- Field: same styling as search bar -->\n <div\n class=\"cqa-inline-flex cqa-items-center cqa-gap-2 cqa-px-6 cqa-py-3 cqa-text-[14px] cqa-border cqa-border-solid cqa-border-gray-200 cqa-rounded-md cqa-bg-white cqa-shadow-sm cqa-transition-colors\"\n [ngClass]=\"[fullWidth ? 'cqa-w-full' : widthClasses[size]]\">\n <input\n type=\"text\"\n class=\"cqa-flex-1 cqa-min-w-0 cqa-border-none cqa-outline-none cqa-bg-transparent placeholder:cqa-text-gray-300 disabled:cqa-text-gray-400 disabled:cqa-cursor-not-allowed cqa-font-['SF_Pro_Text'] cqa-font-normal cqa-text-[12.3px] cqa-leading-none cqa-tracking-normal cqa-align-middle cqa-text-black-100\"\n [placeholder]=\"placeholder\"\n [value]=\"inputValue\"\n (input)=\"onInput($event)\"\n (focus)=\"onFocus()\"\n (blur)=\"onBlur()\"\n [disabled]=\"disabled\"\n [attr.aria-label]=\"ariaLabel\"\n [attr.aria-expanded]=\"panelOpen\"\n aria-autocomplete=\"list\"\n aria-haspopup=\"listbox\"\n autocomplete=\"off\"\n autocapitalize=\"none\"\n spellcheck=\"false\"\n [attr.autofocus]=\"autoFocus ? '' : null\"\n />\n <button\n *ngIf=\"showClear && inputValue\"\n type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-p-0 cqa-w-4 cqa-h-4 cqa-border-0 cqa-bg-transparent cqa-cursor-pointer cqa-text-gray-500 cqa-hover:cqa-text-gray-700 disabled:cqa-text-gray-300 cqa-transition-colors cqa-leading-none\"\n (click)=\"clear()\"\n [disabled]=\"disabled\"\n aria-label=\"Clear\">\n <mat-icon\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-leading-none cqa-p-0\"\n [style.width.px]=\"16\"\n [style.height.px]=\"16\"\n [style.fontSize.px]=\"16\"\n [ngClass]=\"{ 'cqa-opacity-[0.38]': disabled }\">\n close\n </mat-icon>\n </button>\n </div>\n <!-- List box: below field, same width as field -->\n <ul\n *ngIf=\"panelOpen\"\n role=\"listbox\"\n class=\"cqa-autocomplete-list cqa-absolute cqa-left-0 cqa-right-0 cqa-top-full cqa-mt-1 cqa-py-1 cqa-border cqa-border-solid cqa-border-gray-200 cqa-rounded-md cqa-bg-white cqa-shadow-sm cqa-max-h-[240px] cqa-overflow-y-auto cqa-scrollbar-thin cqa-z-10 cqa-list-none cqa-m-0 cqa-p-0\"\n [class.cqa-w-full]=\"fullWidth\">\n <li\n *ngFor=\"let option of displayOptions; trackBy: trackByValue\"\n role=\"option\"\n (mousedown)=\"selectOption(option); $event.preventDefault()\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-[12.3px] cqa-leading-none cqa-tracking-normal cqa-text-black-100 cqa-cursor-pointer cqa-font-['SF_Pro_Text'] hover:cqa-bg-gray-100 cqa-transition-colors\">\n {{ getOptionLabel(option) }}\n </li>\n <li\n *ngIf=\"displayOptions.length === 0\"\n class=\"cqa-px-4 cqa-py-3 cqa-text-[12.3px] cqa-text-gray-400 cqa-cursor-default\">\n No matches\n </li>\n </ul>\n</div>\n" }]
|
|
128
|
+
}], propDecorators: { placeholder: [{
|
|
129
|
+
type: Input
|
|
130
|
+
}], options: [{
|
|
131
|
+
type: Input
|
|
132
|
+
}], value: [{
|
|
133
|
+
type: Input
|
|
134
|
+
}], disabled: [{
|
|
135
|
+
type: Input
|
|
136
|
+
}], showClear: [{
|
|
137
|
+
type: Input
|
|
138
|
+
}], ariaLabel: [{
|
|
139
|
+
type: Input
|
|
140
|
+
}], autoFocus: [{
|
|
141
|
+
type: Input
|
|
142
|
+
}], size: [{
|
|
143
|
+
type: Input
|
|
144
|
+
}], fullWidth: [{
|
|
145
|
+
type: Input
|
|
146
|
+
}], valueChange: [{
|
|
147
|
+
type: Output
|
|
148
|
+
}], optionSelect: [{
|
|
149
|
+
type: Output
|
|
150
|
+
}], cleared: [{
|
|
151
|
+
type: Output
|
|
152
|
+
}], onEscape: [{
|
|
153
|
+
type: HostListener,
|
|
154
|
+
args: ['document:keydown.escape']
|
|
155
|
+
}] } });
|
|
156
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"autocomplete.component.js","sourceRoot":"","sources":["../../../../../src/lib/autocomplete/autocomplete.component.ts","../../../../../src/lib/autocomplete/autocomplete.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  HostListener,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n} from '@angular/core';\nimport { CqaAutocompleteOption } from './autocomplete.model';\n\ntype AutocompleteSize = 'sm' | 'md' | 'lg';\n\n@Component({\n  selector: 'cqa-autocomplete',\n  templateUrl: './autocomplete.component.html',\n  host: { class: 'cqa-ui-root' },\n})\nexport class AutocompleteComponent implements OnInit, OnChanges {\n  /** Placeholder text for the input */\n  @Input() placeholder = 'Search or select...';\n\n  /** Options shown in the dropdown (filtered by input when open) */\n  @Input() options: CqaAutocompleteOption[] = [];\n\n  /** Initial or controlled value (displayed in the input) */\n  @Input() value = '';\n\n  /** Disable the input */\n  @Input() disabled = false;\n\n  /** Whether to show the clear button when there is text */\n  @Input() showClear = true;\n\n  /** Accessible label for the input */\n  @Input() ariaLabel = 'Auto-complete';\n\n  /** Auto focus the input when rendered */\n  @Input() autoFocus = false;\n\n  /** Size variant (matches search bar widths) */\n  @Input() size: AutocompleteSize = 'md';\n\n  /** Stretch to full width of container */\n  @Input() fullWidth = false;\n\n  /** Emit when the input value changes (e.g. for two-way binding) */\n  @Output() valueChange = new EventEmitter<string>();\n\n  /** Emit when an option is selected from the list (value from option) */\n  @Output() optionSelect = new EventEmitter<CqaAutocompleteOption>();\n\n  /** Emit when the value is cleared via the clear button */\n  @Output() cleared = new EventEmitter<void>();\n\n  inputValue = '';\n  panelOpen = false;\n  private blurTimeout: ReturnType<typeof setTimeout> | null = null;\n\n  ngOnInit(): void {\n    // Initialize inputValue from value input\n    if (this.value !== undefined && this.value !== null && this.value !== '') {\n      this.inputValue = String(this.value);\n    } else {\n      this.inputValue = '';\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['value']) {\n      const newValue = changes['value'].currentValue;\n      if (newValue !== undefined && newValue !== null && newValue !== '') {\n        const stringValue = String(newValue);\n        if (stringValue !== this.inputValue) {\n          this.inputValue = stringValue;\n        }\n      } else {\n        // Clear input if value is empty/null/undefined\n        if (this.inputValue !== '') {\n          this.inputValue = '';\n        }\n      }\n    }\n  }\n\n  readonly widthClasses: Record<AutocompleteSize, string> = {\n    sm: 'cqa-w-[295px]',\n    md: 'cqa-w-[395px]',\n    lg: 'cqa-w-[495px]',\n  };\n\n  get displayOptions(): CqaAutocompleteOption[] {\n    const list = this.options ?? [];\n    const q = (this.inputValue ?? '').toLowerCase().trim();\n    if (!q) {\n      return list;\n    }\n    return list.filter((opt) => {\n      const label = (opt.label ?? opt.value ?? '').toLowerCase();\n      return label.includes(q);\n    });\n  }\n\n  onInput(event: Event): void {\n    const target = event.target as HTMLInputElement | null;\n    const nextValue = target?.value ?? '';\n    this.inputValue = nextValue;\n    this.valueChange.emit(this.inputValue);\n    this.panelOpen = true;\n  }\n\n  onFocus(): void {\n    if (this.blurTimeout) {\n      clearTimeout(this.blurTimeout);\n      this.blurTimeout = null;\n    }\n    this.panelOpen = true;\n  }\n\n  onBlur(): void {\n    this.blurTimeout = setTimeout(() => {\n      this.panelOpen = false;\n      this.blurTimeout = null;\n    }, 150);\n  }\n\n  selectOption(option: CqaAutocompleteOption): void {\n    const displayLabel = option.label ?? option.value;\n    this.inputValue = displayLabel;\n    this.valueChange.emit(this.inputValue);\n    this.optionSelect.emit(option);\n    this.panelOpen = false;\n  }\n\n  clear(): void {\n    if (this.disabled || this.inputValue === '') {\n      return;\n    }\n    this.inputValue = '';\n    this.valueChange.emit('');\n    this.cleared.emit();\n    this.panelOpen = false;\n  }\n\n  getOptionLabel(option: CqaAutocompleteOption): string {\n    return option.label ?? option.value ?? '';\n  }\n\n  trackByValue(_index: number, option: CqaAutocompleteOption): string {\n    return option.value;\n  }\n\n  @HostListener('document:keydown.escape')\n  onEscape(): void {\n    this.panelOpen = false;\n  }\n}\n","<div\n  class=\"cqa-autocomplete-root cqa-relative\"\n  [style.display]=\"fullWidth ? 'block' : 'inline-block'\"\n  [style.width]=\"fullWidth ? '100%' : 'auto'\">\n  <!-- Field: same styling as search bar -->\n  <div\n    class=\"cqa-inline-flex cqa-items-center cqa-gap-2 cqa-px-6 cqa-py-3 cqa-text-[14px] cqa-border cqa-border-solid cqa-border-gray-200 cqa-rounded-md cqa-bg-white cqa-shadow-sm cqa-transition-colors\"\n    [ngClass]=\"[fullWidth ? 'cqa-w-full' : widthClasses[size]]\">\n    <input\n      type=\"text\"\n      class=\"cqa-flex-1 cqa-min-w-0 cqa-border-none cqa-outline-none cqa-bg-transparent placeholder:cqa-text-gray-300 disabled:cqa-text-gray-400 disabled:cqa-cursor-not-allowed cqa-font-['SF_Pro_Text'] cqa-font-normal cqa-text-[12.3px] cqa-leading-none cqa-tracking-normal cqa-align-middle cqa-text-black-100\"\n      [placeholder]=\"placeholder\"\n      [value]=\"inputValue\"\n      (input)=\"onInput($event)\"\n      (focus)=\"onFocus()\"\n      (blur)=\"onBlur()\"\n      [disabled]=\"disabled\"\n      [attr.aria-label]=\"ariaLabel\"\n      [attr.aria-expanded]=\"panelOpen\"\n      aria-autocomplete=\"list\"\n      aria-haspopup=\"listbox\"\n      autocomplete=\"off\"\n      autocapitalize=\"none\"\n      spellcheck=\"false\"\n      [attr.autofocus]=\"autoFocus ? '' : null\"\n    />\n    <button\n      *ngIf=\"showClear && inputValue\"\n      type=\"button\"\n      class=\"cqa-flex cqa-items-center cqa-justify-center cqa-p-0 cqa-w-4 cqa-h-4 cqa-border-0 cqa-bg-transparent cqa-cursor-pointer cqa-text-gray-500 cqa-hover:cqa-text-gray-700 disabled:cqa-text-gray-300 cqa-transition-colors cqa-leading-none\"\n      (click)=\"clear()\"\n      [disabled]=\"disabled\"\n      aria-label=\"Clear\">\n      <mat-icon\n        class=\"cqa-flex cqa-items-center cqa-justify-center cqa-leading-none cqa-p-0\"\n        [style.width.px]=\"16\"\n        [style.height.px]=\"16\"\n        [style.fontSize.px]=\"16\"\n        [ngClass]=\"{ 'cqa-opacity-[0.38]': disabled }\">\n        close\n      </mat-icon>\n    </button>\n  </div>\n  <!-- List box: below field, same width as field -->\n  <ul\n    *ngIf=\"panelOpen\"\n    role=\"listbox\"\n    class=\"cqa-autocomplete-list cqa-absolute cqa-left-0 cqa-right-0 cqa-top-full cqa-mt-1 cqa-py-1 cqa-border cqa-border-solid cqa-border-gray-200 cqa-rounded-md cqa-bg-white cqa-shadow-sm cqa-max-h-[240px] cqa-overflow-y-auto cqa-scrollbar-thin cqa-z-10 cqa-list-none cqa-m-0 cqa-p-0\"\n    [class.cqa-w-full]=\"fullWidth\">\n    <li\n      *ngFor=\"let option of displayOptions; trackBy: trackByValue\"\n      role=\"option\"\n      (mousedown)=\"selectOption(option); $event.preventDefault()\"\n      class=\"cqa-px-4 cqa-py-2 cqa-text-[12.3px] cqa-leading-none cqa-tracking-normal cqa-text-black-100 cqa-cursor-pointer cqa-font-['SF_Pro_Text'] hover:cqa-bg-gray-100 cqa-transition-colors\">\n      {{ getOptionLabel(option) }}\n    </li>\n    <li\n      *ngIf=\"displayOptions.length === 0\"\n      class=\"cqa-px-4 cqa-py-3 cqa-text-[12.3px] cqa-text-gray-400 cqa-cursor-default\">\n      No matches\n    </li>\n  </ul>\n</div>\n"]}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0b2NvbXBsZXRlLm1vZGVsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9hdXRvY29tcGxldGUvYXV0b2NvbXBsZXRlLm1vZGVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIE9wdGlvbiBmb3IgdGhlIGF1dG9jb21wbGV0ZSBkcm9wZG93bi5cbiAqIFVzZSBsYWJlbCBmb3IgZGlzcGxheSBhbmQgdmFsdWUgZm9yIHRoZSBzZWxlY3RlZCB2YWx1ZSAoZS5nLiBmb3IgSURzKS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBDcWFBdXRvY29tcGxldGVPcHRpb24ge1xuICAvKiogVmFsdWUgZW1pdHRlZCBvbiBzZWxlY3Rpb24gKGUuZy4gaWQgb3IgY29kZSkgKi9cbiAgdmFsdWU6IHN0cmluZztcbiAgLyoqIFRleHQgc2hvd24gaW4gdGhlIGxpc3QgKGRlZmF1bHRzIHRvIHZhbHVlIGlmIG9taXR0ZWQpICovXG4gIGxhYmVsPzogc3RyaW5nO1xufVxuIl19
|
|
@@ -37,18 +37,7 @@ export class StepBuilderDatabaseComponent {
|
|
|
37
37
|
});
|
|
38
38
|
}
|
|
39
39
|
ngOnInit() {
|
|
40
|
-
|
|
41
|
-
if (this.queries.length === 0) {
|
|
42
|
-
this.addNewQuery();
|
|
43
|
-
}
|
|
44
|
-
else {
|
|
45
|
-
this.queries.forEach((query, index) => {
|
|
46
|
-
this.addQueryToForm(query);
|
|
47
|
-
if (index === 0) {
|
|
48
|
-
this.selectedQueryIndex = 0;
|
|
49
|
-
}
|
|
50
|
-
});
|
|
51
|
-
}
|
|
40
|
+
this.initializeFormWithInitialValues();
|
|
52
41
|
}
|
|
53
42
|
ngOnChanges(changes) {
|
|
54
43
|
if (changes['queries'] && !changes['queries'].firstChange) {
|
|
@@ -61,6 +50,27 @@ export class StepBuilderDatabaseComponent {
|
|
|
61
50
|
this.addQueryToForm(query);
|
|
62
51
|
});
|
|
63
52
|
}
|
|
53
|
+
if (changes['initialDbEnvironment'] && changes['initialDbEnvironment'].currentValue) {
|
|
54
|
+
this.databaseForm.patchValue({ dbEnvironment: changes['initialDbEnvironment'].currentValue });
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
initializeFormWithInitialValues() {
|
|
58
|
+
// Set initial DB environment if provided
|
|
59
|
+
if (this.initialDbEnvironment) {
|
|
60
|
+
this.databaseForm.patchValue({ dbEnvironment: this.initialDbEnvironment });
|
|
61
|
+
}
|
|
62
|
+
// Initialize with queries if provided, otherwise add default query
|
|
63
|
+
if (this.queries.length === 0) {
|
|
64
|
+
this.addNewQuery();
|
|
65
|
+
}
|
|
66
|
+
else {
|
|
67
|
+
this.queries.forEach((query, index) => {
|
|
68
|
+
this.addQueryToForm(query);
|
|
69
|
+
if (index === 0) {
|
|
70
|
+
this.selectedQueryIndex = 0;
|
|
71
|
+
}
|
|
72
|
+
});
|
|
73
|
+
}
|
|
64
74
|
}
|
|
65
75
|
get queriesFormArray() {
|
|
66
76
|
return this.databaseForm.get('queries');
|
|
@@ -178,10 +188,10 @@ export class StepBuilderDatabaseComponent {
|
|
|
178
188
|
}
|
|
179
189
|
}
|
|
180
190
|
StepBuilderDatabaseComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepBuilderDatabaseComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component });
|
|
181
|
-
StepBuilderDatabaseComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: StepBuilderDatabaseComponent, selector: "cqa-step-builder-database", inputs: { dbEnvironmentOptions: "dbEnvironmentOptions", queries: "queries", queryResults: "queryResults", isLoading: "isLoading" }, outputs: { createStep: "createStep", cancelled: "cancelled", runQuery: "runQuery", addQuery: "addQuery", deleteQuery: "deleteQuery" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n Create Database Test Step\n </h2>\n\n <!-- DB Environment Section -->\n <div class=\"cqa-flex cqa-items-end cqa-justify-between cqa-mb-6\">\n <div class=\"cqa-flex-1\" style=\"max-width: 280px;\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block\">\n DB Environment <span class=\"cqa-text-red-500\">*</span>\n </label>\n <cqa-dynamic-select [form]=\"databaseForm\" [config]=\"getDbEnvironmentConfig()\">\n </cqa-dynamic-select>\n <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Uses Database environments from Environments.\n </p>\n </div>\n <div class=\"cqa-flex cqa-items-end cqa-mb-1\">\n <cqa-button variant=\"filled\" text=\"Run Query\" [customClass]=\"'cqa-whitespace-nowrap'\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\">\n </cqa-button>\n </div>\n </div>\n\n <!-- Main Content: Two Column Layout -->\n <div class=\"cqa-flex cqa-gap-4 cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n <!-- Left Panel: Query List -->\n <div\n class=\"cqa-w-64 cqa-flex-shrink-0 cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden\"\n style=\"max-width: 174px;\">\n <!-- <div class=\"cqa-p-3 cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Queries</h3>\n </div> -->\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-2\">\n <div *ngFor=\"let query of queries; let i = index\"\n class=\"cqa-flex cqa-items-center cqa-justify-between cqa-p-2 cqa-rounded cqa-cursor-pointer cqa-mb-1\"\n [class.cqa-bg-blue-50]=\"selectedQueryIndex === i\" [class.cqa-border]=\"selectedQueryIndex === i\"\n [class.cqa-border-blue-500]=\"selectedQueryIndex === i\" (click)=\"onSelectQuery(i)\" style=\"height: 49px;\">\n <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n <cqa-badge *ngIf=\"query.status\"\n [label]=\"query.status === 'passed' ? 'Passed' : query.status === 'failed' ? 'Failed' : 'Pending'\"\n [variant]=\"query.status === 'passed' ? 'success' : query.status === 'failed' ? 'error' : 'default'\"\n size=\"small\">\n </cqa-badge>\n </div>\n </div>\n </div>\n\n <!-- Right Panel: Query Editor -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-overflow-auto\">\n <div class=\"cqa-flex-1 cqa-flex cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-gap-4\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-4\" style=\"width: 50%;\">\n <div class=\"\">\n <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900 cqa-text-[14px]\">Query & Store Response</h3>\n </div>\n\n <div class=\"cqa-flex cqa-justify-between\">\n <!-- SQL Query Textarea -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col\">\n <cqa-custom-textarea [placeholder]=\"'Enter your SQL query here...'\"\n [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\" [fullWidth]=\"true\" [rows]=\"8\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n </div>\n </div>\n\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-flex cqa-flex-col cqa-gap-4\">\n\n\n <!-- Variable Input -->\n <div class=\"cqa-flex cqa-gap-4\">\n <div class=\"cqa-flex-1\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block cqa-text-[14px] cqa-font-semibold\">\n Variable\n </label>\n <cqa-custom-input [placeholder]=\"'Variable name'\"\n [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\" [fullWidth]=\"true\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('variable')?.setValue($event)\">\n </cqa-custom-input>\n <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Use letters, numbers, underscore. No spaces. Unique per step. Case sensitive.\n </p>\n </div>\n </div>\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <cqa-button variant=\"outlined\" text=\"Add Query\" icon=\"add\" (clicked)=\"addNewQuery()\">\n </cqa-button>\n <!-- <button type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-10 cqa-h-10 cqa-rounded cqa-text-gray-500 hover:cqa-text-gray-700 hover:cqa-bg-gray-100 cqa-transition-colors cqa-border cqa-border-gray-300\"\n (click)=\"deleteQueryById(getCurrentQuery()?.id || '')\" [disabled]=\"queries.length <= 1\"\n [attr.aria-label]=\"'Delete query'\">\n <mat-icon class=\"cqa-text-lg\">delete</mat-icon>\n </button> -->\n <cqa-button variant=\"text\" icon=\"delete\" (clicked)=\"deleteQueryById(getCurrentQuery()?.id || '')\"\n [disabled]=\"queries.length <= 1\" [customClass]=\"'cqa-text-red-500'\" [attr.aria-label]=\"'Delete query'\">\n </cqa-button>\n <cqa-button variant=\"filled\" text=\"Run\" icon=\"play_arrow\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!getCurrentQueryFormGroup()?.get('query')?.value || !databaseForm.get('dbEnvironment')?.value || isLoading\">\n </cqa-button>\n </div>\n </div>\n </div>\n <!-- Results Section -->\n <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-mb-6\">\n <!-- Tabs -->\n <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'output'\" [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'output'\" (click)=\"onTabChange('output')\">\n Output\n </button>\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'verification'\" (click)=\"onTabChange('verification')\">\n Verification\n </button>\n </div>\n\n <!-- Tab Content -->\n <div class=\"cqa-p-4\">\n <!-- Output Tab -->\n <div *ngIf=\"selectedTab === 'output'\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-mb-3\">\n <h4 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Query Results</h4>\n <cqa-button variant=\"text\" text=\"Copy\" icon=\"content_copy\" [customClass]=\"'cqa-text-blue-600'\"\n (clicked)=\"onCopyResults()\">\n </cqa-button>\n </div>\n\n <!-- Results Table -->\n <div *ngIf=\"queryResults && queryResults.length > 0\" class=\"cqa-overflow-x-auto\">\n <table class=\"cqa-w-full cqa-border-collapse\">\n <thead>\n <tr class=\"cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n <th *ngFor=\"let key of getTableColumns()\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-left cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">\n {{ key }}\n </th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let row of queryResults\" class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n <td *ngFor=\"let key of getTableColumns()\" class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-text-gray-900\">\n {{ row[key] }}\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n\n <div *ngIf=\"!queryResults || queryResults.length === 0\"\n class=\"cqa-text-center cqa-py-8 cqa-text-gray-400 cqa-text-sm\">\n No results yet. Run a query to see results.\n </div>\n </div>\n\n <!-- Verification Tab -->\n <div *ngIf=\"selectedTab === 'verification'\">\n <p class=\"cqa-text-sm cqa-text-gray-600\">Verification settings will be available here.</p>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n\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\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button class=\"cqa-w-1/2\" variant=\"filled\" text=\"Create Step\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>", components: [{ type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }, { type: i3.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }, { type: i4.BadgeComponent, selector: "cqa-badge", inputs: ["type", "label", "icon", "iconLibrary", "variant", "size", "backgroundColor", "textColor", "borderColor", "iconBackgroundColor", "iconColor", "iconSize", "inlineStyles", "key", "value", "keyTextColor", "valueTextColor", "isLoading"] }, { 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.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"] }], directives: [{ type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
|
|
191
|
+
StepBuilderDatabaseComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: StepBuilderDatabaseComponent, selector: "cqa-step-builder-database", inputs: { dbEnvironmentOptions: "dbEnvironmentOptions", queries: "queries", queryResults: "queryResults", isLoading: "isLoading", initialDbEnvironment: "initialDbEnvironment", editMode: "editMode" }, outputs: { createStep: "createStep", cancelled: "cancelled", runQuery: "runQuery", addQuery: "addQuery", deleteQuery: "deleteQuery" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n {{ editMode ? 'Update Database Test Step' : 'Create Database Test Step' }}\n </h2>\n\n <!-- DB Environment Section (separator line only on bottom) -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-pb-[0.5rem] cqa-mb-4 cqa-border-0 cqa-border-b cqa-border-solid cqa-border-gray-200\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-block\">\n DB Environment <span class=\"cqa-text-red-500\">*</span>\n </label>\n <div class=\"cqa-flex cqa-w-full cqa-flex-nowrap cqa-items-center cqa-justify-between cqa-gap-3\">\n <div class=\"cqa-min-w-0\" style=\"width: 280px;\">\n <cqa-dynamic-select [form]=\"databaseForm\" [config]=\"getDbEnvironmentConfig()\">\n </cqa-dynamic-select>\n </div>\n <cqa-button variant=\"filled\" text=\"Run Query\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\"\n [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[24px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[37px] hover:!cqa-bg-[#1B1FEB] cqa-flex-shrink-0'\">\n </cqa-button>\n </div>\n <p class=\"cqa-text-xs cqa-text-gray-500\">\n Uses Database environments from Environments.\n </p>\n </div>\n\n <!-- Main Content: Two Column Layout -->\n <div class=\"cqa-flex cqa-gap-[12px] cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n <!-- Left Panel: Query List -->\n <div\n class=\"cqa-w-[175px] cqa-min-w-[175px] cqa-flex-shrink-0 cqa-flex cqa-flex-col cqa-border-0 cqa-border-r cqa-border-r-[1px] cqa-border-solid cqa-border-[#E0E0E0] cqa-bg-[#FAFAFA] cqa-overflow-hidden\">\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-2\">\n <div *ngFor=\"let query of queries; let i = index\"\n class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-[10px] cqa-w-full cqa-pt-[14px] cqa-pb-[14px] cqa-pl-[16px] cqa-pr-[16px] cqa-rounded-[8px] cqa-cursor-pointer cqa-mb-1\"\n [class.cqa-bg-blue-50]=\"selectedQueryIndex === i\"\n [class.cqa-border]=\"selectedQueryIndex === i\"\n [class.cqa-border-blue-500]=\"selectedQueryIndex === i\"\n [class.cqa-bg-grey-400]=\"selectedQueryIndex !== i\"\n (click)=\"onSelectQuery(i)\"\n style=\"height: 49px; min-height: 49px;\">\n <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n <cqa-badge *ngIf=\"query.status\"\n [label]=\"query.status === 'passed' ? 'Passed' : query.status === 'failed' ? 'Failed' : 'Pending'\"\n [variant]=\"query.status === 'passed' ? 'success' : query.status === 'failed' ? 'error' : 'default'\"\n size=\"small\">\n </cqa-badge>\n </div>\n </div>\n </div>\n\n <!-- Right Panel: Query Editor -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-overflow-auto\">\n <div class=\"cqa-flex-1 cqa-flex cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-gap-[10px] cqa-mb-4\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-[6px]\" style=\"width: 50%;\">\n <div class=\"\">\n <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900 cqa-text-[14px]\">Query & Store Response</h3>\n </div>\n\n <div class=\"cqa-flex cqa-justify-between\">\n <!-- SQL Query Textarea -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-min-w-0 cqa-w-full cqa-rounded-md cqa-bg-gray-50/50\">\n <cqa-custom-textarea [placeholder]=\"'Enter your SQL query here...'\"\n [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\" [fullWidth]=\"true\" [rows]=\"8\"\n [textareaInlineStyle]=\"'padding: 1rem;'\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n </div>\n </div>\n\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-flex cqa-flex-col cqa-gap-[10px] cqa-mb-[8px]\">\n\n\n <!-- Variable Input -->\n <div class=\"cqa-flex cqa-gap-4\">\n <div class=\"cqa-flex-1\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block cqa-text-[14px] cqa-font-semibold\">\n Variable\n </label>\n <cqa-custom-input [placeholder]=\"'Variable name'\"\n [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\" [fullWidth]=\"true\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('variable')?.setValue($event)\">\n </cqa-custom-input>\n <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Use letters, numbers, underscore. No spaces. Unique per step. Case sensitive.\n </p>\n </div>\n </div>\n <!-- Action Buttons (pinned to bottom) -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-mt-auto\">\n <cqa-button variant=\"outlined\" text=\"Add Query\" icon=\"add\" (clicked)=\"addNewQuery()\"\n [customClass]=\"'cqa-bg-white !cqa-border-[#414146] cqa-text-sm cqa-font-semibold cqa-text-[#414146] cqa-whitespace-nowrap !cqa-py-[10px] !cqa-px-[10px] !cqa-gap-[8px] cqa-rounded-lg hover:!cqa-border-[#414146] hover:cqa-bg-gray-50'\"\n [iconColor]=\"'#414146'\">\n </cqa-button>\n <cqa-button type=\"button\" variant=\"text\"\n [customClass]=\"'cqa-min-w-[40px] cqa-w-10 cqa-h-9 cqa-rounded cqa-bg-gray-100 cqa-flex cqa-items-center cqa-justify-center hover:cqa-bg-gray-200 cqa-p-0'\"\n [tooltip]=\"'Delete query'\" (clicked)=\"deleteQueryById(getCurrentQuery()?.id || '')\"\n [disabled]=\"queries.length <= 1\" [attr.aria-label]=\"'Delete query'\">\n <svg class=\"cqa-flex-shrink-0\" 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=\"#6B7280\" />\n </svg>\n </cqa-button>\n <cqa-button variant=\"filled\" text=\"Run\" icon=\"play_arrow\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!getCurrentQueryFormGroup()?.get('query')?.value || !databaseForm.get('dbEnvironment')?.value || isLoading\"\n [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[16px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[38px] hover:!cqa-bg-[#1B1FEB] !cqa-border-none'\"\n [iconColor]=\"'#FBFCFF'\">\n </cqa-button>\n </div>\n </div>\n </div>\n <!-- Results Section -->\n <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-mb-6\">\n <!-- Tabs -->\n <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'output'\" [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'output'\" (click)=\"onTabChange('output')\">\n Output\n </button>\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'verification'\" (click)=\"onTabChange('verification')\">\n Verification\n </button>\n </div>\n\n <!-- Tab Content -->\n <div class=\"cqa-p-4\">\n <!-- Output Tab -->\n <div *ngIf=\"selectedTab === 'output'\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-mb-3\">\n <h4 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Query Results</h4>\n <cqa-button variant=\"text\" text=\"Copy\" icon=\"content_copy\" [customClass]=\"'cqa-text-blue-600'\"\n (clicked)=\"onCopyResults()\">\n </cqa-button>\n </div>\n\n <!-- Results Table -->\n <div *ngIf=\"queryResults && queryResults.length > 0\" class=\"cqa-overflow-x-auto\">\n <table class=\"cqa-w-full cqa-border-collapse\">\n <thead>\n <tr class=\"cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n <th *ngFor=\"let key of getTableColumns()\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-left cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">\n {{ key }}\n </th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let row of queryResults\" class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n <td *ngFor=\"let key of getTableColumns()\" class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-text-gray-900\">\n {{ row[key] }}\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n\n <div *ngIf=\"!queryResults || queryResults.length === 0\"\n class=\"cqa-text-center cqa-py-8 cqa-text-gray-400 cqa-text-sm\">\n No results yet. Run a query to see results.\n </div>\n </div>\n\n <!-- Verification Tab -->\n <div *ngIf=\"selectedTab === 'verification'\">\n <p class=\"cqa-text-sm cqa-text-gray-600\">Verification settings will be available here.</p>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n\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\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button class=\"cqa-w-1/2\" variant=\"filled\" [text]=\"editMode ? 'Update Step' : 'Create Step'\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>", components: [{ type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }, { type: i3.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }, { type: i4.BadgeComponent, selector: "cqa-badge", inputs: ["type", "label", "icon", "iconLibrary", "variant", "size", "backgroundColor", "textColor", "borderColor", "iconBackgroundColor", "iconColor", "iconSize", "inlineStyles", "key", "value", "keyTextColor", "valueTextColor", "isLoading"] }, { 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.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"] }], directives: [{ type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
|
|
182
192
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepBuilderDatabaseComponent, decorators: [{
|
|
183
193
|
type: Component,
|
|
184
|
-
args: [{ selector: 'cqa-step-builder-database', host: { class: 'cqa-ui-root' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n Create Database Test Step\n </h2>\n\n <!-- DB Environment Section -->\n <div class=\"cqa-flex cqa-items-end cqa-justify-between cqa-mb-6\">\n <div class=\"cqa-flex-1\" style=\"max-width: 280px;\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block\">\n DB Environment <span class=\"cqa-text-red-500\">*</span>\n </label>\n <cqa-dynamic-select [form]=\"databaseForm\" [config]=\"getDbEnvironmentConfig()\">\n </cqa-dynamic-select>\n <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Uses Database environments from Environments.\n </p>\n </div>\n <div class=\"cqa-flex cqa-items-end cqa-mb-1\">\n <cqa-button variant=\"filled\" text=\"Run Query\" [customClass]=\"'cqa-whitespace-nowrap'\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\">\n </cqa-button>\n </div>\n </div>\n\n <!-- Main Content: Two Column Layout -->\n <div class=\"cqa-flex cqa-gap-4 cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n <!-- Left Panel: Query List -->\n <div\n class=\"cqa-w-64 cqa-flex-shrink-0 cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden\"\n style=\"max-width: 174px;\">\n <!-- <div class=\"cqa-p-3 cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Queries</h3>\n </div> -->\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-2\">\n <div *ngFor=\"let query of queries; let i = index\"\n class=\"cqa-flex cqa-items-center cqa-justify-between cqa-p-2 cqa-rounded cqa-cursor-pointer cqa-mb-1\"\n [class.cqa-bg-blue-50]=\"selectedQueryIndex === i\" [class.cqa-border]=\"selectedQueryIndex === i\"\n [class.cqa-border-blue-500]=\"selectedQueryIndex === i\" (click)=\"onSelectQuery(i)\" style=\"height: 49px;\">\n <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n <cqa-badge *ngIf=\"query.status\"\n [label]=\"query.status === 'passed' ? 'Passed' : query.status === 'failed' ? 'Failed' : 'Pending'\"\n [variant]=\"query.status === 'passed' ? 'success' : query.status === 'failed' ? 'error' : 'default'\"\n size=\"small\">\n </cqa-badge>\n </div>\n </div>\n </div>\n\n <!-- Right Panel: Query Editor -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-overflow-auto\">\n <div class=\"cqa-flex-1 cqa-flex cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-gap-4\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-4\" style=\"width: 50%;\">\n <div class=\"\">\n <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900 cqa-text-[14px]\">Query & Store Response</h3>\n </div>\n\n <div class=\"cqa-flex cqa-justify-between\">\n <!-- SQL Query Textarea -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col\">\n <cqa-custom-textarea [placeholder]=\"'Enter your SQL query here...'\"\n [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\" [fullWidth]=\"true\" [rows]=\"8\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n </div>\n </div>\n\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-flex cqa-flex-col cqa-gap-4\">\n\n\n <!-- Variable Input -->\n <div class=\"cqa-flex cqa-gap-4\">\n <div class=\"cqa-flex-1\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block cqa-text-[14px] cqa-font-semibold\">\n Variable\n </label>\n <cqa-custom-input [placeholder]=\"'Variable name'\"\n [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\" [fullWidth]=\"true\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('variable')?.setValue($event)\">\n </cqa-custom-input>\n <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Use letters, numbers, underscore. No spaces. Unique per step. Case sensitive.\n </p>\n </div>\n </div>\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <cqa-button variant=\"outlined\" text=\"Add Query\" icon=\"add\" (clicked)=\"addNewQuery()\">\n </cqa-button>\n <!-- <button type=\"button\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-10 cqa-h-10 cqa-rounded cqa-text-gray-500 hover:cqa-text-gray-700 hover:cqa-bg-gray-100 cqa-transition-colors cqa-border cqa-border-gray-300\"\n (click)=\"deleteQueryById(getCurrentQuery()?.id || '')\" [disabled]=\"queries.length <= 1\"\n [attr.aria-label]=\"'Delete query'\">\n <mat-icon class=\"cqa-text-lg\">delete</mat-icon>\n </button> -->\n <cqa-button variant=\"text\" icon=\"delete\" (clicked)=\"deleteQueryById(getCurrentQuery()?.id || '')\"\n [disabled]=\"queries.length <= 1\" [customClass]=\"'cqa-text-red-500'\" [attr.aria-label]=\"'Delete query'\">\n </cqa-button>\n <cqa-button variant=\"filled\" text=\"Run\" icon=\"play_arrow\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!getCurrentQueryFormGroup()?.get('query')?.value || !databaseForm.get('dbEnvironment')?.value || isLoading\">\n </cqa-button>\n </div>\n </div>\n </div>\n <!-- Results Section -->\n <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-mb-6\">\n <!-- Tabs -->\n <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'output'\" [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'output'\" (click)=\"onTabChange('output')\">\n Output\n </button>\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'verification'\" (click)=\"onTabChange('verification')\">\n Verification\n </button>\n </div>\n\n <!-- Tab Content -->\n <div class=\"cqa-p-4\">\n <!-- Output Tab -->\n <div *ngIf=\"selectedTab === 'output'\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-mb-3\">\n <h4 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Query Results</h4>\n <cqa-button variant=\"text\" text=\"Copy\" icon=\"content_copy\" [customClass]=\"'cqa-text-blue-600'\"\n (clicked)=\"onCopyResults()\">\n </cqa-button>\n </div>\n\n <!-- Results Table -->\n <div *ngIf=\"queryResults && queryResults.length > 0\" class=\"cqa-overflow-x-auto\">\n <table class=\"cqa-w-full cqa-border-collapse\">\n <thead>\n <tr class=\"cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n <th *ngFor=\"let key of getTableColumns()\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-left cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">\n {{ key }}\n </th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let row of queryResults\" class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n <td *ngFor=\"let key of getTableColumns()\" class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-text-gray-900\">\n {{ row[key] }}\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n\n <div *ngIf=\"!queryResults || queryResults.length === 0\"\n class=\"cqa-text-center cqa-py-8 cqa-text-gray-400 cqa-text-sm\">\n No results yet. Run a query to see results.\n </div>\n </div>\n\n <!-- Verification Tab -->\n <div *ngIf=\"selectedTab === 'verification'\">\n <p class=\"cqa-text-sm cqa-text-gray-600\">Verification settings will be available here.</p>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n\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\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button class=\"cqa-w-1/2\" variant=\"filled\" text=\"Create Step\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>", styles: [] }]
|
|
194
|
+
args: [{ selector: 'cqa-step-builder-database', host: { class: 'cqa-ui-root' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n <!-- Header -->\n <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n {{ editMode ? 'Update Database Test Step' : 'Create Database Test Step' }}\n </h2>\n\n <!-- DB Environment Section (separator line only on bottom) -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-pb-[0.5rem] cqa-mb-4 cqa-border-0 cqa-border-b cqa-border-solid cqa-border-gray-200\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-block\">\n DB Environment <span class=\"cqa-text-red-500\">*</span>\n </label>\n <div class=\"cqa-flex cqa-w-full cqa-flex-nowrap cqa-items-center cqa-justify-between cqa-gap-3\">\n <div class=\"cqa-min-w-0\" style=\"width: 280px;\">\n <cqa-dynamic-select [form]=\"databaseForm\" [config]=\"getDbEnvironmentConfig()\">\n </cqa-dynamic-select>\n </div>\n <cqa-button variant=\"filled\" text=\"Run Query\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\"\n [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[24px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[37px] hover:!cqa-bg-[#1B1FEB] cqa-flex-shrink-0'\">\n </cqa-button>\n </div>\n <p class=\"cqa-text-xs cqa-text-gray-500\">\n Uses Database environments from Environments.\n </p>\n </div>\n\n <!-- Main Content: Two Column Layout -->\n <div class=\"cqa-flex cqa-gap-[12px] cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n <!-- Left Panel: Query List -->\n <div\n class=\"cqa-w-[175px] cqa-min-w-[175px] cqa-flex-shrink-0 cqa-flex cqa-flex-col cqa-border-0 cqa-border-r cqa-border-r-[1px] cqa-border-solid cqa-border-[#E0E0E0] cqa-bg-[#FAFAFA] cqa-overflow-hidden\">\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-2\">\n <div *ngFor=\"let query of queries; let i = index\"\n class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-[10px] cqa-w-full cqa-pt-[14px] cqa-pb-[14px] cqa-pl-[16px] cqa-pr-[16px] cqa-rounded-[8px] cqa-cursor-pointer cqa-mb-1\"\n [class.cqa-bg-blue-50]=\"selectedQueryIndex === i\"\n [class.cqa-border]=\"selectedQueryIndex === i\"\n [class.cqa-border-blue-500]=\"selectedQueryIndex === i\"\n [class.cqa-bg-grey-400]=\"selectedQueryIndex !== i\"\n (click)=\"onSelectQuery(i)\"\n style=\"height: 49px; min-height: 49px;\">\n <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n <cqa-badge *ngIf=\"query.status\"\n [label]=\"query.status === 'passed' ? 'Passed' : query.status === 'failed' ? 'Failed' : 'Pending'\"\n [variant]=\"query.status === 'passed' ? 'success' : query.status === 'failed' ? 'error' : 'default'\"\n size=\"small\">\n </cqa-badge>\n </div>\n </div>\n </div>\n\n <!-- Right Panel: Query Editor -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-overflow-auto\">\n <div class=\"cqa-flex-1 cqa-flex cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-gap-[10px] cqa-mb-4\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-[6px]\" style=\"width: 50%;\">\n <div class=\"\">\n <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900 cqa-text-[14px]\">Query & Store Response</h3>\n </div>\n\n <div class=\"cqa-flex cqa-justify-between\">\n <!-- SQL Query Textarea -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-min-w-0 cqa-w-full cqa-rounded-md cqa-bg-gray-50/50\">\n <cqa-custom-textarea [placeholder]=\"'Enter your SQL query here...'\"\n [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\" [fullWidth]=\"true\" [rows]=\"8\"\n [textareaInlineStyle]=\"'padding: 1rem;'\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\n </div>\n </div>\n\n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-flex cqa-flex-col cqa-gap-[10px] cqa-mb-[8px]\">\n\n\n <!-- Variable Input -->\n <div class=\"cqa-flex cqa-gap-4\">\n <div class=\"cqa-flex-1\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block cqa-text-[14px] cqa-font-semibold\">\n Variable\n </label>\n <cqa-custom-input [placeholder]=\"'Variable name'\"\n [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\" [fullWidth]=\"true\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('variable')?.setValue($event)\">\n </cqa-custom-input>\n <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Use letters, numbers, underscore. No spaces. Unique per step. Case sensitive.\n </p>\n </div>\n </div>\n <!-- Action Buttons (pinned to bottom) -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-mt-auto\">\n <cqa-button variant=\"outlined\" text=\"Add Query\" icon=\"add\" (clicked)=\"addNewQuery()\"\n [customClass]=\"'cqa-bg-white !cqa-border-[#414146] cqa-text-sm cqa-font-semibold cqa-text-[#414146] cqa-whitespace-nowrap !cqa-py-[10px] !cqa-px-[10px] !cqa-gap-[8px] cqa-rounded-lg hover:!cqa-border-[#414146] hover:cqa-bg-gray-50'\"\n [iconColor]=\"'#414146'\">\n </cqa-button>\n <cqa-button type=\"button\" variant=\"text\"\n [customClass]=\"'cqa-min-w-[40px] cqa-w-10 cqa-h-9 cqa-rounded cqa-bg-gray-100 cqa-flex cqa-items-center cqa-justify-center hover:cqa-bg-gray-200 cqa-p-0'\"\n [tooltip]=\"'Delete query'\" (clicked)=\"deleteQueryById(getCurrentQuery()?.id || '')\"\n [disabled]=\"queries.length <= 1\" [attr.aria-label]=\"'Delete query'\">\n <svg class=\"cqa-flex-shrink-0\" 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=\"#6B7280\" />\n </svg>\n </cqa-button>\n <cqa-button variant=\"filled\" text=\"Run\" icon=\"play_arrow\" (clicked)=\"onRunQuery()\"\n [disabled]=\"!getCurrentQueryFormGroup()?.get('query')?.value || !databaseForm.get('dbEnvironment')?.value || isLoading\"\n [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[16px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[38px] hover:!cqa-bg-[#1B1FEB] !cqa-border-none'\"\n [iconColor]=\"'#FBFCFF'\">\n </cqa-button>\n </div>\n </div>\n </div>\n <!-- Results Section -->\n <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-mb-6\">\n <!-- Tabs -->\n <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'output'\" [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'output'\" (click)=\"onTabChange('output')\">\n Output\n </button>\n <button type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'verification'\" (click)=\"onTabChange('verification')\">\n Verification\n </button>\n </div>\n\n <!-- Tab Content -->\n <div class=\"cqa-p-4\">\n <!-- Output Tab -->\n <div *ngIf=\"selectedTab === 'output'\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-mb-3\">\n <h4 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Query Results</h4>\n <cqa-button variant=\"text\" text=\"Copy\" icon=\"content_copy\" [customClass]=\"'cqa-text-blue-600'\"\n (clicked)=\"onCopyResults()\">\n </cqa-button>\n </div>\n\n <!-- Results Table -->\n <div *ngIf=\"queryResults && queryResults.length > 0\" class=\"cqa-overflow-x-auto\">\n <table class=\"cqa-w-full cqa-border-collapse\">\n <thead>\n <tr class=\"cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n <th *ngFor=\"let key of getTableColumns()\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-left cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">\n {{ key }}\n </th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let row of queryResults\" class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n <td *ngFor=\"let key of getTableColumns()\" class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-text-gray-900\">\n {{ row[key] }}\n </td>\n </tr>\n </tbody>\n </table>\n </div>\n\n <div *ngIf=\"!queryResults || queryResults.length === 0\"\n class=\"cqa-text-center cqa-py-8 cqa-text-gray-400 cqa-text-sm\">\n No results yet. Run a query to see results.\n </div>\n </div>\n\n <!-- Verification Tab -->\n <div *ngIf=\"selectedTab === 'verification'\">\n <p class=\"cqa-text-sm cqa-text-gray-600\">Verification settings will be available here.</p>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n\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\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button class=\"cqa-w-1/2\" variant=\"filled\" [text]=\"editMode ? 'Update Step' : 'Create Step'\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>", styles: [] }]
|
|
185
195
|
}], ctorParameters: function () { return [{ type: i1.FormBuilder }]; }, propDecorators: { dbEnvironmentOptions: [{
|
|
186
196
|
type: Input
|
|
187
197
|
}], queries: [{
|
|
@@ -190,6 +200,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
|
|
|
190
200
|
type: Input
|
|
191
201
|
}], isLoading: [{
|
|
192
202
|
type: Input
|
|
203
|
+
}], initialDbEnvironment: [{
|
|
204
|
+
type: Input
|
|
205
|
+
}], editMode: [{
|
|
206
|
+
type: Input
|
|
193
207
|
}], createStep: [{
|
|
194
208
|
type: Output
|
|
195
209
|
}], cancelled: [{
|
|
@@ -201,4 +215,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
|
|
|
201
215
|
}], deleteQuery: [{
|
|
202
216
|
type: Output
|
|
203
217
|
}] } });
|
|
204
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"step-builder-database.component.js","sourceRoot":"","sources":["../../../../../../src/lib/step-builder/step-builder-database/step-builder-database.component.ts","../../../../../../src/lib/step-builder/step-builder-database/step-builder-database.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,EAAE,YAAY,EAAoC,MAAM,eAAe,CAAC;AACzG,OAAO,EAAkD,UAAU,EAAE,MAAM,gBAAgB,CAAC;;;;;;;;;AAqB5F,MAAM,OAAO,4BAA4B;IAgCvC,YAAoB,EAAe;QAAf,OAAE,GAAF,EAAE,CAAa;QA/BnC,0CAA0C;QACjC,yBAAoB,GAAmB,EAAE,CAAC;QAEnD,2BAA2B;QAClB,YAAO,GAAoB,EAAE,CAAC;QAEvC,yBAAyB;QAChB,iBAAY,GAAU,EAAE,CAAC;QAElC,oBAAoB;QACX,cAAS,GAAY,KAAK,CAAC;QAEpC,gCAAgC;QACtB,eAAU,GAAG,IAAI,YAAY,EAAoB,CAAC;QAE5D,0BAA0B;QAChB,cAAS,GAAG,IAAI,YAAY,EAAQ,CAAC;QAE/C,6BAA6B;QACnB,aAAQ,GAAG,IAAI,YAAY,EAA8D,CAAC;QAEpG,+BAA+B;QACrB,aAAQ,GAAG,IAAI,YAAY,EAAQ,CAAC;QAE9C,iCAAiC;QACvB,gBAAW,GAAG,IAAI,YAAY,EAAU,CAAC;QAGnD,gBAAW,GAA8B,QAAQ,CAAC;QAClD,uBAAkB,GAAW,CAAC,CAAC;QAG7B,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,EAAE,CAAC,KAAK,CAAC;YAChC,aAAa,EAAE,CAAC,EAAE,EAAE,UAAU,CAAC,QAAQ,CAAC;YACxC,OAAO,EAAE,IAAI,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC;SAC3B,CAAC,CAAC;IACL,CAAC;IAED,QAAQ;QACN,iDAAiD;QACjD,IAAI,IAAI,CAAC,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE;YAC7B,IAAI,CAAC,WAAW,EAAE,CAAC;SACpB;aAAM;YACL,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,KAAK,EAAE,EAAE;gBACpC,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;gBAC3B,IAAI,KAAK,KAAK,CAAC,EAAE;oBACf,IAAI,CAAC,kBAAkB,GAAG,CAAC,CAAC;iBAC7B;YACH,CAAC,CAAC,CAAC;SACJ;IACH,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,WAAW,EAAE;YACzD,yCAAyC;YACzC,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,SAAS,CAAc,CAAC;YACnE,OAAO,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE;gBAChC,YAAY,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;aAC1B;YACD,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;gBAC3B,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;YAC7B,CAAC,CAAC,CAAC;SACJ;IACH,CAAC;IAED,IAAI,gBAAgB;QAClB,OAAO,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,SAAS,CAAc,CAAC;IACvD,CAAC;IAED,WAAW;QACT,MAAM,QAAQ,GAAkB;YAC9B,EAAE,EAAE,SAAS,IAAI,CAAC,GAAG,EAAE,EAAE;YACzB,KAAK,EAAE,EAAE;YACT,QAAQ,EAAE,EAAE;YACZ,MAAM,EAAE,SAAS;SAClB,CAAC;QACF,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC5B,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,CAAC;QAC9B,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC;QAClD,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;IACvB,CAAC;IAEO,cAAc,CAAC,KAAoB;QACzC,MAAM,UAAU,GAAG,IAAI,CAAC,EAAE,CAAC,KAAK,CAAC;YAC/B,EAAE,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC;YACd,KAAK,EAAE,CAAC,KAAK,CAAC,KAAK,EAAE,UAAU,CAAC,QAAQ,CAAC;YACzC,QAAQ,EAAE,CAAC,KAAK,CAAC,QAAQ,EAAE,UAAU,CAAC,QAAQ,CAAC;SAChD,CAAC,CAAC;QACH,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IACzC,CAAC;IAED,eAAe,CAAC,OAAe;QAC7B,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,OAAO,CAAC,CAAC;QAC5D,IAAI,KAAK,KAAK,CAAC,CAAC,EAAE;YAChB,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;YAC9B,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;YACtC,IAAI,IAAI,CAAC,kBAAkB,IAAI,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE;gBAClD,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;aAChE;YACD,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;SAChC;IACH,CAAC;IAED,sBAAsB;QACpB,OAAO;YACL,GAAG,EAAE,eAAe;YACpB,WAAW,EAAE,uBAAuB;YACpC,QAAQ,EAAE,KAAK;YACf,UAAU,EAAE,KAAK;YACjB,OAAO,EAAE,IAAI,CAAC,oBAAoB;SACnC,CAAC;IACJ,CAAC;IAED,eAAe;QACb,IAAI,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,IAAI,IAAI,CAAC,kBAAkB,IAAI,CAAC,IAAI,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE;YAC5G,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;SAC9C;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED,wBAAwB;QACtB,IAAI,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,kBAAkB,IAAI,CAAC,IAAI,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,gBAAgB,CAAC,MAAM,EAAE;YACnH,OAAO,IAAI,CAAC,gBAAgB,CAAC,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAc,CAAC;SACvE;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED,UAAU;QACR,MAAM,aAAa,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,eAAe,CAAC,EAAE,KAAK,CAAC;QACpE,MAAM,YAAY,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;QAC5C,MAAM,cAAc,GAAG,IAAI,CAAC,wBAAwB,EAAE,CAAC;QAEvD,IAAI,YAAY,IAAI,cAAc,IAAI,aAAa,EAAE;YACnD,MAAM,KAAK,GAAG,cAAc,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,KAAK,IAAI,EAAE,CAAC;YACvD,MAAM,QAAQ,GAAG,cAAc,CAAC,GAAG,CAAC,UAAU,CAAC,EAAE,KAAK,IAAI,EAAE,CAAC;YAE7D,0BAA0B;YAC1B,YAAY,CAAC,KAAK,GAAG,KAAK,CAAC;YAC3B,YAAY,CAAC,QAAQ,GAAG,QAAQ,CAAC;YAEjC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC;gBACjB,KAAK;gBACL,QAAQ;gBACR,aAAa;aACd,CAAC,CAAC;SACJ;IACH,CAAC;IAED,aAAa,CAAC,KAAa;QACzB,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;IAClC,CAAC;IAED,WAAW,CAAC,GAA8B;QACxC,IAAI,CAAC,WAAW,GAAG,GAAG,CAAC;IACzB,CAAC;IAED,aAAa;QACX,kCAAkC;QAClC,IAAI,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE;YACrD,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,YAAY,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;YACxD,SAAS,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE;gBAC5C,mDAAmD;YACrD,CAAC,CAAC,CAAC;SACJ;IACH,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;IACxB,CAAC;IAED,YAAY;QACV,IAAI,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE;YAC3B,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC;YAC1C,MAAM,QAAQ,GAAqB;gBACjC,aAAa,EAAE,SAAS,CAAC,aAAa;gBACtC,OAAO,EAAE,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,KAAK,EAAE,EAAE;oBACzC,MAAM,cAAc,GAAG,IAAI,CAAC,gBAAgB,CAAC,EAAE,CAAC,KAAK,CAAc,CAAC;oBACpE,OAAO;wBACL,EAAE,EAAE,KAAK,CAAC,EAAE;wBACZ,KAAK,EAAE,cAAc,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,KAAK,IAAI,EAAE;wBAC/C,QAAQ,EAAE,cAAc,CAAC,GAAG,CAAC,UAAU,CAAC,EAAE,KAAK,IAAI,EAAE;wBACrD,MAAM,EAAE,KAAK,CAAC,MAAM;qBACrB,CAAC;gBACJ,CAAC,CAAC;aACH,CAAC;YACF,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SAChC;IACH,CAAC;IAED,eAAe;QACb,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE;YACxD,OAAO,EAAE,CAAC;SACX;QACD,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC;IAC3C,CAAC;;yHAnMU,4BAA4B;6GAA5B,4BAA4B,0YCtBzC,4gTAwLM;2FDlKO,4BAA4B;kBANxC,SAAS;+BACE,2BAA2B,QAG/B,EAAE,KAAK,EAAE,aAAa,EAAE;kGAIrB,oBAAoB;sBAA5B,KAAK;gBAGG,OAAO;sBAAf,KAAK;gBAGG,YAAY;sBAApB,KAAK;gBAGG,SAAS;sBAAjB,KAAK;gBAGI,UAAU;sBAAnB,MAAM;gBAGG,SAAS;sBAAlB,MAAM;gBAGG,QAAQ;sBAAjB,MAAM;gBAGG,QAAQ;sBAAjB,MAAM;gBAGG,WAAW;sBAApB,MAAM","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, OnChanges, SimpleChanges } from '@angular/core';\nimport { FormBuilder, FormGroup, FormArray, FormControl, Validators } from '@angular/forms';\nimport { DynamicSelectFieldConfig, SelectOption } from '../../dynamic-select/dynamic-select-field.component';\n\nexport interface DatabaseQuery {\n  id: string;\n  query: string;\n  variable: string;\n  status?: 'passed' | 'failed' | 'pending';\n}\n\nexport interface DatabaseFormData {\n  dbEnvironment: string;\n  queries: DatabaseQuery[];\n}\n\n@Component({\n  selector: 'cqa-step-builder-database',\n  templateUrl: './step-builder-database.component.html',\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' }\n})\nexport class StepBuilderDatabaseComponent implements OnInit, OnChanges {\n  /** Options for DB environment dropdown */\n  @Input() dbEnvironmentOptions: SelectOption[] = [];\n\n  /** Initial queries list */\n  @Input() queries: DatabaseQuery[] = [];\n\n  /** Query results data */\n  @Input() queryResults: any[] = [];\n\n  /** Loading state */\n  @Input() isLoading: boolean = false;\n\n  /** Emit when step is created */\n  @Output() createStep = new EventEmitter<DatabaseFormData>();\n\n  /** Emit when cancelled */\n  @Output() cancelled = new EventEmitter<void>();\n\n  /** Emit when query is run */\n  @Output() runQuery = new EventEmitter<{ query: string; variable: string; dbEnvironment: string }>();\n\n  /** Emit when query is added */\n  @Output() addQuery = new EventEmitter<void>();\n\n  /** Emit when query is deleted */\n  @Output() deleteQuery = new EventEmitter<string>();\n\n  databaseForm: FormGroup;\n  selectedTab: 'output' | 'verification' = 'output';\n  selectedQueryIndex: number = 0;\n\n  constructor(private fb: FormBuilder) {\n    this.databaseForm = this.fb.group({\n      dbEnvironment: ['', Validators.required],\n      queries: this.fb.array([])\n    });\n  }\n\n  ngOnInit(): void {\n    // Initialize with default query if none provided\n    if (this.queries.length === 0) {\n      this.addNewQuery();\n    } else {\n      this.queries.forEach((query, index) => {\n        this.addQueryToForm(query);\n        if (index === 0) {\n          this.selectedQueryIndex = 0;\n        }\n      });\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['queries'] && !changes['queries'].firstChange) {\n      // Rebuild form array when queries change\n      const queriesArray = this.databaseForm.get('queries') as FormArray;\n      while (queriesArray.length !== 0) {\n        queriesArray.removeAt(0);\n      }\n      this.queries.forEach(query => {\n        this.addQueryToForm(query);\n      });\n    }\n  }\n\n  get queriesFormArray(): FormArray {\n    return this.databaseForm.get('queries') as FormArray;\n  }\n\n  addNewQuery(): void {\n    const newQuery: DatabaseQuery = {\n      id: `query_${Date.now()}`,\n      query: '',\n      variable: '',\n      status: 'pending'\n    };\n    this.queries.push(newQuery);\n    this.addQueryToForm(newQuery);\n    this.selectedQueryIndex = this.queries.length - 1;\n    this.addQuery.emit();\n  }\n\n  private addQueryToForm(query: DatabaseQuery): void {\n    const queryGroup = this.fb.group({\n      id: [query.id],\n      query: [query.query, Validators.required],\n      variable: [query.variable, Validators.required]\n    });\n    this.queriesFormArray.push(queryGroup);\n  }\n\n  deleteQueryById(queryId: string): void {\n    const index = this.queries.findIndex(q => q.id === queryId);\n    if (index !== -1) {\n      this.queries.splice(index, 1);\n      this.queriesFormArray.removeAt(index);\n      if (this.selectedQueryIndex >= this.queries.length) {\n        this.selectedQueryIndex = Math.max(0, this.queries.length - 1);\n      }\n      this.deleteQuery.emit(queryId);\n    }\n  }\n\n  getDbEnvironmentConfig(): DynamicSelectFieldConfig {\n    return {\n      key: 'dbEnvironment',\n      placeholder: 'Select DB environment',\n      multiple: false,\n      searchable: false,\n      options: this.dbEnvironmentOptions\n    };\n  }\n\n  getCurrentQuery(): DatabaseQuery | null {\n    if (this.queries.length > 0 && this.selectedQueryIndex >= 0 && this.selectedQueryIndex < this.queries.length) {\n      return this.queries[this.selectedQueryIndex];\n    }\n    return null;\n  }\n\n  getCurrentQueryFormGroup(): FormGroup | null {\n    if (this.queriesFormArray && this.selectedQueryIndex >= 0 && this.selectedQueryIndex < this.queriesFormArray.length) {\n      return this.queriesFormArray.at(this.selectedQueryIndex) as FormGroup;\n    }\n    return null;\n  }\n\n  onRunQuery(): void {\n    const dbEnvironment = this.databaseForm.get('dbEnvironment')?.value;\n    const currentQuery = this.getCurrentQuery();\n    const queryFormGroup = this.getCurrentQueryFormGroup();\n\n    if (currentQuery && queryFormGroup && dbEnvironment) {\n      const query = queryFormGroup.get('query')?.value || '';\n      const variable = queryFormGroup.get('variable')?.value || '';\n      \n      // Update the query object\n      currentQuery.query = query;\n      currentQuery.variable = variable;\n\n      this.runQuery.emit({\n        query,\n        variable,\n        dbEnvironment\n      });\n    }\n  }\n\n  onSelectQuery(index: number): void {\n    this.selectedQueryIndex = index;\n  }\n\n  onTabChange(tab: 'output' | 'verification'): void {\n    this.selectedTab = tab;\n  }\n\n  onCopyResults(): void {\n    // Copy query results to clipboard\n    if (this.queryResults && this.queryResults.length > 0) {\n      const text = JSON.stringify(this.queryResults, null, 2);\n      navigator.clipboard.writeText(text).then(() => {\n        // Could emit an event or show a toast notification\n      });\n    }\n  }\n\n  onCancel(): void {\n    this.cancelled.emit();\n  }\n\n  onCreateStep(): void {\n    if (this.databaseForm.valid) {\n      const formValue = this.databaseForm.value;\n      const stepData: DatabaseFormData = {\n        dbEnvironment: formValue.dbEnvironment,\n        queries: this.queries.map((query, index) => {\n          const queryFormGroup = this.queriesFormArray.at(index) as FormGroup;\n          return {\n            id: query.id,\n            query: queryFormGroup.get('query')?.value || '',\n            variable: queryFormGroup.get('variable')?.value || '',\n            status: query.status\n          };\n        })\n      };\n      this.createStep.emit(stepData);\n    }\n  }\n\n  getTableColumns(): string[] {\n    if (!this.queryResults || this.queryResults.length === 0) {\n      return [];\n    }\n    return Object.keys(this.queryResults[0]);\n  }\n}\n\n","<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n  <!-- Header -->\n  <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n    Create Database Test Step\n  </h2>\n\n  <!-- DB Environment Section -->\n  <div class=\"cqa-flex cqa-items-end cqa-justify-between cqa-mb-6\">\n    <div class=\"cqa-flex-1\" style=\"max-width: 280px;\">\n      <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block\">\n        DB Environment <span class=\"cqa-text-red-500\">*</span>\n      </label>\n      <cqa-dynamic-select [form]=\"databaseForm\" [config]=\"getDbEnvironmentConfig()\">\n      </cqa-dynamic-select>\n      <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n        Uses Database environments from Environments.\n      </p>\n    </div>\n    <div class=\"cqa-flex cqa-items-end cqa-mb-1\">\n      <cqa-button variant=\"filled\" text=\"Run Query\" [customClass]=\"'cqa-whitespace-nowrap'\" (clicked)=\"onRunQuery()\"\n        [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\">\n      </cqa-button>\n    </div>\n  </div>\n\n  <!-- Main Content: Two Column Layout -->\n  <div class=\"cqa-flex cqa-gap-4 cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n    <!-- Left Panel: Query List -->\n    <div\n      class=\"cqa-w-64 cqa-flex-shrink-0 cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden\"\n      style=\"max-width: 174px;\">\n      <!-- <div class=\"cqa-p-3 cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n        <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Queries</h3>\n      </div> -->\n      <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-2\">\n        <div *ngFor=\"let query of queries; let i = index\"\n          class=\"cqa-flex cqa-items-center cqa-justify-between cqa-p-2 cqa-rounded cqa-cursor-pointer cqa-mb-1\"\n          [class.cqa-bg-blue-50]=\"selectedQueryIndex === i\" [class.cqa-border]=\"selectedQueryIndex === i\"\n          [class.cqa-border-blue-500]=\"selectedQueryIndex === i\" (click)=\"onSelectQuery(i)\" style=\"height: 49px;\">\n          <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n          <cqa-badge *ngIf=\"query.status\"\n            [label]=\"query.status === 'passed' ? 'Passed' : query.status === 'failed' ? 'Failed' : 'Pending'\"\n            [variant]=\"query.status === 'passed' ? 'success' : query.status === 'failed' ? 'error' : 'default'\"\n            size=\"small\">\n          </cqa-badge>\n        </div>\n      </div>\n    </div>\n\n    <!-- Right Panel: Query Editor -->\n    <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-overflow-auto\">\n      <div class=\"cqa-flex-1 cqa-flex cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-gap-4\">\n        <div class=\"cqa-flex cqa-flex-col cqa-gap-4\" style=\"width: 50%;\">\n          <div class=\"\">\n            <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900 cqa-text-[14px]\">Query & Store Response</h3>\n          </div>\n\n          <div class=\"cqa-flex cqa-justify-between\">\n            <!-- SQL Query Textarea -->\n            <div class=\"cqa-flex-1 cqa-flex cqa-flex-col\">\n              <cqa-custom-textarea [placeholder]=\"'Enter your SQL query here...'\"\n                [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\" [fullWidth]=\"true\" [rows]=\"8\"\n                (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n              </cqa-custom-textarea>\n            </div>\n          </div>\n        </div>\n\n        <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-flex cqa-flex-col cqa-gap-4\">\n\n\n          <!-- Variable Input -->\n          <div class=\"cqa-flex cqa-gap-4\">\n            <div class=\"cqa-flex-1\">\n              <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block cqa-text-[14px] cqa-font-semibold\">\n                Variable\n              </label>\n              <cqa-custom-input [placeholder]=\"'Variable name'\"\n                [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\" [fullWidth]=\"true\"\n                (valueChange)=\"getCurrentQueryFormGroup()?.get('variable')?.setValue($event)\">\n              </cqa-custom-input>\n              <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n                Use letters, numbers, underscore. No spaces. Unique per step. Case sensitive.\n              </p>\n            </div>\n          </div>\n          <!-- Action Buttons -->\n          <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n            <cqa-button variant=\"outlined\" text=\"Add Query\" icon=\"add\" (clicked)=\"addNewQuery()\">\n            </cqa-button>\n            <!-- <button type=\"button\"\n              class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-10 cqa-h-10 cqa-rounded cqa-text-gray-500 hover:cqa-text-gray-700 hover:cqa-bg-gray-100 cqa-transition-colors cqa-border cqa-border-gray-300\"\n              (click)=\"deleteQueryById(getCurrentQuery()?.id || '')\" [disabled]=\"queries.length <= 1\"\n              [attr.aria-label]=\"'Delete query'\">\n              <mat-icon class=\"cqa-text-lg\">delete</mat-icon>\n            </button> -->\n            <cqa-button variant=\"text\" icon=\"delete\" (clicked)=\"deleteQueryById(getCurrentQuery()?.id || '')\"\n              [disabled]=\"queries.length <= 1\" [customClass]=\"'cqa-text-red-500'\" [attr.aria-label]=\"'Delete query'\">\n            </cqa-button>\n            <cqa-button variant=\"filled\" text=\"Run\" icon=\"play_arrow\" (clicked)=\"onRunQuery()\"\n              [disabled]=\"!getCurrentQueryFormGroup()?.get('query')?.value || !databaseForm.get('dbEnvironment')?.value || isLoading\">\n            </cqa-button>\n          </div>\n        </div>\n      </div>\n      <!-- Results Section -->\n      <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-mb-6\">\n        <!-- Tabs -->\n        <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n          <button type=\"button\"\n            class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n            [class.cqa-text-blue-600]=\"selectedTab === 'output'\" [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n            [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n            [class.cqa-border-transparent]=\"selectedTab !== 'output'\" (click)=\"onTabChange('output')\">\n            Output\n          </button>\n          <button type=\"button\"\n            class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n            [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n            [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n            [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n            [class.cqa-border-transparent]=\"selectedTab !== 'verification'\" (click)=\"onTabChange('verification')\">\n            Verification\n          </button>\n        </div>\n\n        <!-- Tab Content -->\n        <div class=\"cqa-p-4\">\n          <!-- Output Tab -->\n          <div *ngIf=\"selectedTab === 'output'\">\n            <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-mb-3\">\n              <h4 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Query Results</h4>\n              <cqa-button variant=\"text\" text=\"Copy\" icon=\"content_copy\" [customClass]=\"'cqa-text-blue-600'\"\n                (clicked)=\"onCopyResults()\">\n              </cqa-button>\n            </div>\n\n            <!-- Results Table -->\n            <div *ngIf=\"queryResults && queryResults.length > 0\" class=\"cqa-overflow-x-auto\">\n              <table class=\"cqa-w-full cqa-border-collapse\">\n                <thead>\n                  <tr class=\"cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n                    <th *ngFor=\"let key of getTableColumns()\"\n                      class=\"cqa-px-4 cqa-py-2 cqa-text-left cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">\n                      {{ key }}\n                    </th>\n                  </tr>\n                </thead>\n                <tbody>\n                  <tr *ngFor=\"let row of queryResults\" class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n                    <td *ngFor=\"let key of getTableColumns()\" class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-text-gray-900\">\n                      {{ row[key] }}\n                    </td>\n                  </tr>\n                </tbody>\n              </table>\n            </div>\n\n            <div *ngIf=\"!queryResults || queryResults.length === 0\"\n              class=\"cqa-text-center cqa-py-8 cqa-text-gray-400 cqa-text-sm\">\n              No results yet. Run a query to see results.\n            </div>\n          </div>\n\n          <!-- Verification Tab -->\n          <div *ngIf=\"selectedTab === 'verification'\">\n            <p class=\"cqa-text-sm cqa-text-gray-600\">Verification settings will be available here.</p>\n          </div>\n        </div>\n      </div>\n    </div>\n  </div>\n\n\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\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCancel()\">\n    </cqa-button>\n    <cqa-button class=\"cqa-w-1/2\" variant=\"filled\" text=\"Create Step\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCreateStep()\">\n    </cqa-button>\n  </div>\n</div>"]}
|
|
218
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"step-builder-database.component.js","sourceRoot":"","sources":["../../../../../../src/lib/step-builder/step-builder-database/step-builder-database.component.ts","../../../../../../src/lib/step-builder/step-builder-database/step-builder-database.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, OnChanges, SimpleChanges } from '@angular/core';\nimport { FormBuilder, FormGroup, FormArray, FormControl, Validators } from '@angular/forms';\nimport { DynamicSelectFieldConfig, SelectOption } from '../../dynamic-select/dynamic-select-field.component';\n\nexport interface DatabaseQuery {\n  id: string;\n  query: string;\n  variable: string;\n  status?: 'passed' | 'failed' | 'pending';\n}\n\nexport interface DatabaseFormData {\n  dbEnvironment: string;\n  queries: DatabaseQuery[];\n}\n\n@Component({\n  selector: 'cqa-step-builder-database',\n  templateUrl: './step-builder-database.component.html',\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' }\n})\nexport class StepBuilderDatabaseComponent implements OnInit, OnChanges {\n  /** Options for DB environment dropdown */\n  @Input() dbEnvironmentOptions: SelectOption[] = [];\n\n  /** Initial queries list */\n  @Input() queries: DatabaseQuery[] = [];\n\n  /** Query results data */\n  @Input() queryResults: any[] = [];\n\n  /** Loading state */\n  @Input() isLoading: boolean = false;\n\n  /** Initial DB environment for edit mode */\n  @Input() initialDbEnvironment?: string;\n\n  /** Whether in edit mode */\n  @Input() editMode?: boolean;\n\n  /** Emit when step is created */\n  @Output() createStep = new EventEmitter<DatabaseFormData>();\n\n  /** Emit when cancelled */\n  @Output() cancelled = new EventEmitter<void>();\n\n  /** Emit when query is run */\n  @Output() runQuery = new EventEmitter<{ query: string; variable: string; dbEnvironment: string }>();\n\n  /** Emit when query is added */\n  @Output() addQuery = new EventEmitter<void>();\n\n  /** Emit when query is deleted */\n  @Output() deleteQuery = new EventEmitter<string>();\n\n  databaseForm: FormGroup;\n  selectedTab: 'output' | 'verification' = 'output';\n  selectedQueryIndex: number = 0;\n\n  constructor(private fb: FormBuilder) {\n    this.databaseForm = this.fb.group({\n      dbEnvironment: ['', Validators.required],\n      queries: this.fb.array([])\n    });\n  }\n\n  ngOnInit(): void {\n    this.initializeFormWithInitialValues();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['queries'] && !changes['queries'].firstChange) {\n      // Rebuild form array when queries change\n      const queriesArray = this.databaseForm.get('queries') as FormArray;\n      while (queriesArray.length !== 0) {\n        queriesArray.removeAt(0);\n      }\n      this.queries.forEach(query => {\n        this.addQueryToForm(query);\n      });\n    }\n    if (changes['initialDbEnvironment'] && changes['initialDbEnvironment'].currentValue) {\n      this.databaseForm.patchValue({ dbEnvironment: changes['initialDbEnvironment'].currentValue });\n    }\n  }\n\n  private initializeFormWithInitialValues(): void {\n    // Set initial DB environment if provided\n    if (this.initialDbEnvironment) {\n      this.databaseForm.patchValue({ dbEnvironment: this.initialDbEnvironment });\n    }\n\n    // Initialize with queries if provided, otherwise add default query\n    if (this.queries.length === 0) {\n      this.addNewQuery();\n    } else {\n      this.queries.forEach((query, index) => {\n        this.addQueryToForm(query);\n        if (index === 0) {\n          this.selectedQueryIndex = 0;\n        }\n      });\n    }\n  }\n\n  get queriesFormArray(): FormArray {\n    return this.databaseForm.get('queries') as FormArray;\n  }\n\n  addNewQuery(): void {\n    const newQuery: DatabaseQuery = {\n      id: `query_${Date.now()}`,\n      query: '',\n      variable: '',\n      status: 'pending'\n    };\n    this.queries.push(newQuery);\n    this.addQueryToForm(newQuery);\n    this.selectedQueryIndex = this.queries.length - 1;\n    this.addQuery.emit();\n  }\n\n  private addQueryToForm(query: DatabaseQuery): void {\n    const queryGroup = this.fb.group({\n      id: [query.id],\n      query: [query.query, Validators.required],\n      variable: [query.variable, Validators.required]\n    });\n    this.queriesFormArray.push(queryGroup);\n  }\n\n  deleteQueryById(queryId: string): void {\n    const index = this.queries.findIndex(q => q.id === queryId);\n    if (index !== -1) {\n      this.queries.splice(index, 1);\n      this.queriesFormArray.removeAt(index);\n      if (this.selectedQueryIndex >= this.queries.length) {\n        this.selectedQueryIndex = Math.max(0, this.queries.length - 1);\n      }\n      this.deleteQuery.emit(queryId);\n    }\n  }\n\n  getDbEnvironmentConfig(): DynamicSelectFieldConfig {\n    return {\n      key: 'dbEnvironment',\n      placeholder: 'Select DB environment',\n      multiple: false,\n      searchable: false,\n      options: this.dbEnvironmentOptions\n    };\n  }\n\n  getCurrentQuery(): DatabaseQuery | null {\n    if (this.queries.length > 0 && this.selectedQueryIndex >= 0 && this.selectedQueryIndex < this.queries.length) {\n      return this.queries[this.selectedQueryIndex];\n    }\n    return null;\n  }\n\n  getCurrentQueryFormGroup(): FormGroup | null {\n    if (this.queriesFormArray && this.selectedQueryIndex >= 0 && this.selectedQueryIndex < this.queriesFormArray.length) {\n      return this.queriesFormArray.at(this.selectedQueryIndex) as FormGroup;\n    }\n    return null;\n  }\n\n  onRunQuery(): void {\n    const dbEnvironment = this.databaseForm.get('dbEnvironment')?.value;\n    const currentQuery = this.getCurrentQuery();\n    const queryFormGroup = this.getCurrentQueryFormGroup();\n\n    if (currentQuery && queryFormGroup && dbEnvironment) {\n      const query = queryFormGroup.get('query')?.value || '';\n      const variable = queryFormGroup.get('variable')?.value || '';\n      \n      // Update the query object\n      currentQuery.query = query;\n      currentQuery.variable = variable;\n\n      this.runQuery.emit({\n        query,\n        variable,\n        dbEnvironment\n      });\n    }\n  }\n\n  onSelectQuery(index: number): void {\n    this.selectedQueryIndex = index;\n  }\n\n  onTabChange(tab: 'output' | 'verification'): void {\n    this.selectedTab = tab;\n  }\n\n  onCopyResults(): void {\n    // Copy query results to clipboard\n    if (this.queryResults && this.queryResults.length > 0) {\n      const text = JSON.stringify(this.queryResults, null, 2);\n      navigator.clipboard.writeText(text).then(() => {\n        // Could emit an event or show a toast notification\n      });\n    }\n  }\n\n  onCancel(): void {\n    this.cancelled.emit();\n  }\n\n  onCreateStep(): void {\n    if (this.databaseForm.valid) {\n      const formValue = this.databaseForm.value;\n      const stepData: DatabaseFormData = {\n        dbEnvironment: formValue.dbEnvironment,\n        queries: this.queries.map((query, index) => {\n          const queryFormGroup = this.queriesFormArray.at(index) as FormGroup;\n          return {\n            id: query.id,\n            query: queryFormGroup.get('query')?.value || '',\n            variable: queryFormGroup.get('variable')?.value || '',\n            status: query.status\n          };\n        })\n      };\n      this.createStep.emit(stepData);\n    }\n  }\n\n  getTableColumns(): string[] {\n    if (!this.queryResults || this.queryResults.length === 0) {\n      return [];\n    }\n    return Object.keys(this.queryResults[0]);\n  }\n}\n\n","<div class=\"cqa-flex cqa-flex-col cqa-h-full cqa-bg-white cqa-px-4 cqa-py-2\">\n  <!-- Header -->\n  <h2 class=\"cqa-text-[12px] cqa-font-semibold cqa-text-black-100 cqa-mb-4\">\n    {{ editMode ? 'Update Database Test Step' : 'Create Database Test Step' }}\n  </h2>\n\n  <!-- DB Environment Section (separator line only on bottom) -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-pb-[0.5rem] cqa-mb-4 cqa-border-0 cqa-border-b cqa-border-solid cqa-border-gray-200\">\n    <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-block\">\n      DB Environment <span class=\"cqa-text-red-500\">*</span>\n    </label>\n    <div class=\"cqa-flex cqa-w-full cqa-flex-nowrap cqa-items-center cqa-justify-between cqa-gap-3\">\n      <div class=\"cqa-min-w-0\" style=\"width: 280px;\">\n        <cqa-dynamic-select [form]=\"databaseForm\" [config]=\"getDbEnvironmentConfig()\">\n        </cqa-dynamic-select>\n      </div>\n      <cqa-button variant=\"filled\" text=\"Run Query\" (clicked)=\"onRunQuery()\"\n        [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\"\n        [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[24px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[37px] hover:!cqa-bg-[#1B1FEB] cqa-flex-shrink-0'\">\n      </cqa-button>\n    </div>\n    <p class=\"cqa-text-xs cqa-text-gray-500\">\n      Uses Database environments from Environments.\n    </p>\n  </div>\n\n  <!-- Main Content: Two Column Layout -->\n  <div class=\"cqa-flex cqa-gap-[12px] cqa-flex-1 cqa-overflow-hidden cqa-mb-6\">\n    <!-- Left Panel: Query List -->\n    <div\n      class=\"cqa-w-[175px] cqa-min-w-[175px] cqa-flex-shrink-0 cqa-flex cqa-flex-col cqa-border-0 cqa-border-r cqa-border-r-[1px] cqa-border-solid cqa-border-[#E0E0E0] cqa-bg-[#FAFAFA] cqa-overflow-hidden\">\n      <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-2\">\n        <div *ngFor=\"let query of queries; let i = index\"\n          class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-[10px] cqa-w-full cqa-pt-[14px] cqa-pb-[14px] cqa-pl-[16px] cqa-pr-[16px] cqa-rounded-[8px] cqa-cursor-pointer cqa-mb-1\"\n          [class.cqa-bg-blue-50]=\"selectedQueryIndex === i\"\n          [class.cqa-border]=\"selectedQueryIndex === i\"\n          [class.cqa-border-blue-500]=\"selectedQueryIndex === i\"\n          [class.cqa-bg-grey-400]=\"selectedQueryIndex !== i\"\n          (click)=\"onSelectQuery(i)\"\n          style=\"height: 49px; min-height: 49px;\">\n          <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n          <cqa-badge *ngIf=\"query.status\"\n            [label]=\"query.status === 'passed' ? 'Passed' : query.status === 'failed' ? 'Failed' : 'Pending'\"\n            [variant]=\"query.status === 'passed' ? 'success' : query.status === 'failed' ? 'error' : 'default'\"\n            size=\"small\">\n          </cqa-badge>\n        </div>\n      </div>\n    </div>\n\n    <!-- Right Panel: Query Editor -->\n    <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-overflow-auto\">\n      <div class=\"cqa-flex-1 cqa-flex cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-gap-[10px] cqa-mb-4\">\n        <div class=\"cqa-flex cqa-flex-col cqa-gap-[6px]\" style=\"width: 50%;\">\n          <div class=\"\">\n            <h3 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900 cqa-text-[14px]\">Query & Store Response</h3>\n          </div>\n\n          <div class=\"cqa-flex cqa-justify-between\">\n            <!-- SQL Query Textarea -->\n            <div class=\"cqa-flex-1 cqa-flex cqa-flex-col cqa-min-w-0 cqa-w-full cqa-rounded-md cqa-bg-gray-50/50\">\n              <cqa-custom-textarea [placeholder]=\"'Enter your SQL query here...'\"\n                [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\" [fullWidth]=\"true\" [rows]=\"8\"\n                [textareaInlineStyle]=\"'padding: 1rem;'\"\n                (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n              </cqa-custom-textarea>\n            </div>\n          </div>\n        </div>\n\n        <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-flex cqa-flex-col cqa-gap-[10px] cqa-mb-[8px]\">\n\n\n          <!-- Variable Input -->\n          <div class=\"cqa-flex cqa-gap-4\">\n            <div class=\"cqa-flex-1\">\n              <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700 cqa-mb-1.5 cqa-block cqa-text-[14px] cqa-font-semibold\">\n                Variable\n              </label>\n              <cqa-custom-input [placeholder]=\"'Variable name'\"\n                [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\" [fullWidth]=\"true\"\n                (valueChange)=\"getCurrentQueryFormGroup()?.get('variable')?.setValue($event)\">\n              </cqa-custom-input>\n              <p class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n                Use letters, numbers, underscore. No spaces. Unique per step. Case sensitive.\n              </p>\n            </div>\n          </div>\n          <!-- Action Buttons (pinned to bottom) -->\n          <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-mt-auto\">\n            <cqa-button variant=\"outlined\" text=\"Add Query\" icon=\"add\" (clicked)=\"addNewQuery()\"\n              [customClass]=\"'cqa-bg-white !cqa-border-[#414146] cqa-text-sm cqa-font-semibold cqa-text-[#414146] cqa-whitespace-nowrap !cqa-py-[10px] !cqa-px-[10px] !cqa-gap-[8px] cqa-rounded-lg hover:!cqa-border-[#414146] hover:cqa-bg-gray-50'\"\n              [iconColor]=\"'#414146'\">\n            </cqa-button>\n            <cqa-button type=\"button\" variant=\"text\"\n              [customClass]=\"'cqa-min-w-[40px] cqa-w-10 cqa-h-9 cqa-rounded cqa-bg-gray-100 cqa-flex cqa-items-center cqa-justify-center hover:cqa-bg-gray-200 cqa-p-0'\"\n              [tooltip]=\"'Delete query'\" (clicked)=\"deleteQueryById(getCurrentQuery()?.id || '')\"\n              [disabled]=\"queries.length <= 1\" [attr.aria-label]=\"'Delete query'\">\n              <svg class=\"cqa-flex-shrink-0\" 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=\"#6B7280\" />\n              </svg>\n            </cqa-button>\n            <cqa-button variant=\"filled\" text=\"Run\" icon=\"play_arrow\" (clicked)=\"onRunQuery()\"\n              [disabled]=\"!getCurrentQueryFormGroup()?.get('query')?.value || !databaseForm.get('dbEnvironment')?.value || isLoading\"\n              [customClass]=\"'!cqa-bg-[#3F43EE] cqa-text-[#FBFCFF] cqa-text-sm cqa-font-semibold cqa-whitespace-nowrap !cqa-pt-[10px] !cqa-pb-[10px] !cqa-pl-[16px] !cqa-pr-[24px] !cqa-gap-[8px] cqa-rounded-lg cqa-min-h-[38px] hover:!cqa-bg-[#1B1FEB] !cqa-border-none'\"\n              [iconColor]=\"'#FBFCFF'\">\n            </cqa-button>\n          </div>\n        </div>\n      </div>\n      <!-- Results Section -->\n      <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-mb-6\">\n        <!-- Tabs -->\n        <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n          <button type=\"button\"\n            class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n            [class.cqa-text-blue-600]=\"selectedTab === 'output'\" [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n            [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n            [class.cqa-border-transparent]=\"selectedTab !== 'output'\" (click)=\"onTabChange('output')\">\n            Output\n          </button>\n          <button type=\"button\"\n            class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n            [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n            [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n            [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n            [class.cqa-border-transparent]=\"selectedTab !== 'verification'\" (click)=\"onTabChange('verification')\">\n            Verification\n          </button>\n        </div>\n\n        <!-- Tab Content -->\n        <div class=\"cqa-p-4\">\n          <!-- Output Tab -->\n          <div *ngIf=\"selectedTab === 'output'\">\n            <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-mb-3\">\n              <h4 class=\"cqa-text-sm cqa-font-semibold cqa-text-gray-900\">Query Results</h4>\n              <cqa-button variant=\"text\" text=\"Copy\" icon=\"content_copy\" [customClass]=\"'cqa-text-blue-600'\"\n                (clicked)=\"onCopyResults()\">\n              </cqa-button>\n            </div>\n\n            <!-- Results Table -->\n            <div *ngIf=\"queryResults && queryResults.length > 0\" class=\"cqa-overflow-x-auto\">\n              <table class=\"cqa-w-full cqa-border-collapse\">\n                <thead>\n                  <tr class=\"cqa-bg-gray-50 cqa-border-b cqa-border-gray-200\">\n                    <th *ngFor=\"let key of getTableColumns()\"\n                      class=\"cqa-px-4 cqa-py-2 cqa-text-left cqa-text-xs cqa-font-semibold cqa-text-gray-700 cqa-uppercase\">\n                      {{ key }}\n                    </th>\n                  </tr>\n                </thead>\n                <tbody>\n                  <tr *ngFor=\"let row of queryResults\" class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n                    <td *ngFor=\"let key of getTableColumns()\" class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-text-gray-900\">\n                      {{ row[key] }}\n                    </td>\n                  </tr>\n                </tbody>\n              </table>\n            </div>\n\n            <div *ngIf=\"!queryResults || queryResults.length === 0\"\n              class=\"cqa-text-center cqa-py-8 cqa-text-gray-400 cqa-text-sm\">\n              No results yet. Run a query to see results.\n            </div>\n          </div>\n\n          <!-- Verification Tab -->\n          <div *ngIf=\"selectedTab === 'verification'\">\n            <p class=\"cqa-text-sm cqa-text-gray-600\">Verification settings will be available here.</p>\n          </div>\n        </div>\n      </div>\n    </div>\n  </div>\n\n\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\" variant=\"outlined\" text=\"Cancel\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCancel()\">\n    </cqa-button>\n    <cqa-button class=\"cqa-w-1/2\" variant=\"filled\" [text]=\"editMode ? 'Update Step' : 'Create Step'\" [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCreateStep()\">\n    </cqa-button>\n  </div>\n</div>"]}
|