@cqa-lib/cqa-ui 1.1.175 → 1.1.177
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/compare-runs/compare-runs.component.mjs +1 -1
- package/esm2020/lib/dynamic-select/dynamic-select-field.component.mjs +3 -3
- package/esm2020/lib/execution-screen/db-query-execution-item/db-query-execution-item.component.mjs +70 -5
- package/esm2020/lib/execution-screen/db-verification-step/db-verification-step.component.mjs +15 -3
- package/esm2020/lib/iterations-loop/iterations-loop.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-action/step-builder-action.component.mjs +29 -9
- package/esm2020/lib/step-builder/step-builder-condition/step-builder-condition.component.mjs +330 -0
- package/esm2020/lib/step-builder/step-builder-database/step-builder-database.component.mjs +205 -0
- package/esm2020/lib/table/dynamic-table/dynamic-table.component.mjs +35 -3
- package/esm2020/lib/templates/table-template.component.mjs +7 -4
- package/esm2020/lib/test-case-details/custom-edit-step/custom-edit-step-data.mjs +5 -0
- package/esm2020/lib/test-case-details/custom-edit-step/custom-edit-step-ref.mjs +32 -0
- package/esm2020/lib/test-case-details/custom-edit-step/custom-edit-step.component.mjs +71 -0
- package/esm2020/lib/test-case-details/custom-edit-step/custom-edit-step.service.mjs +97 -0
- package/esm2020/lib/test-case-details/normal-step/normal-step.component.mjs +59 -9
- package/esm2020/lib/ui-kit.module.mjs +18 -3
- package/esm2020/lib/utils/tw-overlay-container.mjs +6 -3
- package/esm2020/public-api.mjs +7 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +939 -34
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +926 -34
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/dynamic-select/dynamic-select-field.component.d.ts +1 -0
- package/lib/execution-screen/db-query-execution-item/db-query-execution-item.component.d.ts +18 -3
- package/lib/execution-screen/db-verification-step/db-verification-step.component.d.ts +7 -1
- package/lib/step-builder/step-builder-action/step-builder-action.component.d.ts +2 -1
- package/lib/step-builder/step-builder-condition/step-builder-condition.component.d.ts +61 -0
- package/lib/step-builder/step-builder-database/step-builder-database.component.d.ts +61 -0
- package/lib/table/dynamic-table/dynamic-table.component.d.ts +7 -1
- package/lib/templates/table-template.component.d.ts +6 -1
- package/lib/test-case-details/custom-edit-step/custom-edit-step-data.d.ts +8 -0
- package/lib/test-case-details/custom-edit-step/custom-edit-step-ref.d.ts +13 -0
- package/lib/test-case-details/custom-edit-step/custom-edit-step.component.d.ts +25 -0
- package/lib/test-case-details/custom-edit-step/custom-edit-step.service.d.ts +23 -0
- package/lib/test-case-details/normal-step/normal-step.component.d.ts +9 -0
- package/lib/ui-kit.module.d.ts +26 -23
- package/package.json +1 -1
- package/public-api.d.ts +6 -0
- package/styles.css +1 -1
|
@@ -0,0 +1,205 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
2
|
+
import { Validators } from '@angular/forms';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/forms";
|
|
5
|
+
import * as i2 from "../../dynamic-select/dynamic-select-field.component";
|
|
6
|
+
import * as i3 from "../../button/button.component";
|
|
7
|
+
import * as i4 from "../../badge/badge.component";
|
|
8
|
+
import * as i5 from "../../custom-textarea/custom-textarea.component";
|
|
9
|
+
import * as i6 from "../../custom-input/custom-input.component";
|
|
10
|
+
import * as i7 from "@angular/material/icon";
|
|
11
|
+
import * as i8 from "@angular/common";
|
|
12
|
+
export class StepBuilderDatabaseComponent {
|
|
13
|
+
constructor(fb) {
|
|
14
|
+
this.fb = fb;
|
|
15
|
+
/** Options for DB environment dropdown */
|
|
16
|
+
this.dbEnvironmentOptions = [];
|
|
17
|
+
/** Initial queries list */
|
|
18
|
+
this.queries = [];
|
|
19
|
+
/** Query results data */
|
|
20
|
+
this.queryResults = [];
|
|
21
|
+
/** Loading state */
|
|
22
|
+
this.isLoading = false;
|
|
23
|
+
/** Emit when step is created */
|
|
24
|
+
this.createStep = new EventEmitter();
|
|
25
|
+
/** Emit when cancelled */
|
|
26
|
+
this.cancelled = new EventEmitter();
|
|
27
|
+
/** Emit when query is run */
|
|
28
|
+
this.runQuery = new EventEmitter();
|
|
29
|
+
/** Emit when query is added */
|
|
30
|
+
this.addQuery = new EventEmitter();
|
|
31
|
+
/** Emit when query is deleted */
|
|
32
|
+
this.deleteQuery = new EventEmitter();
|
|
33
|
+
this.selectedTab = 'output';
|
|
34
|
+
this.selectedQueryIndex = 0;
|
|
35
|
+
this.databaseForm = this.fb.group({
|
|
36
|
+
dbEnvironment: ['', Validators.required],
|
|
37
|
+
queries: this.fb.array([])
|
|
38
|
+
});
|
|
39
|
+
}
|
|
40
|
+
ngOnInit() {
|
|
41
|
+
// Initialize with default query if none provided
|
|
42
|
+
if (this.queries.length === 0) {
|
|
43
|
+
this.addNewQuery();
|
|
44
|
+
}
|
|
45
|
+
else {
|
|
46
|
+
this.queries.forEach((query, index) => {
|
|
47
|
+
this.addQueryToForm(query);
|
|
48
|
+
if (index === 0) {
|
|
49
|
+
this.selectedQueryIndex = 0;
|
|
50
|
+
}
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
ngOnChanges(changes) {
|
|
55
|
+
if (changes['queries'] && !changes['queries'].firstChange) {
|
|
56
|
+
// Rebuild form array when queries change
|
|
57
|
+
const queriesArray = this.databaseForm.get('queries');
|
|
58
|
+
while (queriesArray.length !== 0) {
|
|
59
|
+
queriesArray.removeAt(0);
|
|
60
|
+
}
|
|
61
|
+
this.queries.forEach(query => {
|
|
62
|
+
this.addQueryToForm(query);
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
get queriesFormArray() {
|
|
67
|
+
return this.databaseForm.get('queries');
|
|
68
|
+
}
|
|
69
|
+
addNewQuery() {
|
|
70
|
+
const newQuery = {
|
|
71
|
+
id: `query_${Date.now()}`,
|
|
72
|
+
query: '',
|
|
73
|
+
variable: '',
|
|
74
|
+
status: 'pending'
|
|
75
|
+
};
|
|
76
|
+
this.queries.push(newQuery);
|
|
77
|
+
this.addQueryToForm(newQuery);
|
|
78
|
+
this.selectedQueryIndex = this.queries.length - 1;
|
|
79
|
+
this.addQuery.emit();
|
|
80
|
+
}
|
|
81
|
+
addQueryToForm(query) {
|
|
82
|
+
const queryGroup = this.fb.group({
|
|
83
|
+
id: [query.id],
|
|
84
|
+
query: [query.query, Validators.required],
|
|
85
|
+
variable: [query.variable, Validators.required]
|
|
86
|
+
});
|
|
87
|
+
this.queriesFormArray.push(queryGroup);
|
|
88
|
+
}
|
|
89
|
+
deleteQueryById(queryId) {
|
|
90
|
+
const index = this.queries.findIndex(q => q.id === queryId);
|
|
91
|
+
if (index !== -1) {
|
|
92
|
+
this.queries.splice(index, 1);
|
|
93
|
+
this.queriesFormArray.removeAt(index);
|
|
94
|
+
if (this.selectedQueryIndex >= this.queries.length) {
|
|
95
|
+
this.selectedQueryIndex = Math.max(0, this.queries.length - 1);
|
|
96
|
+
}
|
|
97
|
+
this.deleteQuery.emit(queryId);
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
getDbEnvironmentConfig() {
|
|
101
|
+
return {
|
|
102
|
+
key: 'dbEnvironment',
|
|
103
|
+
placeholder: 'Select DB environment',
|
|
104
|
+
multiple: false,
|
|
105
|
+
searchable: false,
|
|
106
|
+
options: this.dbEnvironmentOptions
|
|
107
|
+
};
|
|
108
|
+
}
|
|
109
|
+
getCurrentQuery() {
|
|
110
|
+
if (this.queries.length > 0 && this.selectedQueryIndex >= 0 && this.selectedQueryIndex < this.queries.length) {
|
|
111
|
+
return this.queries[this.selectedQueryIndex];
|
|
112
|
+
}
|
|
113
|
+
return null;
|
|
114
|
+
}
|
|
115
|
+
getCurrentQueryFormGroup() {
|
|
116
|
+
if (this.queriesFormArray && this.selectedQueryIndex >= 0 && this.selectedQueryIndex < this.queriesFormArray.length) {
|
|
117
|
+
return this.queriesFormArray.at(this.selectedQueryIndex);
|
|
118
|
+
}
|
|
119
|
+
return null;
|
|
120
|
+
}
|
|
121
|
+
onRunQuery() {
|
|
122
|
+
const dbEnvironment = this.databaseForm.get('dbEnvironment')?.value;
|
|
123
|
+
const currentQuery = this.getCurrentQuery();
|
|
124
|
+
const queryFormGroup = this.getCurrentQueryFormGroup();
|
|
125
|
+
if (currentQuery && queryFormGroup && dbEnvironment) {
|
|
126
|
+
const query = queryFormGroup.get('query')?.value || '';
|
|
127
|
+
const variable = queryFormGroup.get('variable')?.value || '';
|
|
128
|
+
// Update the query object
|
|
129
|
+
currentQuery.query = query;
|
|
130
|
+
currentQuery.variable = variable;
|
|
131
|
+
this.runQuery.emit({
|
|
132
|
+
query,
|
|
133
|
+
variable,
|
|
134
|
+
dbEnvironment
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
onSelectQuery(index) {
|
|
139
|
+
this.selectedQueryIndex = index;
|
|
140
|
+
}
|
|
141
|
+
onTabChange(tab) {
|
|
142
|
+
this.selectedTab = tab;
|
|
143
|
+
}
|
|
144
|
+
onCopyResults() {
|
|
145
|
+
// Copy query results to clipboard
|
|
146
|
+
if (this.queryResults && this.queryResults.length > 0) {
|
|
147
|
+
const text = JSON.stringify(this.queryResults, null, 2);
|
|
148
|
+
navigator.clipboard.writeText(text).then(() => {
|
|
149
|
+
// Could emit an event or show a toast notification
|
|
150
|
+
});
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
onCancel() {
|
|
154
|
+
this.cancelled.emit();
|
|
155
|
+
}
|
|
156
|
+
onCreateStep() {
|
|
157
|
+
if (this.databaseForm.valid) {
|
|
158
|
+
const formValue = this.databaseForm.value;
|
|
159
|
+
const stepData = {
|
|
160
|
+
dbEnvironment: formValue.dbEnvironment,
|
|
161
|
+
queries: this.queries.map((query, index) => {
|
|
162
|
+
const queryFormGroup = this.queriesFormArray.at(index);
|
|
163
|
+
return {
|
|
164
|
+
id: query.id,
|
|
165
|
+
query: queryFormGroup.get('query')?.value || '',
|
|
166
|
+
variable: queryFormGroup.get('variable')?.value || '',
|
|
167
|
+
status: query.status
|
|
168
|
+
};
|
|
169
|
+
})
|
|
170
|
+
};
|
|
171
|
+
this.createStep.emit(stepData);
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
getTableColumns() {
|
|
175
|
+
if (!this.queryResults || this.queryResults.length === 0) {
|
|
176
|
+
return [];
|
|
177
|
+
}
|
|
178
|
+
return Object.keys(this.queryResults[0]);
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
StepBuilderDatabaseComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepBuilderDatabaseComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component });
|
|
182
|
+
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-start cqa-gap-4 cqa-mb-6\">\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\">\n DB Environment <span class=\"cqa-text-red-500\">*</span>\n </label>\n <cqa-dynamic-select\n [form]=\"databaseForm\"\n [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\n variant=\"filled\"\n text=\"Run Query\"\n [customClass]=\"'cqa-whitespace-nowrap'\"\n (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 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 <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\"\n [class.cqa-border]=\"selectedQueryIndex === i\"\n [class.cqa-border-blue-500]=\"selectedQueryIndex === i\"\n (click)=\"onSelectQuery(i)\">\n <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n <cqa-badge\n *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-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden\">\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\">Query & Store Response</h3>\n </div>\n \n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-4 cqa-flex cqa-flex-col cqa-gap-4\">\n <!-- SQL Query Textarea -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col\">\n <cqa-custom-textarea\n [placeholder]=\"'Enter your SQL query here...'\"\n [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\"\n [fullWidth]=\"true\"\n [rows]=\"8\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\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\">\n Variable\n </label>\n <cqa-custom-input\n [placeholder]=\"'Variable name'\"\n [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\"\n [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\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <cqa-button\n variant=\"outlined\"\n text=\"Add Query\"\n icon=\"add\"\n (clicked)=\"addNewQuery()\">\n </cqa-button>\n <button\n 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 || '')\"\n [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\n variant=\"filled\"\n text=\"Run\"\n icon=\"play_arrow\"\n (clicked)=\"onRunQuery()\"\n [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\">\n </cqa-button>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Results Section -->\n <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden cqa-mb-6\">\n <!-- Tabs -->\n <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n <button\n type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'output'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'output'\"\n (click)=\"onTabChange('output')\">\n Output\n </button>\n <button\n type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'verification'\"\n (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\n variant=\"text\"\n text=\"Copy\"\n icon=\"content_copy\"\n [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\" \n class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n <td *ngFor=\"let key of getTableColumns()\" \n 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\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\n class=\"cqa-w-1/2\"\n variant=\"outlined\"\n text=\"Cancel\"\n [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button\n class=\"cqa-w-1/2\"\n variant=\"filled\"\n text=\"Create Step\"\n [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>\n\n", components: [{ type: i2.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore"] }, { type: i3.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"] }, { type: i7.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }], directives: [{ type: i8.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i8.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
|
|
183
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: StepBuilderDatabaseComponent, decorators: [{
|
|
184
|
+
type: Component,
|
|
185
|
+
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-start cqa-gap-4 cqa-mb-6\">\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\">\n DB Environment <span class=\"cqa-text-red-500\">*</span>\n </label>\n <cqa-dynamic-select\n [form]=\"databaseForm\"\n [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\n variant=\"filled\"\n text=\"Run Query\"\n [customClass]=\"'cqa-whitespace-nowrap'\"\n (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 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 <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\"\n [class.cqa-border]=\"selectedQueryIndex === i\"\n [class.cqa-border-blue-500]=\"selectedQueryIndex === i\"\n (click)=\"onSelectQuery(i)\">\n <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n <cqa-badge\n *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-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden\">\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\">Query & Store Response</h3>\n </div>\n \n <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-4 cqa-flex cqa-flex-col cqa-gap-4\">\n <!-- SQL Query Textarea -->\n <div class=\"cqa-flex-1 cqa-flex cqa-flex-col\">\n <cqa-custom-textarea\n [placeholder]=\"'Enter your SQL query here...'\"\n [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\"\n [fullWidth]=\"true\"\n [rows]=\"8\"\n (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n </cqa-custom-textarea>\n </div>\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\">\n Variable\n </label>\n <cqa-custom-input\n [placeholder]=\"'Variable name'\"\n [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\"\n [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\n <!-- Action Buttons -->\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <cqa-button\n variant=\"outlined\"\n text=\"Add Query\"\n icon=\"add\"\n (clicked)=\"addNewQuery()\">\n </cqa-button>\n <button\n 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 || '')\"\n [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\n variant=\"filled\"\n text=\"Run\"\n icon=\"play_arrow\"\n (clicked)=\"onRunQuery()\"\n [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\">\n </cqa-button>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Results Section -->\n <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden cqa-mb-6\">\n <!-- Tabs -->\n <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n <button\n type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'output'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'output'\"\n (click)=\"onTabChange('output')\">\n Output\n </button>\n <button\n type=\"button\"\n class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n [class.cqa-border-transparent]=\"selectedTab !== 'verification'\"\n (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\n variant=\"text\"\n text=\"Copy\"\n icon=\"content_copy\"\n [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\" \n class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n <td *ngFor=\"let key of getTableColumns()\" \n 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\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\n class=\"cqa-w-1/2\"\n variant=\"outlined\"\n text=\"Cancel\"\n [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCancel()\">\n </cqa-button>\n <cqa-button\n class=\"cqa-w-1/2\"\n variant=\"filled\"\n text=\"Create Step\"\n [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n (clicked)=\"onCreateStep()\">\n </cqa-button>\n </div>\n</div>\n\n", styles: [] }]
|
|
186
|
+
}], ctorParameters: function () { return [{ type: i1.FormBuilder }]; }, propDecorators: { dbEnvironmentOptions: [{
|
|
187
|
+
type: Input
|
|
188
|
+
}], queries: [{
|
|
189
|
+
type: Input
|
|
190
|
+
}], queryResults: [{
|
|
191
|
+
type: Input
|
|
192
|
+
}], isLoading: [{
|
|
193
|
+
type: Input
|
|
194
|
+
}], createStep: [{
|
|
195
|
+
type: Output
|
|
196
|
+
}], cancelled: [{
|
|
197
|
+
type: Output
|
|
198
|
+
}], runQuery: [{
|
|
199
|
+
type: Output
|
|
200
|
+
}], addQuery: [{
|
|
201
|
+
type: Output
|
|
202
|
+
}], deleteQuery: [{
|
|
203
|
+
type: Output
|
|
204
|
+
}] } });
|
|
205
|
+
//# 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  /** 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-start cqa-gap-4 cqa-mb-6\">\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\">\n        DB Environment <span class=\"cqa-text-red-500\">*</span>\n      </label>\n      <cqa-dynamic-select\n        [form]=\"databaseForm\"\n        [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\n        variant=\"filled\"\n        text=\"Run Query\"\n        [customClass]=\"'cqa-whitespace-nowrap'\"\n        (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 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      <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\"\n             [class.cqa-border]=\"selectedQueryIndex === i\"\n             [class.cqa-border-blue-500]=\"selectedQueryIndex === i\"\n             (click)=\"onSelectQuery(i)\">\n          <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">Query {{ i + 1 }}</span>\n          <cqa-badge\n            *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-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden\">\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\">Query & Store Response</h3>\n      </div>\n      \n      <div class=\"cqa-flex-1 cqa-overflow-y-auto cqa-p-4 cqa-flex cqa-flex-col cqa-gap-4\">\n        <!-- SQL Query Textarea -->\n        <div class=\"cqa-flex-1 cqa-flex cqa-flex-col\">\n          <cqa-custom-textarea\n            [placeholder]=\"'Enter your SQL query here...'\"\n            [value]=\"getCurrentQueryFormGroup()?.get('query')?.value || ''\"\n            [fullWidth]=\"true\"\n            [rows]=\"8\"\n            (valueChange)=\"getCurrentQueryFormGroup()?.get('query')?.setValue($event)\">\n          </cqa-custom-textarea>\n        </div>\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\">\n              Variable\n            </label>\n            <cqa-custom-input\n              [placeholder]=\"'Variable name'\"\n              [value]=\"getCurrentQueryFormGroup()?.get('variable')?.value || ''\"\n              [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\n        <!-- Action Buttons -->\n        <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n          <cqa-button\n            variant=\"outlined\"\n            text=\"Add Query\"\n            icon=\"add\"\n            (clicked)=\"addNewQuery()\">\n          </cqa-button>\n          <button\n            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 || '')\"\n            [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\n            variant=\"filled\"\n            text=\"Run\"\n            icon=\"play_arrow\"\n            (clicked)=\"onRunQuery()\"\n            [disabled]=\"!databaseForm.get('dbEnvironment')?.value || isLoading\">\n          </cqa-button>\n        </div>\n      </div>\n    </div>\n  </div>\n\n  <!-- Results Section -->\n  <div class=\"cqa-flex cqa-flex-col cqa-border cqa-border-gray-200 cqa-rounded-lg cqa-overflow-hidden cqa-mb-6\">\n    <!-- Tabs -->\n    <div class=\"cqa-flex cqa-items-center cqa-border-b cqa-border-gray-200 cqa-bg-gray-50\">\n      <button\n        type=\"button\"\n        class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n        [class.cqa-text-blue-600]=\"selectedTab === 'output'\"\n        [class.cqa-border-blue-600]=\"selectedTab === 'output'\"\n        [class.cqa-text-gray-600]=\"selectedTab !== 'output'\"\n        [class.cqa-border-transparent]=\"selectedTab !== 'output'\"\n        (click)=\"onTabChange('output')\">\n        Output\n      </button>\n      <button\n        type=\"button\"\n        class=\"cqa-px-4 cqa-py-2 cqa-text-sm cqa-font-medium cqa-transition-colors cqa-border-b-2\"\n        [class.cqa-text-blue-600]=\"selectedTab === 'verification'\"\n        [class.cqa-border-blue-600]=\"selectedTab === 'verification'\"\n        [class.cqa-text-gray-600]=\"selectedTab !== 'verification'\"\n        [class.cqa-border-transparent]=\"selectedTab !== 'verification'\"\n        (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\n            variant=\"text\"\n            text=\"Copy\"\n            icon=\"content_copy\"\n            [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\" \n                  class=\"cqa-border-b cqa-border-gray-200 hover:cqa-bg-gray-50\">\n                <td *ngFor=\"let key of getTableColumns()\" \n                    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\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\n      class=\"cqa-w-1/2\"\n      variant=\"outlined\"\n      text=\"Cancel\"\n      [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCancel()\">\n    </cqa-button>\n    <cqa-button\n      class=\"cqa-w-1/2\"\n      variant=\"filled\"\n      text=\"Create Step\"\n      [customClass]=\"'cqa-flex-1 cqa-w-full'\"\n      (clicked)=\"onCreateStep()\">\n    </cqa-button>\n  </div>\n</div>\n\n"]}
|