@cqa-lib/cqa-ui 1.1.175 → 1.1.176
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/dynamic-select/dynamic-select-field.component.mjs +3 -3
- 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/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 +814 -21
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +802 -21
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/dynamic-select/dynamic-select-field.component.d.ts +1 -0
- 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/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"]}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import { InjectionToken } from '@angular/core';
|
|
2
|
+
/** Sentinel returned from afterClosed() when user clicked "Edit in depth". */
|
|
3
|
+
export const CUSTOM_EDIT_STEP_EDIT_IN_DEPTH = Symbol('CustomEditStepEditInDepth');
|
|
4
|
+
export const CUSTOM_EDIT_STEP_DATA = new InjectionToken('CUSTOM_EDIT_STEP_DATA');
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY3VzdG9tLWVkaXQtc3RlcC1kYXRhLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi90ZXN0LWNhc2UtZGV0YWlscy9jdXN0b20tZWRpdC1zdGVwL2N1c3RvbS1lZGl0LXN0ZXAtZGF0YS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBTy9DLDhFQUE4RTtBQUM5RSxNQUFNLENBQUMsTUFBTSw4QkFBOEIsR0FBRyxNQUFNLENBQUMsMkJBQTJCLENBQUMsQ0FBQztBQUVsRixNQUFNLENBQUMsTUFBTSxxQkFBcUIsR0FBRyxJQUFJLGNBQWMsQ0FDckQsdUJBQXVCLENBQ3hCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3Rpb25Ub2tlbiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEN1c3RvbUVkaXRTdGVwRGF0YSB7XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmc7XG4gIGhlbHBVcmw/OiBzdHJpbmc7XG59XG5cbi8qKiBTZW50aW5lbCByZXR1cm5lZCBmcm9tIGFmdGVyQ2xvc2VkKCkgd2hlbiB1c2VyIGNsaWNrZWQgXCJFZGl0IGluIGRlcHRoXCIuICovXG5leHBvcnQgY29uc3QgQ1VTVE9NX0VESVRfU1RFUF9FRElUX0lOX0RFUFRIID0gU3ltYm9sKCdDdXN0b21FZGl0U3RlcEVkaXRJbkRlcHRoJyk7XG5cbmV4cG9ydCBjb25zdCBDVVNUT01fRURJVF9TVEVQX0RBVEEgPSBuZXcgSW5qZWN0aW9uVG9rZW48Q3VzdG9tRWRpdFN0ZXBEYXRhPihcbiAgJ0NVU1RPTV9FRElUX1NURVBfREFUQSdcbik7XG4iXX0=
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { InjectionToken } from '@angular/core';
|
|
2
|
+
import { Subject } from 'rxjs';
|
|
3
|
+
export class CustomEditStepRef {
|
|
4
|
+
constructor(overlayRef) {
|
|
5
|
+
this.overlayRef = overlayRef;
|
|
6
|
+
this.closed$ = new Subject();
|
|
7
|
+
this.isClosed = false;
|
|
8
|
+
this.overlayRef.detachments().subscribe(() => {
|
|
9
|
+
this.finishClose(undefined);
|
|
10
|
+
});
|
|
11
|
+
}
|
|
12
|
+
close(result) {
|
|
13
|
+
if (this.isClosed) {
|
|
14
|
+
return;
|
|
15
|
+
}
|
|
16
|
+
this.finishClose(result);
|
|
17
|
+
this.overlayRef.dispose();
|
|
18
|
+
}
|
|
19
|
+
afterClosed() {
|
|
20
|
+
return this.closed$.asObservable();
|
|
21
|
+
}
|
|
22
|
+
finishClose(result) {
|
|
23
|
+
if (this.isClosed) {
|
|
24
|
+
return;
|
|
25
|
+
}
|
|
26
|
+
this.isClosed = true;
|
|
27
|
+
this.closed$.next(result);
|
|
28
|
+
this.closed$.complete();
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
export const CUSTOM_EDIT_STEP_REF = new InjectionToken('CUSTOM_EDIT_STEP_REF');
|
|
32
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY3VzdG9tLWVkaXQtc3RlcC1yZWYuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3Rlc3QtY2FzZS1kZXRhaWxzL2N1c3RvbS1lZGl0LXN0ZXAvY3VzdG9tLWVkaXQtc3RlcC1yZWYudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUUvQyxPQUFPLEVBQWMsT0FBTyxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBRTNDLE1BQU0sT0FBTyxpQkFBaUI7SUFJNUIsWUFBNkIsVUFBc0I7UUFBdEIsZUFBVSxHQUFWLFVBQVUsQ0FBWTtRQUhsQyxZQUFPLEdBQUcsSUFBSSxPQUFPLEVBQXVCLENBQUM7UUFDdEQsYUFBUSxHQUFHLEtBQUssQ0FBQztRQUd2QixJQUFJLENBQUMsVUFBVSxDQUFDLFdBQVcsRUFBRSxDQUFDLFNBQVMsQ0FBQyxHQUFHLEVBQUU7WUFDM0MsSUFBSSxDQUFDLFdBQVcsQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUM5QixDQUFDLENBQUMsQ0FBQztJQUNMLENBQUM7SUFFRCxLQUFLLENBQUMsTUFBZ0I7UUFDcEIsSUFBSSxJQUFJLENBQUMsUUFBUSxFQUFFO1lBQ2pCLE9BQU87U0FDUjtRQUNELElBQUksQ0FBQyxXQUFXLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDekIsSUFBSSxDQUFDLFVBQVUsQ0FBQyxPQUFPLEVBQUUsQ0FBQztJQUM1QixDQUFDO0lBRUQsV0FBVztRQUNULE9BQU8sSUFBSSxDQUFDLE9BQU8sQ0FBQyxZQUFZLEVBQUUsQ0FBQztJQUNyQyxDQUFDO0lBRU8sV0FBVyxDQUFDLE1BQTJCO1FBQzdDLElBQUksSUFBSSxDQUFDLFFBQVEsRUFBRTtZQUNqQixPQUFPO1NBQ1I7UUFDRCxJQUFJLENBQUMsUUFBUSxHQUFHLElBQUksQ0FBQztRQUNyQixJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUMxQixJQUFJLENBQUMsT0FBTyxDQUFDLFFBQVEsRUFBRSxDQUFDO0lBQzFCLENBQUM7Q0FDRjtBQUVELE1BQU0sQ0FBQyxNQUFNLG9CQUFvQixHQUFHLElBQUksY0FBYyxDQUNwRCxzQkFBc0IsQ0FDdkIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGlvblRva2VuIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBPdmVybGF5UmVmIH0gZnJvbSAnQGFuZ3VsYXIvY2RrL292ZXJsYXknO1xuaW1wb3J0IHsgT2JzZXJ2YWJsZSwgU3ViamVjdCB9IGZyb20gJ3J4anMnO1xuXG5leHBvcnQgY2xhc3MgQ3VzdG9tRWRpdFN0ZXBSZWY8VFJlc3VsdCA9IHN0cmluZyB8IHVuZGVmaW5lZD4ge1xuICBwcml2YXRlIHJlYWRvbmx5IGNsb3NlZCQgPSBuZXcgU3ViamVjdDxUUmVzdWx0IHwgdW5kZWZpbmVkPigpO1xuICBwcml2YXRlIGlzQ2xvc2VkID0gZmFsc2U7XG5cbiAgY29uc3RydWN0b3IocHJpdmF0ZSByZWFkb25seSBvdmVybGF5UmVmOiBPdmVybGF5UmVmKSB7XG4gICAgdGhpcy5vdmVybGF5UmVmLmRldGFjaG1lbnRzKCkuc3Vic2NyaWJlKCgpID0+IHtcbiAgICAgIHRoaXMuZmluaXNoQ2xvc2UodW5kZWZpbmVkKTtcbiAgICB9KTtcbiAgfVxuXG4gIGNsb3NlKHJlc3VsdD86IFRSZXN1bHQpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5pc0Nsb3NlZCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB0aGlzLmZpbmlzaENsb3NlKHJlc3VsdCk7XG4gICAgdGhpcy5vdmVybGF5UmVmLmRpc3Bvc2UoKTtcbiAgfVxuXG4gIGFmdGVyQ2xvc2VkKCk6IE9ic2VydmFibGU8VFJlc3VsdCB8IHVuZGVmaW5lZD4ge1xuICAgIHJldHVybiB0aGlzLmNsb3NlZCQuYXNPYnNlcnZhYmxlKCk7XG4gIH1cblxuICBwcml2YXRlIGZpbmlzaENsb3NlKHJlc3VsdDogVFJlc3VsdCB8IHVuZGVmaW5lZCk6IHZvaWQge1xuICAgIGlmICh0aGlzLmlzQ2xvc2VkKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIHRoaXMuaXNDbG9zZWQgPSB0cnVlO1xuICAgIHRoaXMuY2xvc2VkJC5uZXh0KHJlc3VsdCk7XG4gICAgdGhpcy5jbG9zZWQkLmNvbXBsZXRlKCk7XG4gIH1cbn1cblxuZXhwb3J0IGNvbnN0IENVU1RPTV9FRElUX1NURVBfUkVGID0gbmV3IEluamVjdGlvblRva2VuPEN1c3RvbUVkaXRTdGVwUmVmPihcbiAgJ0NVU1RPTV9FRElUX1NURVBfUkVGJ1xuKTtcbiJdfQ==
|
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Inject, Optional, Output, } from '@angular/core';
|
|
2
|
+
import { CUSTOM_EDIT_STEP_DATA, CUSTOM_EDIT_STEP_EDIT_IN_DEPTH, } from './custom-edit-step-data';
|
|
3
|
+
import { CUSTOM_EDIT_STEP_REF } from './../custom-edit-step/custom-edit-step-ref';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@angular/material/icon";
|
|
6
|
+
import * as i2 from "../../custom-textarea/custom-textarea.component";
|
|
7
|
+
import * as i3 from "../../button/button.component";
|
|
8
|
+
import * as i4 from "@angular/common";
|
|
9
|
+
import * as i5 from "./../custom-edit-step/custom-edit-step-ref";
|
|
10
|
+
export class CustomEditStepComponent {
|
|
11
|
+
constructor(ref, data) {
|
|
12
|
+
this.ref = ref;
|
|
13
|
+
this.apply = new EventEmitter();
|
|
14
|
+
this.cancel = new EventEmitter();
|
|
15
|
+
this.editInDepth = new EventEmitter();
|
|
16
|
+
/** Local value while editing (for two-way binding in textarea) */
|
|
17
|
+
this.value = '';
|
|
18
|
+
this.helpUrl = '';
|
|
19
|
+
/** Tooltip shown when hovering over the "Need help ?" icon and text */
|
|
20
|
+
this.helpTooltipText = 'Not sure what to do? Click to go to our detailed step creation documentation';
|
|
21
|
+
/** Whether the help tooltip is visible (custom tooltip for use inside overlay) */
|
|
22
|
+
this.showHelpTooltip = false;
|
|
23
|
+
if (data) {
|
|
24
|
+
this.value = data.description ?? '';
|
|
25
|
+
this.helpUrl = data.helpUrl ?? '';
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
onApply() {
|
|
29
|
+
this.apply.emit(this.value);
|
|
30
|
+
this.ref.close(this.value);
|
|
31
|
+
}
|
|
32
|
+
onCancel() {
|
|
33
|
+
this.cancel.emit();
|
|
34
|
+
this.ref.close(undefined);
|
|
35
|
+
}
|
|
36
|
+
onClose() {
|
|
37
|
+
this.onCancel();
|
|
38
|
+
}
|
|
39
|
+
onEditInDepth(event) {
|
|
40
|
+
event.preventDefault();
|
|
41
|
+
this.editInDepth.emit();
|
|
42
|
+
this.ref.close(CUSTOM_EDIT_STEP_EDIT_IN_DEPTH);
|
|
43
|
+
}
|
|
44
|
+
onHelp(event) {
|
|
45
|
+
if (this.helpUrl) {
|
|
46
|
+
event.preventDefault();
|
|
47
|
+
window.open(this.helpUrl, '_blank');
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
CustomEditStepComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: CustomEditStepComponent, deps: [{ token: CUSTOM_EDIT_STEP_REF }, { token: CUSTOM_EDIT_STEP_DATA, optional: true }], target: i0.ɵɵFactoryTarget.Component });
|
|
52
|
+
CustomEditStepComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: CustomEditStepComponent, selector: "cqa-custom-edit-step", outputs: { apply: "apply", cancel: "cancel", editInDepth: "editInDepth" }, host: { classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div\n class=\"cqa-bg-white cqa-rounded-[12px] cqa-shadow-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-w-[500px] cqa-flex cqa-flex-col cqa-gap-[12px] cqa-p-2 cqa-box-border\">\n <!-- Header: title left; Need help? + close icon right -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-px-4\">\n <h2 class=\"cqa-text-[16px] cqa-leading-[24px] cqa-font-bold cqa-text-[#111827] cqa-m-0 cqa-font-[600]\">\n Step Description\n </h2>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <!-- Need help? with custom tooltip (works inside overlay) -->\n <div class=\"cqa-relative cqa-inline-flex\"\n (mouseenter)=\"showHelpTooltip = true\" (mouseleave)=\"showHelpTooltip = false\">\n <a *ngIf=\"helpUrl\" href=\"#\" (click)=\"onHelp($event)\"\n class=\"cqa-flex cqa-items-center cqa-gap-1 cqa-text-[#3F43EE] cqa-text-[12px] cqa-leading-[21px] cqa-no-underline hover:cqa-underline cqa-cursor-pointer\">\n <svg width=\"17\" height=\"16\" viewBox=\"0 0 17 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" class=\"cqa-flex-shrink-0\" aria-hidden=\"true\">\n <g clip-path=\"url(#help-icon-clip)\">\n <path d=\"M8.50033 14.6663C12.4123 14.6663 15.5837 11.6816 15.5837 7.99967C15.5837 4.31778 12.4123 1.33301 8.50033 1.33301C4.58831 1.33301 1.41699 4.31778 1.41699 7.99967C1.41699 11.6816 4.58831 14.6663 8.50033 14.6663Z\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M6.43848 6.00038C6.60501 5.55483 6.93371 5.17912 7.36636 4.9398C7.79901 4.70049 8.30769 4.61301 8.80231 4.69285C9.29693 4.7727 9.74556 5.01473 10.0687 5.37607C10.3919 5.7374 10.5688 6.19473 10.5681 6.66705C10.5681 8.00038 8.44306 8.66705 8.44306 8.66705\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M8.5 11.333H8.50966\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </g>\n <defs>\n <clipPath id=\"help-icon-clip\">\n <rect width=\"17\" height=\"16\" fill=\"white\"/>\n </clipPath>\n </defs>\n </svg>\n Need help ?\n </a>\n <span *ngIf=\"!helpUrl\" class=\"cqa-flex cqa-items-center cqa-gap-1.5 cqa-font-[500] cqa-text-[10px] cqa-cursor-default\">\n <svg width=\"17\" height=\"16\" viewBox=\"0 0 17 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" class=\"cqa-flex-shrink-0 cqa-text-[#3F43EE]\" aria-hidden=\"true\">\n <g clip-path=\"url(#help-icon-clip-nolink)\">\n <path d=\"M8.50033 14.6663C12.4123 14.6663 15.5837 11.6816 15.5837 7.99967C15.5837 4.31778 12.4123 1.33301 8.50033 1.33301C4.58831 1.33301 1.41699 4.31778 1.41699 7.99967C1.41699 11.6816 4.58831 14.6663 8.50033 14.6663Z\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M6.43848 6.00038C6.60501 5.55483 6.93371 5.17912 7.36636 4.9398C7.79901 4.70049 8.30769 4.61301 8.80231 4.69285C9.29693 4.7727 9.74556 5.01473 10.0687 5.37607C10.3919 5.7374 10.5688 6.19473 10.5681 6.66705C10.5681 8.00038 8.44306 8.66705 8.44306 8.66705\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M8.5 11.333H8.50966\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </g>\n <defs>\n <clipPath id=\"help-icon-clip-nolink\">\n <rect width=\"17\" height=\"16\" fill=\"white\"/>\n </clipPath>\n </defs>\n </svg>\n Need help ?\n </span>\n <!-- Custom tooltip (exact spec: 306\u00D720 content, 6px radius, #0A0A0A, no arrow) -->\n <div *ngIf=\"showHelpTooltip\" class=\"cqa-absolute cqa-pointer-events-none cqa-z-[100]\"\n style=\"top: -24px; left: -125px;\"\n role=\"tooltip\">\n <div class=\"cqa-text-white cqa-text-center cqa-whitespace-nowrap\"\n style=\"width: 306px; min-height: 20px; border-radius: 6px; opacity: 1; padding: 4px 8px; background-color: #0A0A0A; line-height: 20px; font-size: 8px;\">\n {{ helpTooltipText }}\n </div>\n </div>\n </div>\n <button type=\"button\" (click)=\"onClose()\"\n class=\"cqa-p-1 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center\"\n title=\"Close\" aria-label=\"Close\">\n <mat-icon class=\"!cqa-w-5 !cqa-h-5 !cqa-text-[20px]\">close</mat-icon>\n </button>\n </div>\n </div>\n\n <!-- Line below header (full width of modal, no side margin) -->\n <div class=\"cqa--mx-2 cqa-w-[calc(100%+1rem)] cqa-flex-shrink-0\">\n <div class=\"cqa-h-px cqa-w-full cqa-bg-[#E5E7EB]\" role=\"presentation\"></div>\n </div>\n\n <!-- Content: textarea + helper -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n <cqa-custom-textarea\n [(value)]=\"value\"\n placeholder=\"Enter description\"\n [fullWidth]=\"true\"\n [rows]=\"4\"\n resize=\"vertical\"\n size=\"md\">\n </cqa-custom-textarea>\n <p class=\"cqa-text-[12px] cqa-leading-[18px] cqa-text-[#0A0A0A] cqa-m-0\">\n Provide a clear description of the step's purpose\n </p>\n </div>\n\n <!-- Line below content (full width of modal, no side margin) -->\n <div class=\"cqa--mx-2 cqa-w-[calc(100%+1rem)] cqa-flex-shrink-0\">\n <div class=\"cqa-h-px cqa-w-full cqa-bg-[#E5E7EB]\" role=\"presentation\"></div>\n </div>\n\n <!-- Footer: Cancel, Apply (full width in one row) -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <div class=\"cqa-flex cqa-items-stretch cqa-gap-2 cqa-w-full\">\n <div class=\"cqa-flex-1 cqa-min-w-0\">\n <cqa-button variant=\"outlined\" btnSize=\"lg\" [text]=\"'Cancel'\" [fullWidth]=\"true\" (clicked)=\"onCancel()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"></cqa-button>\n </div>\n <div class=\"cqa-flex-1 cqa-min-w-0\">\n <cqa-button variant=\"filled\" btnSize=\"lg\" [text]=\"'Apply'\" [fullWidth]=\"true\" (clicked)=\"onApply()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#3F43EE]'\"></cqa-button>\n </div>\n </div>\n <a href=\"#\" (click)=\"onEditInDepth($event)\"\n class=\"cqa-text-[#3F43EE] cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-1.5 cqa-no-underline hover:cqa-no-underline cqa-self-center\">\n <mat-icon class=\"!cqa-w-4 !cqa-h-4 !cqa-text-[16px]\">open_in_new</mat-icon>\n Edit in depth (open detailed right panel)\n </a>\n </div>\n</div>\n", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i2.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: i3.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
53
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: CustomEditStepComponent, decorators: [{
|
|
54
|
+
type: Component,
|
|
55
|
+
args: [{ selector: 'cqa-custom-edit-step', host: { class: 'cqa-ui-root' }, changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n class=\"cqa-bg-white cqa-rounded-[12px] cqa-shadow-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-w-[500px] cqa-flex cqa-flex-col cqa-gap-[12px] cqa-p-2 cqa-box-border\">\n <!-- Header: title left; Need help? + close icon right -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-px-4\">\n <h2 class=\"cqa-text-[16px] cqa-leading-[24px] cqa-font-bold cqa-text-[#111827] cqa-m-0 cqa-font-[600]\">\n Step Description\n </h2>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <!-- Need help? with custom tooltip (works inside overlay) -->\n <div class=\"cqa-relative cqa-inline-flex\"\n (mouseenter)=\"showHelpTooltip = true\" (mouseleave)=\"showHelpTooltip = false\">\n <a *ngIf=\"helpUrl\" href=\"#\" (click)=\"onHelp($event)\"\n class=\"cqa-flex cqa-items-center cqa-gap-1 cqa-text-[#3F43EE] cqa-text-[12px] cqa-leading-[21px] cqa-no-underline hover:cqa-underline cqa-cursor-pointer\">\n <svg width=\"17\" height=\"16\" viewBox=\"0 0 17 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" class=\"cqa-flex-shrink-0\" aria-hidden=\"true\">\n <g clip-path=\"url(#help-icon-clip)\">\n <path d=\"M8.50033 14.6663C12.4123 14.6663 15.5837 11.6816 15.5837 7.99967C15.5837 4.31778 12.4123 1.33301 8.50033 1.33301C4.58831 1.33301 1.41699 4.31778 1.41699 7.99967C1.41699 11.6816 4.58831 14.6663 8.50033 14.6663Z\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M6.43848 6.00038C6.60501 5.55483 6.93371 5.17912 7.36636 4.9398C7.79901 4.70049 8.30769 4.61301 8.80231 4.69285C9.29693 4.7727 9.74556 5.01473 10.0687 5.37607C10.3919 5.7374 10.5688 6.19473 10.5681 6.66705C10.5681 8.00038 8.44306 8.66705 8.44306 8.66705\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M8.5 11.333H8.50966\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </g>\n <defs>\n <clipPath id=\"help-icon-clip\">\n <rect width=\"17\" height=\"16\" fill=\"white\"/>\n </clipPath>\n </defs>\n </svg>\n Need help ?\n </a>\n <span *ngIf=\"!helpUrl\" class=\"cqa-flex cqa-items-center cqa-gap-1.5 cqa-font-[500] cqa-text-[10px] cqa-cursor-default\">\n <svg width=\"17\" height=\"16\" viewBox=\"0 0 17 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" class=\"cqa-flex-shrink-0 cqa-text-[#3F43EE]\" aria-hidden=\"true\">\n <g clip-path=\"url(#help-icon-clip-nolink)\">\n <path d=\"M8.50033 14.6663C12.4123 14.6663 15.5837 11.6816 15.5837 7.99967C15.5837 4.31778 12.4123 1.33301 8.50033 1.33301C4.58831 1.33301 1.41699 4.31778 1.41699 7.99967C1.41699 11.6816 4.58831 14.6663 8.50033 14.6663Z\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M6.43848 6.00038C6.60501 5.55483 6.93371 5.17912 7.36636 4.9398C7.79901 4.70049 8.30769 4.61301 8.80231 4.69285C9.29693 4.7727 9.74556 5.01473 10.0687 5.37607C10.3919 5.7374 10.5688 6.19473 10.5681 6.66705C10.5681 8.00038 8.44306 8.66705 8.44306 8.66705\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M8.5 11.333H8.50966\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </g>\n <defs>\n <clipPath id=\"help-icon-clip-nolink\">\n <rect width=\"17\" height=\"16\" fill=\"white\"/>\n </clipPath>\n </defs>\n </svg>\n Need help ?\n </span>\n <!-- Custom tooltip (exact spec: 306\u00D720 content, 6px radius, #0A0A0A, no arrow) -->\n <div *ngIf=\"showHelpTooltip\" class=\"cqa-absolute cqa-pointer-events-none cqa-z-[100]\"\n style=\"top: -24px; left: -125px;\"\n role=\"tooltip\">\n <div class=\"cqa-text-white cqa-text-center cqa-whitespace-nowrap\"\n style=\"width: 306px; min-height: 20px; border-radius: 6px; opacity: 1; padding: 4px 8px; background-color: #0A0A0A; line-height: 20px; font-size: 8px;\">\n {{ helpTooltipText }}\n </div>\n </div>\n </div>\n <button type=\"button\" (click)=\"onClose()\"\n class=\"cqa-p-1 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center\"\n title=\"Close\" aria-label=\"Close\">\n <mat-icon class=\"!cqa-w-5 !cqa-h-5 !cqa-text-[20px]\">close</mat-icon>\n </button>\n </div>\n </div>\n\n <!-- Line below header (full width of modal, no side margin) -->\n <div class=\"cqa--mx-2 cqa-w-[calc(100%+1rem)] cqa-flex-shrink-0\">\n <div class=\"cqa-h-px cqa-w-full cqa-bg-[#E5E7EB]\" role=\"presentation\"></div>\n </div>\n\n <!-- Content: textarea + helper -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n <cqa-custom-textarea\n [(value)]=\"value\"\n placeholder=\"Enter description\"\n [fullWidth]=\"true\"\n [rows]=\"4\"\n resize=\"vertical\"\n size=\"md\">\n </cqa-custom-textarea>\n <p class=\"cqa-text-[12px] cqa-leading-[18px] cqa-text-[#0A0A0A] cqa-m-0\">\n Provide a clear description of the step's purpose\n </p>\n </div>\n\n <!-- Line below content (full width of modal, no side margin) -->\n <div class=\"cqa--mx-2 cqa-w-[calc(100%+1rem)] cqa-flex-shrink-0\">\n <div class=\"cqa-h-px cqa-w-full cqa-bg-[#E5E7EB]\" role=\"presentation\"></div>\n </div>\n\n <!-- Footer: Cancel, Apply (full width in one row) -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <div class=\"cqa-flex cqa-items-stretch cqa-gap-2 cqa-w-full\">\n <div class=\"cqa-flex-1 cqa-min-w-0\">\n <cqa-button variant=\"outlined\" btnSize=\"lg\" [text]=\"'Cancel'\" [fullWidth]=\"true\" (clicked)=\"onCancel()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"></cqa-button>\n </div>\n <div class=\"cqa-flex-1 cqa-min-w-0\">\n <cqa-button variant=\"filled\" btnSize=\"lg\" [text]=\"'Apply'\" [fullWidth]=\"true\" (clicked)=\"onApply()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#3F43EE]'\"></cqa-button>\n </div>\n </div>\n <a href=\"#\" (click)=\"onEditInDepth($event)\"\n class=\"cqa-text-[#3F43EE] cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-1.5 cqa-no-underline hover:cqa-no-underline cqa-self-center\">\n <mat-icon class=\"!cqa-w-4 !cqa-h-4 !cqa-text-[16px]\">open_in_new</mat-icon>\n Edit in depth (open detailed right panel)\n </a>\n </div>\n</div>\n" }]
|
|
56
|
+
}], ctorParameters: function () { return [{ type: i5.CustomEditStepRef, decorators: [{
|
|
57
|
+
type: Inject,
|
|
58
|
+
args: [CUSTOM_EDIT_STEP_REF]
|
|
59
|
+
}] }, { type: undefined, decorators: [{
|
|
60
|
+
type: Optional
|
|
61
|
+
}, {
|
|
62
|
+
type: Inject,
|
|
63
|
+
args: [CUSTOM_EDIT_STEP_DATA]
|
|
64
|
+
}] }]; }, propDecorators: { apply: [{
|
|
65
|
+
type: Output
|
|
66
|
+
}], cancel: [{
|
|
67
|
+
type: Output
|
|
68
|
+
}], editInDepth: [{
|
|
69
|
+
type: Output
|
|
70
|
+
}] } });
|
|
71
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-edit-step.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/custom-edit-step/custom-edit-step.component.ts","../../../../../../src/lib/test-case-details/custom-edit-step/custom-edit-step.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EACvB,SAAS,EACT,YAAY,EACZ,MAAM,EACN,QAAQ,EACR,MAAM,GACP,MAAM,eAAe,CAAC;AACvB,OAAO,EACL,qBAAqB,EACrB,8BAA8B,GAE/B,MAAM,yBAAyB,CAAC;AACjC,OAAO,EAAE,oBAAoB,EAAqB,MAAM,4CAA4C,CAAC;;;;;;;AAQrG,MAAM,OAAO,uBAAuB;IAalC,YACwC,GAAsB,EACjB,IAAyB;QAD9B,QAAG,GAAH,GAAG,CAAmB;QAbpD,UAAK,GAAG,IAAI,YAAY,EAAU,CAAC;QACnC,WAAM,GAAG,IAAI,YAAY,EAAQ,CAAC;QAClC,gBAAW,GAAG,IAAI,YAAY,EAAQ,CAAC;QAEjD,kEAAkE;QAClE,UAAK,GAAW,EAAE,CAAC;QACnB,YAAO,GAAW,EAAE,CAAC;QACrB,uEAAuE;QACvE,oBAAe,GAAG,8EAA8E,CAAC;QACjG,kFAAkF;QAClF,oBAAe,GAAG,KAAK,CAAC;QAMtB,IAAI,IAAI,EAAE;YACR,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,WAAW,IAAI,EAAE,CAAC;YACpC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,OAAO,IAAI,EAAE,CAAC;SACnC;IACH,CAAC;IAED,OAAO;QACL,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC5B,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC7B,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;QACnB,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;IAC5B,CAAC;IAED,OAAO;QACL,IAAI,CAAC,QAAQ,EAAE,CAAC;IAClB,CAAC;IAED,aAAa,CAAC,KAAY;QACxB,KAAK,CAAC,cAAc,EAAE,CAAC;QACvB,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC;QACxB,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,8BAAqC,CAAC,CAAC;IACxD,CAAC;IAED,MAAM,CAAC,KAAY;QACjB,IAAI,IAAI,CAAC,OAAO,EAAE;YAChB,KAAK,CAAC,cAAc,EAAE,CAAC;YACvB,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;SACrC;IACH,CAAC;;oHAhDU,uBAAuB,kBAcxB,oBAAoB,aACR,qBAAqB;wGAfhC,uBAAuB,gLCrBpC,2kNAwGA;2FDnFa,uBAAuB;kBANnC,SAAS;+BACE,sBAAsB,QAE1B,EAAE,KAAK,EAAE,aAAa,EAAE,mBACb,uBAAuB,CAAC,MAAM;;0BAgB5C,MAAM;2BAAC,oBAAoB;;0BAC3B,QAAQ;;0BAAI,MAAM;2BAAC,qBAAqB;4CAdjC,KAAK;sBAAd,MAAM;gBACG,MAAM;sBAAf,MAAM;gBACG,WAAW;sBAApB,MAAM","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  EventEmitter,\n  Inject,\n  Optional,\n  Output,\n} from '@angular/core';\nimport {\n  CUSTOM_EDIT_STEP_DATA,\n  CUSTOM_EDIT_STEP_EDIT_IN_DEPTH,\n  CustomEditStepData,\n} from './custom-edit-step-data';\nimport { CUSTOM_EDIT_STEP_REF, CustomEditStepRef } from './../custom-edit-step/custom-edit-step-ref';\n\n@Component({\n  selector: 'cqa-custom-edit-step',\n  templateUrl: './custom-edit-step.component.html',\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class CustomEditStepComponent {\n  @Output() apply = new EventEmitter<string>();\n  @Output() cancel = new EventEmitter<void>();\n  @Output() editInDepth = new EventEmitter<void>();\n\n  /** Local value while editing (for two-way binding in textarea) */\n  value: string = '';\n  helpUrl: string = '';\n  /** Tooltip shown when hovering over the \"Need help ?\" icon and text */\n  helpTooltipText = 'Not sure what to do? Click to go to our detailed step creation documentation';\n  /** Whether the help tooltip is visible (custom tooltip for use inside overlay) */\n  showHelpTooltip = false;\n\n  constructor(\n    @Inject(CUSTOM_EDIT_STEP_REF) private ref: CustomEditStepRef,\n    @Optional() @Inject(CUSTOM_EDIT_STEP_DATA) data?: CustomEditStepData\n  ) {\n    if (data) {\n      this.value = data.description ?? '';\n      this.helpUrl = data.helpUrl ?? '';\n    }\n  }\n\n  onApply(): void {\n    this.apply.emit(this.value);\n    this.ref.close(this.value);\n  }\n\n  onCancel(): void {\n    this.cancel.emit();\n    this.ref.close(undefined);\n  }\n\n  onClose(): void {\n    this.onCancel();\n  }\n\n  onEditInDepth(event: Event): void {\n    event.preventDefault();\n    this.editInDepth.emit();\n    this.ref.close(CUSTOM_EDIT_STEP_EDIT_IN_DEPTH as any);\n  }\n\n  onHelp(event: Event): void {\n    if (this.helpUrl) {\n      event.preventDefault();\n      window.open(this.helpUrl, '_blank');\n    }\n  }\n}\n","<div\n  class=\"cqa-bg-white cqa-rounded-[12px] cqa-shadow-lg cqa-border cqa-border-solid cqa-border-[#E5E7EB] cqa-w-[500px] cqa-flex cqa-flex-col cqa-gap-[12px] cqa-p-2 cqa-box-border\">\n  <!-- Header: title left; Need help? + close icon right -->\n  <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2 cqa-px-4\">\n    <h2 class=\"cqa-text-[16px] cqa-leading-[24px] cqa-font-bold cqa-text-[#111827] cqa-m-0 cqa-font-[600]\">\n      Step Description\n    </h2>\n    <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n      <!-- Need help? with custom tooltip (works inside overlay) -->\n      <div class=\"cqa-relative cqa-inline-flex\"\n        (mouseenter)=\"showHelpTooltip = true\" (mouseleave)=\"showHelpTooltip = false\">\n        <a *ngIf=\"helpUrl\" href=\"#\" (click)=\"onHelp($event)\"\n          class=\"cqa-flex cqa-items-center cqa-gap-1 cqa-text-[#3F43EE] cqa-text-[12px] cqa-leading-[21px] cqa-no-underline hover:cqa-underline cqa-cursor-pointer\">\n          <svg width=\"17\" height=\"16\" viewBox=\"0 0 17 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" class=\"cqa-flex-shrink-0\" aria-hidden=\"true\">\n            <g clip-path=\"url(#help-icon-clip)\">\n              <path d=\"M8.50033 14.6663C12.4123 14.6663 15.5837 11.6816 15.5837 7.99967C15.5837 4.31778 12.4123 1.33301 8.50033 1.33301C4.58831 1.33301 1.41699 4.31778 1.41699 7.99967C1.41699 11.6816 4.58831 14.6663 8.50033 14.6663Z\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n              <path d=\"M6.43848 6.00038C6.60501 5.55483 6.93371 5.17912 7.36636 4.9398C7.79901 4.70049 8.30769 4.61301 8.80231 4.69285C9.29693 4.7727 9.74556 5.01473 10.0687 5.37607C10.3919 5.7374 10.5688 6.19473 10.5681 6.66705C10.5681 8.00038 8.44306 8.66705 8.44306 8.66705\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n              <path d=\"M8.5 11.333H8.50966\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n            </g>\n            <defs>\n              <clipPath id=\"help-icon-clip\">\n                <rect width=\"17\" height=\"16\" fill=\"white\"/>\n              </clipPath>\n            </defs>\n          </svg>\n          Need help ?\n        </a>\n        <span *ngIf=\"!helpUrl\" class=\"cqa-flex cqa-items-center cqa-gap-1.5 cqa-font-[500] cqa-text-[10px] cqa-cursor-default\">\n          <svg width=\"17\" height=\"16\" viewBox=\"0 0 17 16\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" class=\"cqa-flex-shrink-0 cqa-text-[#3F43EE]\" aria-hidden=\"true\">\n            <g clip-path=\"url(#help-icon-clip-nolink)\">\n              <path d=\"M8.50033 14.6663C12.4123 14.6663 15.5837 11.6816 15.5837 7.99967C15.5837 4.31778 12.4123 1.33301 8.50033 1.33301C4.58831 1.33301 1.41699 4.31778 1.41699 7.99967C1.41699 11.6816 4.58831 14.6663 8.50033 14.6663Z\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n              <path d=\"M6.43848 6.00038C6.60501 5.55483 6.93371 5.17912 7.36636 4.9398C7.79901 4.70049 8.30769 4.61301 8.80231 4.69285C9.29693 4.7727 9.74556 5.01473 10.0687 5.37607C10.3919 5.7374 10.5688 6.19473 10.5681 6.66705C10.5681 8.00038 8.44306 8.66705 8.44306 8.66705\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n              <path d=\"M8.5 11.333H8.50966\" stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n            </g>\n            <defs>\n              <clipPath id=\"help-icon-clip-nolink\">\n                <rect width=\"17\" height=\"16\" fill=\"white\"/>\n              </clipPath>\n            </defs>\n          </svg>\n          Need help ?\n        </span>\n        <!-- Custom tooltip (exact spec: 306×20 content, 6px radius, #0A0A0A, no arrow) -->\n        <div *ngIf=\"showHelpTooltip\" class=\"cqa-absolute cqa-pointer-events-none cqa-z-[100]\"\n          style=\"top: -24px; left: -125px;\"\n          role=\"tooltip\">\n          <div class=\"cqa-text-white cqa-text-center cqa-whitespace-nowrap\"\n            style=\"width: 306px; min-height: 20px; border-radius: 6px; opacity: 1; padding: 4px 8px; background-color: #0A0A0A; line-height: 20px; font-size: 8px;\">\n            {{ helpTooltipText }}\n          </div>\n        </div>\n      </div>\n      <button type=\"button\" (click)=\"onClose()\"\n        class=\"cqa-p-1 cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-flex cqa-items-center cqa-justify-center\"\n        title=\"Close\" aria-label=\"Close\">\n        <mat-icon class=\"!cqa-w-5 !cqa-h-5 !cqa-text-[20px]\">close</mat-icon>\n      </button>\n    </div>\n  </div>\n\n  <!-- Line below header (full width of modal, no side margin) -->\n  <div class=\"cqa--mx-2 cqa-w-[calc(100%+1rem)] cqa-flex-shrink-0\">\n    <div class=\"cqa-h-px cqa-w-full cqa-bg-[#E5E7EB]\" role=\"presentation\"></div>\n  </div>\n\n  <!-- Content: textarea + helper -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5\">\n    <cqa-custom-textarea\n      [(value)]=\"value\"\n      placeholder=\"Enter description\"\n      [fullWidth]=\"true\"\n      [rows]=\"4\"\n      resize=\"vertical\"\n      size=\"md\">\n    </cqa-custom-textarea>\n    <p class=\"cqa-text-[12px] cqa-leading-[18px] cqa-text-[#0A0A0A] cqa-m-0\">\n      Provide a clear description of the step's purpose\n    </p>\n  </div>\n\n  <!-- Line below content (full width of modal, no side margin) -->\n  <div class=\"cqa--mx-2 cqa-w-[calc(100%+1rem)] cqa-flex-shrink-0\">\n    <div class=\"cqa-h-px cqa-w-full cqa-bg-[#E5E7EB]\" role=\"presentation\"></div>\n  </div>\n\n  <!-- Footer: Cancel, Apply (full width in one row) -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n    <div class=\"cqa-flex cqa-items-stretch cqa-gap-2 cqa-w-full\">\n      <div class=\"cqa-flex-1 cqa-min-w-0\">\n        <cqa-button variant=\"outlined\" btnSize=\"lg\" [text]=\"'Cancel'\" [fullWidth]=\"true\" (clicked)=\"onCancel()\"\n          [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"></cqa-button>\n      </div>\n      <div class=\"cqa-flex-1 cqa-min-w-0\">\n        <cqa-button variant=\"filled\" btnSize=\"lg\" [text]=\"'Apply'\" [fullWidth]=\"true\" (clicked)=\"onApply()\"\n          [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#3F43EE]'\"></cqa-button>\n      </div>\n    </div>\n    <a href=\"#\" (click)=\"onEditInDepth($event)\"\n      class=\"cqa-text-[#3F43EE] cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-flex cqa-items-center cqa-gap-1.5 cqa-no-underline hover:cqa-no-underline cqa-self-center\">\n      <mat-icon class=\"!cqa-w-4 !cqa-h-4 !cqa-text-[16px]\">open_in_new</mat-icon>\n      Edit in depth (open detailed right panel)\n    </a>\n  </div>\n</div>\n"]}
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
import { Injectable, Injector } from '@angular/core';
|
|
2
|
+
import { OverlayConfig } from '@angular/cdk/overlay';
|
|
3
|
+
import { ComponentPortal } from '@angular/cdk/portal';
|
|
4
|
+
import { filter } from 'rxjs/operators';
|
|
5
|
+
import { CustomEditStepComponent } from './custom-edit-step.component';
|
|
6
|
+
import { CustomEditStepRef } from './custom-edit-step-ref';
|
|
7
|
+
import { CUSTOM_EDIT_STEP_DATA, } from './custom-edit-step-data';
|
|
8
|
+
import { CUSTOM_EDIT_STEP_REF } from './custom-edit-step-ref';
|
|
9
|
+
import * as i0 from "@angular/core";
|
|
10
|
+
import * as i1 from "@angular/cdk/overlay";
|
|
11
|
+
export class CustomEditStepService {
|
|
12
|
+
constructor(overlay, injector) {
|
|
13
|
+
this.overlay = overlay;
|
|
14
|
+
this.injector = injector;
|
|
15
|
+
/** Currently open Step Description overlay ref; only one panel is allowed at a time. */
|
|
16
|
+
this.currentRef = null;
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Opens the Step Description modal positioned just below the given origin element.
|
|
20
|
+
* If a panel is already open, returns the existing ref and does not open a duplicate.
|
|
21
|
+
* @param origin Element (e.g. description span or edit icon) to position below
|
|
22
|
+
* @param data Initial description and optional help URL
|
|
23
|
+
* @returns Ref with afterClosed() and close(); afterClosed emits the new description on Apply, undefined on Cancel, or EDIT_IN_DEPTH symbol when user clicks "Edit in depth"
|
|
24
|
+
*/
|
|
25
|
+
open(origin, data) {
|
|
26
|
+
if (this.currentRef) {
|
|
27
|
+
return this.currentRef;
|
|
28
|
+
}
|
|
29
|
+
const originEl = origin.nativeElement;
|
|
30
|
+
const gap = 8;
|
|
31
|
+
const modalWidth = 500;
|
|
32
|
+
const margin = 8;
|
|
33
|
+
originEl.scrollIntoView({ block: 'nearest', behavior: 'auto' });
|
|
34
|
+
const positionStrategy = this.overlay.position().global();
|
|
35
|
+
const overlayRef = this.overlay.create(new OverlayConfig({
|
|
36
|
+
hasBackdrop: true,
|
|
37
|
+
backdropClass: 'cdk-overlay-transparent-backdrop',
|
|
38
|
+
scrollStrategy: this.overlay.scrollStrategies.reposition(),
|
|
39
|
+
positionStrategy,
|
|
40
|
+
panelClass: ['cqa-custom-edit-step-panel', 'cqa-ui-root'],
|
|
41
|
+
width: 'min(560px, 90vw)',
|
|
42
|
+
}));
|
|
43
|
+
const updatePosition = () => {
|
|
44
|
+
const rect = originEl.getBoundingClientRect();
|
|
45
|
+
const viewportWidth = window.innerWidth;
|
|
46
|
+
let top = rect.bottom + gap;
|
|
47
|
+
let left = rect.left;
|
|
48
|
+
if (left + modalWidth > viewportWidth - margin) {
|
|
49
|
+
left = viewportWidth - modalWidth - margin;
|
|
50
|
+
}
|
|
51
|
+
if (left < margin) {
|
|
52
|
+
left = margin;
|
|
53
|
+
}
|
|
54
|
+
if (top < margin) {
|
|
55
|
+
top = margin;
|
|
56
|
+
}
|
|
57
|
+
positionStrategy.top(`${top}px`).left(`${left}px`);
|
|
58
|
+
overlayRef.updatePosition();
|
|
59
|
+
};
|
|
60
|
+
const editStepRef = new CustomEditStepRef(overlayRef);
|
|
61
|
+
const injector = Injector.create({
|
|
62
|
+
parent: this.injector,
|
|
63
|
+
providers: [
|
|
64
|
+
{ provide: CUSTOM_EDIT_STEP_REF, useValue: editStepRef },
|
|
65
|
+
{ provide: CUSTOM_EDIT_STEP_DATA, useValue: data },
|
|
66
|
+
],
|
|
67
|
+
});
|
|
68
|
+
updatePosition();
|
|
69
|
+
const portal = new ComponentPortal(CustomEditStepComponent, undefined, injector);
|
|
70
|
+
overlayRef.attach(portal);
|
|
71
|
+
requestAnimationFrame(() => updatePosition());
|
|
72
|
+
const scrollListener = () => updatePosition();
|
|
73
|
+
window.addEventListener('scroll', scrollListener, true);
|
|
74
|
+
window.addEventListener('resize', scrollListener);
|
|
75
|
+
this.currentRef = editStepRef;
|
|
76
|
+
editStepRef.afterClosed().subscribe(() => {
|
|
77
|
+
this.currentRef = null;
|
|
78
|
+
window.removeEventListener('scroll', scrollListener, true);
|
|
79
|
+
window.removeEventListener('resize', scrollListener);
|
|
80
|
+
});
|
|
81
|
+
overlayRef.backdropClick().subscribe(() => editStepRef.close(undefined));
|
|
82
|
+
overlayRef
|
|
83
|
+
.keydownEvents()
|
|
84
|
+
.pipe(filter((e) => e.key === 'Escape' || e.key === 'Esc'))
|
|
85
|
+
.subscribe(() => editStepRef.close(undefined));
|
|
86
|
+
return editStepRef;
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
CustomEditStepService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: CustomEditStepService, deps: [{ token: i1.Overlay }, { token: i0.Injector }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
90
|
+
CustomEditStepService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: CustomEditStepService, providedIn: 'root' });
|
|
91
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: CustomEditStepService, decorators: [{
|
|
92
|
+
type: Injectable,
|
|
93
|
+
args: [{
|
|
94
|
+
providedIn: 'root',
|
|
95
|
+
}]
|
|
96
|
+
}], ctorParameters: function () { return [{ type: i1.Overlay }, { type: i0.Injector }]; } });
|
|
97
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-edit-step.service.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/custom-edit-step/custom-edit-step.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { ElementRef, Injectable, Injector } from '@angular/core';\nimport { Overlay, OverlayConfig } from '@angular/cdk/overlay';\nimport { ComponentPortal } from '@angular/cdk/portal';\nimport { filter } from 'rxjs/operators';\nimport { CustomEditStepComponent } from './custom-edit-step.component';\nimport { CustomEditStepRef } from './custom-edit-step-ref';\nimport {\n  CUSTOM_EDIT_STEP_DATA,\n  CUSTOM_EDIT_STEP_EDIT_IN_DEPTH,\n  CustomEditStepData,\n} from './custom-edit-step-data';\nimport { CUSTOM_EDIT_STEP_REF } from './custom-edit-step-ref';\n\nexport type CustomEditStepResult =\n  | string\n  | undefined\n  | typeof CUSTOM_EDIT_STEP_EDIT_IN_DEPTH;\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class CustomEditStepService {\n  /** Currently open Step Description overlay ref; only one panel is allowed at a time. */\n  private currentRef: CustomEditStepRef<CustomEditStepResult> | null = null;\n\n  constructor(private readonly overlay: Overlay, private readonly injector: Injector) {}\n\n  /**\n   * Opens the Step Description modal positioned just below the given origin element.\n   * If a panel is already open, returns the existing ref and does not open a duplicate.\n   * @param origin Element (e.g. description span or edit icon) to position below\n   * @param data Initial description and optional help URL\n   * @returns Ref with afterClosed() and close(); afterClosed emits the new description on Apply, undefined on Cancel, or EDIT_IN_DEPTH symbol when user clicks \"Edit in depth\"\n   */\n  open(\n    origin: ElementRef<HTMLElement>,\n    data: CustomEditStepData\n  ): CustomEditStepRef<CustomEditStepResult> {\n    if (this.currentRef) {\n      return this.currentRef;\n    }\n    const originEl = origin.nativeElement;\n    const gap = 8;\n    const modalWidth = 500;\n    const margin = 8;\n\n    originEl.scrollIntoView({ block: 'nearest', behavior: 'auto' });\n\n    const positionStrategy = this.overlay.position().global();\n\n    const overlayRef = this.overlay.create(\n      new OverlayConfig({\n        hasBackdrop: true,\n        backdropClass: 'cdk-overlay-transparent-backdrop',\n        scrollStrategy: this.overlay.scrollStrategies.reposition(),\n        positionStrategy,\n        panelClass: ['cqa-custom-edit-step-panel', 'cqa-ui-root'],\n        width: 'min(560px, 90vw)',\n      })\n    );\n\n    const updatePosition = (): void => {\n      const rect = originEl.getBoundingClientRect();\n      const viewportWidth = window.innerWidth;\n      let top = rect.bottom + gap;\n      let left = rect.left;\n      if (left + modalWidth > viewportWidth - margin) {\n        left = viewportWidth - modalWidth - margin;\n      }\n      if (left < margin) {\n        left = margin;\n      }\n      if (top < margin) {\n        top = margin;\n      }\n      positionStrategy.top(`${top}px`).left(`${left}px`);\n      overlayRef.updatePosition();\n    };\n\n    const editStepRef = new CustomEditStepRef<CustomEditStepResult>(overlayRef);\n\n    const injector = Injector.create({\n      parent: this.injector,\n      providers: [\n        { provide: CUSTOM_EDIT_STEP_REF, useValue: editStepRef },\n        { provide: CUSTOM_EDIT_STEP_DATA, useValue: data },\n      ],\n    });\n\n    updatePosition();\n\n    const portal = new ComponentPortal(CustomEditStepComponent, undefined, injector);\n    overlayRef.attach(portal);\n\n    requestAnimationFrame(() => updatePosition());\n\n    const scrollListener = (): void => updatePosition();\n    window.addEventListener('scroll', scrollListener, true);\n    window.addEventListener('resize', scrollListener);\n    this.currentRef = editStepRef;\n    editStepRef.afterClosed().subscribe(() => {\n      this.currentRef = null;\n      window.removeEventListener('scroll', scrollListener, true);\n      window.removeEventListener('resize', scrollListener);\n    });\n\n    overlayRef.backdropClick().subscribe(() => editStepRef.close(undefined));\n    overlayRef\n      .keydownEvents()\n      .pipe(\n        filter((e: KeyboardEvent) => e.key === 'Escape' || e.key === 'Esc')\n      )\n      .subscribe(() => editStepRef.close(undefined));\n\n    return editStepRef;\n  }\n}\n"]}
|