@cqa-lib/cqa-ui 1.1.179 → 1.1.181
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/custom-textarea/custom-textarea.component.mjs +6 -3
- package/esm2020/lib/custom-toggle/custom-toggle.component.mjs +35 -0
- package/esm2020/lib/execution-screen/ai-action-step/ai-action-step.component.mjs +6 -3
- package/esm2020/lib/execution-screen/ai-agent-step/ai-agent-step.component.mjs +6 -3
- package/esm2020/lib/execution-screen/api-step/api-step.component.mjs +6 -3
- package/esm2020/lib/execution-screen/basic-step/basic-step.component.mjs +6 -3
- package/esm2020/lib/execution-screen/condition-step/condition-step.component.mjs +6 -3
- package/esm2020/lib/execution-screen/db-verification-step/db-verification-step.component.mjs +6 -3
- package/esm2020/lib/execution-screen/document-verification-step/document-verification-step.component.mjs +9 -3
- package/esm2020/lib/execution-screen/file-download-step/file-download-step.component.mjs +6 -3
- package/esm2020/lib/execution-screen/live-execution-step/live-execution-step.component.mjs +6 -3
- package/esm2020/lib/execution-screen/loop-step/loop-step.component.mjs +6 -3
- package/esm2020/lib/execution-screen/step-group/step-group.component.mjs +6 -3
- package/esm2020/lib/file-upload/file-upload.component.mjs +87 -0
- package/esm2020/lib/item-list/item-list.component.mjs +59 -0
- package/esm2020/lib/item-list/item-list.model.mjs +2 -0
- package/esm2020/lib/step-builder/step-builder-ai-agent/step-builder-ai-agent.component.mjs +136 -0
- package/esm2020/lib/step-builder/step-builder-custom-code/step-builder-custom-code.component.mjs +65 -0
- package/esm2020/lib/step-builder/step-builder-database/step-builder-database.component.mjs +1 -1
- package/esm2020/lib/step-builder/step-builder-document/step-builder-document.component.mjs +241 -0
- package/esm2020/lib/step-builder/step-builder-document-generation-template-step/step-builder-document-generation-template-step.component.mjs +176 -0
- package/esm2020/lib/step-builder/step-builder-record-step/step-builder-record-step.component.mjs +31 -0
- package/esm2020/lib/test-case-details/custom-edit-step/custom-edit-step.component.mjs +3 -3
- package/esm2020/lib/test-case-details/element-list/element-list.component.mjs +50 -0
- package/esm2020/lib/test-case-details/element-popup/element-popup-data.mjs +5 -0
- package/esm2020/lib/test-case-details/element-popup/element-popup-ref.mjs +32 -0
- package/esm2020/lib/test-case-details/element-popup/element-popup.component.mjs +272 -0
- package/esm2020/lib/test-case-details/element-popup/element-popup.service.mjs +97 -0
- package/esm2020/lib/test-case-details/normal-step/normal-step.component.mjs +151 -19
- package/esm2020/lib/test-case-details/test-data-modal/test-data-modal-data.mjs +5 -0
- package/esm2020/lib/test-case-details/test-data-modal/test-data-modal-environment.model.mjs +10 -0
- package/esm2020/lib/test-case-details/test-data-modal/test-data-modal-parameter.model.mjs +8 -0
- package/esm2020/lib/test-case-details/test-data-modal/test-data-modal-ref.mjs +32 -0
- package/esm2020/lib/test-case-details/test-data-modal/test-data-modal.component.mjs +301 -0
- package/esm2020/lib/test-case-details/test-data-modal/test-data-modal.service.mjs +97 -0
- package/esm2020/lib/ui-kit.module.mjs +58 -3
- package/esm2020/lib/utils/tw-overlay-container.mjs +7 -3
- package/esm2020/public-api.mjs +19 -1
- package/fesm2015/cqa-lib-cqa-ui.mjs +2971 -1106
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs +2966 -1121
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
- package/lib/custom-textarea/custom-textarea.component.d.ts +2 -1
- package/lib/custom-toggle/custom-toggle.component.d.ts +12 -0
- package/lib/execution-screen/ai-action-step/ai-action-step.component.d.ts +2 -1
- package/lib/execution-screen/ai-agent-step/ai-agent-step.component.d.ts +2 -1
- package/lib/execution-screen/api-step/api-step.component.d.ts +2 -1
- package/lib/execution-screen/basic-step/basic-step.component.d.ts +2 -1
- package/lib/execution-screen/condition-step/condition-step.component.d.ts +2 -1
- package/lib/execution-screen/db-verification-step/db-verification-step.component.d.ts +2 -1
- package/lib/execution-screen/document-verification-step/document-verification-step.component.d.ts +2 -1
- package/lib/execution-screen/file-download-step/file-download-step.component.d.ts +2 -1
- package/lib/execution-screen/live-execution-step/live-execution-step.component.d.ts +2 -1
- package/lib/execution-screen/loop-step/loop-step.component.d.ts +2 -1
- package/lib/execution-screen/step-group/step-group.component.d.ts +2 -1
- package/lib/file-upload/file-upload.component.d.ts +26 -0
- package/lib/item-list/item-list.component.d.ts +25 -0
- package/lib/item-list/item-list.model.d.ts +16 -0
- package/lib/step-builder/step-builder-ai-agent/step-builder-ai-agent.component.d.ts +47 -0
- package/lib/step-builder/step-builder-custom-code/step-builder-custom-code.component.d.ts +27 -0
- package/lib/step-builder/step-builder-document/step-builder-document.component.d.ts +81 -0
- package/lib/step-builder/step-builder-document-generation-template-step/step-builder-document-generation-template-step.component.d.ts +55 -0
- package/lib/step-builder/step-builder-record-step/step-builder-record-step.component.d.ts +16 -0
- package/lib/test-case-details/element-list/element-list.component.d.ts +26 -0
- package/lib/test-case-details/element-popup/element-popup-data.d.ts +16 -0
- package/lib/test-case-details/element-popup/element-popup-ref.d.ts +13 -0
- package/lib/test-case-details/element-popup/element-popup.component.d.ts +60 -0
- package/lib/test-case-details/element-popup/element-popup.service.d.ts +23 -0
- package/lib/test-case-details/normal-step/normal-step.component.d.ts +17 -2
- package/lib/test-case-details/test-data-modal/test-data-modal-data.d.ts +31 -0
- package/lib/test-case-details/test-data-modal/test-data-modal-environment.model.d.ts +12 -0
- package/lib/test-case-details/test-data-modal/test-data-modal-parameter.model.d.ts +11 -0
- package/lib/test-case-details/test-data-modal/test-data-modal-ref.d.ts +13 -0
- package/lib/test-case-details/test-data-modal/test-data-modal.component.d.ts +91 -0
- package/lib/test-case-details/test-data-modal/test-data-modal.service.d.ts +23 -0
- package/lib/ui-kit.module.d.ts +60 -49
- package/package.json +1 -1
- package/public-api.d.ts +18 -0
- package/styles.css +1 -1
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import { InjectionToken } from '@angular/core';
|
|
2
|
+
/** Sentinel returned from afterClosed() when user clicked "Edit in depth". */
|
|
3
|
+
export const ELEMENT_POPUP_EDIT_IN_DEPTH = Symbol('ElementPopupEditInDepth');
|
|
4
|
+
export const ELEMENT_POPUP_DATA = new InjectionToken('ELEMENT_POPUP_DATA');
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWxlbWVudC1wb3B1cC1kYXRhLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi90ZXN0LWNhc2UtZGV0YWlscy9lbGVtZW50LXBvcHVwL2VsZW1lbnQtcG9wdXAtZGF0YS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBZ0IvQyw4RUFBOEU7QUFDOUUsTUFBTSxDQUFDLE1BQU0sMkJBQTJCLEdBQUcsTUFBTSxDQUFDLHlCQUF5QixDQUFDLENBQUM7QUFFN0UsTUFBTSxDQUFDLE1BQU0sa0JBQWtCLEdBQUcsSUFBSSxjQUFjLENBQ2xELG9CQUFvQixDQUNyQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiXG5pbXBvcnQgeyBJbmplY3Rpb25Ub2tlbiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEVsZW1lbnRQb3B1cERhdGEge1xuICBlbGVtZW50PzogRWxlbWVudFBvcHVwRGF0YUVsZW1lbnRzO1xuICBsYWJlbHM6IHN0cmluZ1tdO1xuICBlbGVtZW50czogRWxlbWVudFBvcHVwRGF0YUVsZW1lbnRzW107XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmc7XG4gIGhlbHBVcmw/OiBzdHJpbmc7XG59XG5cblxuZXhwb3J0IGludGVyZmFjZSBFbGVtZW50UG9wdXBEYXRhRWxlbWVudHMge1xuICBsYWJlbHM6IHN0cmluZ1tdO1xuICB0aXRsZTogc3RyaW5nO1xuICBzZWxlY3Rvcj86IHN0cmluZztcbn1cbi8qKiBTZW50aW5lbCByZXR1cm5lZCBmcm9tIGFmdGVyQ2xvc2VkKCkgd2hlbiB1c2VyIGNsaWNrZWQgXCJFZGl0IGluIGRlcHRoXCIuICovXG5leHBvcnQgY29uc3QgRUxFTUVOVF9QT1BVUF9FRElUX0lOX0RFUFRIID0gU3ltYm9sKCdFbGVtZW50UG9wdXBFZGl0SW5EZXB0aCcpO1xuXG5leHBvcnQgY29uc3QgRUxFTUVOVF9QT1BVUF9EQVRBID0gbmV3IEluamVjdGlvblRva2VuPEVsZW1lbnRQb3B1cERhdGE+KFxuICAnRUxFTUVOVF9QT1BVUF9EQVRBJ1xuKTtcbiJdfQ==
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { InjectionToken } from '@angular/core';
|
|
2
|
+
import { Subject } from 'rxjs';
|
|
3
|
+
export class ElementPopupRef {
|
|
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_ELEMENT_POPUP_REF = new InjectionToken('CUSTOM_ELEMENT_POPUP_REF');
|
|
32
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWxlbWVudC1wb3B1cC1yZWYuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3Rlc3QtY2FzZS1kZXRhaWxzL2VsZW1lbnQtcG9wdXAvZWxlbWVudC1wb3B1cC1yZWYudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUUvQyxPQUFPLEVBQWMsT0FBTyxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBRTNDLE1BQU0sT0FBTyxlQUFlO0lBSTFCLFlBQTZCLFVBQXNCO1FBQXRCLGVBQVUsR0FBVixVQUFVLENBQVk7UUFIbEMsWUFBTyxHQUFHLElBQUksT0FBTyxFQUF1QixDQUFDO1FBQ3RELGFBQVEsR0FBRyxLQUFLLENBQUM7UUFHdkIsSUFBSSxDQUFDLFVBQVUsQ0FBQyxXQUFXLEVBQUUsQ0FBQyxTQUFTLENBQUMsR0FBRyxFQUFFO1lBQzNDLElBQUksQ0FBQyxXQUFXLENBQUMsU0FBUyxDQUFDLENBQUM7UUFDOUIsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDO0lBRUQsS0FBSyxDQUFDLE1BQWdCO1FBQ3BCLElBQUksSUFBSSxDQUFDLFFBQVEsRUFBRTtZQUNqQixPQUFPO1NBQ1I7UUFDRCxJQUFJLENBQUMsV0FBVyxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBQ3pCLElBQUksQ0FBQyxVQUFVLENBQUMsT0FBTyxFQUFFLENBQUM7SUFDNUIsQ0FBQztJQUVELFdBQVc7UUFDVCxPQUFPLElBQUksQ0FBQyxPQUFPLENBQUMsWUFBWSxFQUFFLENBQUM7SUFDckMsQ0FBQztJQUVPLFdBQVcsQ0FBQyxNQUEyQjtRQUM3QyxJQUFJLElBQUksQ0FBQyxRQUFRLEVBQUU7WUFDakIsT0FBTztTQUNSO1FBQ0QsSUFBSSxDQUFDLFFBQVEsR0FBRyxJQUFJLENBQUM7UUFDckIsSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDMUIsSUFBSSxDQUFDLE9BQU8sQ0FBQyxRQUFRLEVBQUUsQ0FBQztJQUMxQixDQUFDO0NBQ0Y7QUFFRCxNQUFNLENBQUMsTUFBTSx3QkFBd0IsR0FBRyxJQUFJLGNBQWMsQ0FDeEQsMEJBQTBCLENBQzNCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3Rpb25Ub2tlbiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgT3ZlcmxheVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2Nkay9vdmVybGF5JztcbmltcG9ydCB7IE9ic2VydmFibGUsIFN1YmplY3QgfSBmcm9tICdyeGpzJztcblxuZXhwb3J0IGNsYXNzIEVsZW1lbnRQb3B1cFJlZjxUUmVzdWx0ID0gc3RyaW5nIHwgdW5kZWZpbmVkPiB7XG4gIHByaXZhdGUgcmVhZG9ubHkgY2xvc2VkJCA9IG5ldyBTdWJqZWN0PFRSZXN1bHQgfCB1bmRlZmluZWQ+KCk7XG4gIHByaXZhdGUgaXNDbG9zZWQgPSBmYWxzZTtcblxuICBjb25zdHJ1Y3Rvcihwcml2YXRlIHJlYWRvbmx5IG92ZXJsYXlSZWY6IE92ZXJsYXlSZWYpIHtcbiAgICB0aGlzLm92ZXJsYXlSZWYuZGV0YWNobWVudHMoKS5zdWJzY3JpYmUoKCkgPT4ge1xuICAgICAgdGhpcy5maW5pc2hDbG9zZSh1bmRlZmluZWQpO1xuICAgIH0pO1xuICB9XG5cbiAgY2xvc2UocmVzdWx0PzogVFJlc3VsdCk6IHZvaWQge1xuICAgIGlmICh0aGlzLmlzQ2xvc2VkKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIHRoaXMuZmluaXNoQ2xvc2UocmVzdWx0KTtcbiAgICB0aGlzLm92ZXJsYXlSZWYuZGlzcG9zZSgpO1xuICB9XG5cbiAgYWZ0ZXJDbG9zZWQoKTogT2JzZXJ2YWJsZTxUUmVzdWx0IHwgdW5kZWZpbmVkPiB7XG4gICAgcmV0dXJuIHRoaXMuY2xvc2VkJC5hc09ic2VydmFibGUoKTtcbiAgfVxuXG4gIHByaXZhdGUgZmluaXNoQ2xvc2UocmVzdWx0OiBUUmVzdWx0IHwgdW5kZWZpbmVkKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuaXNDbG9zZWQpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhpcy5pc0Nsb3NlZCA9IHRydWU7XG4gICAgdGhpcy5jbG9zZWQkLm5leHQocmVzdWx0KTtcbiAgICB0aGlzLmNsb3NlZCQuY29tcGxldGUoKTtcbiAgfVxufVxuXG5leHBvcnQgY29uc3QgQ1VTVE9NX0VMRU1FTlRfUE9QVVBfUkVGID0gbmV3IEluamVjdGlvblRva2VuPEVsZW1lbnRQb3B1cFJlZj4oXG4gICdDVVNUT01fRUxFTUVOVF9QT1BVUF9SRUYnXG4pO1xuIl19
|
|
@@ -0,0 +1,272 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Inject, Input, Optional, Output, } from '@angular/core';
|
|
2
|
+
import { ELEMENT_POPUP_DATA, ELEMENT_POPUP_EDIT_IN_DEPTH, } from './element-popup-data';
|
|
3
|
+
import { CUSTOM_ELEMENT_POPUP_REF } from './../element-popup/element-popup-ref';
|
|
4
|
+
import { FormBuilder, Validators } from '@angular/forms';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "@angular/forms";
|
|
7
|
+
import * as i2 from "@angular/material/icon";
|
|
8
|
+
import * as i3 from "../../custom-input/custom-input.component";
|
|
9
|
+
import * as i4 from "../../dynamic-select/dynamic-select-field.component";
|
|
10
|
+
import * as i5 from "../../button/button.component";
|
|
11
|
+
import * as i6 from "../../badge/badge.component";
|
|
12
|
+
import * as i7 from "../../search-bar/search-bar.component";
|
|
13
|
+
import * as i8 from "../element-list/element-list.component";
|
|
14
|
+
import * as i9 from "@angular/common";
|
|
15
|
+
import * as i10 from "./../element-popup/element-popup-ref";
|
|
16
|
+
export class ElementPopupComponent {
|
|
17
|
+
constructor(ref, data, fb, cdr) {
|
|
18
|
+
this.ref = ref;
|
|
19
|
+
this.cdr = cdr;
|
|
20
|
+
this.value = '';
|
|
21
|
+
this.helpUrl = '';
|
|
22
|
+
this.labels = [];
|
|
23
|
+
this.element = { title: '', selector: '', labels: [] };
|
|
24
|
+
this.elements = [];
|
|
25
|
+
this.enableForm = false;
|
|
26
|
+
this.isOnRecord = false;
|
|
27
|
+
this.apply = new EventEmitter();
|
|
28
|
+
this.cancel = new EventEmitter();
|
|
29
|
+
this.editInDepth = new EventEmitter();
|
|
30
|
+
this.searchElement = new EventEmitter();
|
|
31
|
+
/** Tooltip shown when hovering over the "Need help ?" icon and text */
|
|
32
|
+
this.helpTooltipText = 'Not sure what to do? Click to go to our detailed step creation documentation';
|
|
33
|
+
/** Whether the help tooltip is visible (custom tooltip for use inside overlay) */
|
|
34
|
+
this.showHelpTooltip = false;
|
|
35
|
+
/** Whether we're in edit mode (true) or create mode (false) */
|
|
36
|
+
this.isEditMode = false;
|
|
37
|
+
/** Cached select options to avoid recomputing on every change detection */
|
|
38
|
+
this.cachedSelectOptions = [];
|
|
39
|
+
this.injectedData = data;
|
|
40
|
+
// Initialize FormBuilder if not injected (for Storybook/testing scenarios)
|
|
41
|
+
this.fb = fb || new FormBuilder();
|
|
42
|
+
this.initializeForm();
|
|
43
|
+
}
|
|
44
|
+
ngOnChanges(changes) {
|
|
45
|
+
// Update cached options when labels input changes
|
|
46
|
+
if (changes['labels'] && !changes['labels'].firstChange) {
|
|
47
|
+
this.updateSelectOptions();
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
ngOnInit() {
|
|
51
|
+
// Use injected data if inputs are not provided (for backward compatibility)
|
|
52
|
+
if (this.injectedData) {
|
|
53
|
+
if (!this.value) {
|
|
54
|
+
this.value = this.injectedData.description ?? '';
|
|
55
|
+
}
|
|
56
|
+
if (!this.helpUrl) {
|
|
57
|
+
this.helpUrl = this.injectedData.helpUrl ?? '';
|
|
58
|
+
}
|
|
59
|
+
if (this.labels.length === 0) {
|
|
60
|
+
this.labels = this.injectedData.labels ?? [];
|
|
61
|
+
}
|
|
62
|
+
if (this.elements.length === 0) {
|
|
63
|
+
this.elements = this.injectedData.elements ?? [];
|
|
64
|
+
}
|
|
65
|
+
if (!this.element || !this.element.title) {
|
|
66
|
+
this.element = this.injectedData.element ?? { title: '', selector: '', labels: [] };
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
// Update select options after labels are potentially set
|
|
70
|
+
this.updateSelectOptions();
|
|
71
|
+
}
|
|
72
|
+
initializeForm() {
|
|
73
|
+
this.form = this.fb.group({
|
|
74
|
+
name: ['', [Validators.required]],
|
|
75
|
+
screenName: ['', [Validators.required]],
|
|
76
|
+
value: ['', [Validators.required]],
|
|
77
|
+
label: [null]
|
|
78
|
+
});
|
|
79
|
+
}
|
|
80
|
+
populateFormForEdit() {
|
|
81
|
+
if (this.element && this.element.title) {
|
|
82
|
+
this.isEditMode = true;
|
|
83
|
+
this.form.patchValue({
|
|
84
|
+
name: this.element.title || '',
|
|
85
|
+
screenName: this.element.selector || '',
|
|
86
|
+
value: this.element.selector || '',
|
|
87
|
+
label: this.element.labels && this.element.labels.length > 0 ? this.element.labels[0] : null
|
|
88
|
+
});
|
|
89
|
+
}
|
|
90
|
+
else {
|
|
91
|
+
this.isEditMode = false;
|
|
92
|
+
this.form.reset();
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
resetForm() {
|
|
96
|
+
this.isEditMode = false;
|
|
97
|
+
this.form.reset({
|
|
98
|
+
name: '',
|
|
99
|
+
screenName: '',
|
|
100
|
+
value: '',
|
|
101
|
+
label: null
|
|
102
|
+
});
|
|
103
|
+
}
|
|
104
|
+
onApply() {
|
|
105
|
+
if (this.form.invalid) {
|
|
106
|
+
// Mark all fields as touched to show validation errors
|
|
107
|
+
Object.keys(this.form.controls).forEach(key => {
|
|
108
|
+
this.form.get(key)?.markAsTouched();
|
|
109
|
+
});
|
|
110
|
+
return;
|
|
111
|
+
}
|
|
112
|
+
const formValue = this.form.value;
|
|
113
|
+
const elementData = {
|
|
114
|
+
title: formValue.name,
|
|
115
|
+
selector: formValue.value,
|
|
116
|
+
labels: formValue.label ? [formValue.label] : []
|
|
117
|
+
};
|
|
118
|
+
if (this.isEditMode) {
|
|
119
|
+
// Update existing element
|
|
120
|
+
this.element = elementData;
|
|
121
|
+
}
|
|
122
|
+
else {
|
|
123
|
+
// Create new element
|
|
124
|
+
this.elements = [...this.elements, elementData];
|
|
125
|
+
this.element = elementData;
|
|
126
|
+
}
|
|
127
|
+
// Emit the selector value (for backward compatibility)
|
|
128
|
+
this.value = formValue.value;
|
|
129
|
+
this.apply.emit(formValue.value);
|
|
130
|
+
this.ref.close(formValue.value);
|
|
131
|
+
// Close form after applying
|
|
132
|
+
this.enableForm = false;
|
|
133
|
+
this.resetForm();
|
|
134
|
+
}
|
|
135
|
+
toggleForm() {
|
|
136
|
+
if (!this.enableForm) {
|
|
137
|
+
// Opening form - check if we're editing existing element
|
|
138
|
+
this.populateFormForEdit();
|
|
139
|
+
}
|
|
140
|
+
else {
|
|
141
|
+
// Closing form - reset
|
|
142
|
+
this.resetForm();
|
|
143
|
+
}
|
|
144
|
+
this.enableForm = !this.enableForm;
|
|
145
|
+
}
|
|
146
|
+
/** Called when "Create New" button is clicked - explicitly sets create mode */
|
|
147
|
+
openCreateForm() {
|
|
148
|
+
this.isEditMode = false;
|
|
149
|
+
this.resetForm();
|
|
150
|
+
this.enableForm = true;
|
|
151
|
+
}
|
|
152
|
+
toggleRecord() {
|
|
153
|
+
this.isOnRecord = !this.isOnRecord;
|
|
154
|
+
}
|
|
155
|
+
onCancel() {
|
|
156
|
+
// Reset form when canceling
|
|
157
|
+
if (this.enableForm) {
|
|
158
|
+
this.resetForm();
|
|
159
|
+
this.enableForm = false;
|
|
160
|
+
}
|
|
161
|
+
this.cancel.emit();
|
|
162
|
+
this.ref.close(undefined);
|
|
163
|
+
}
|
|
164
|
+
onClose() {
|
|
165
|
+
this.onCancel();
|
|
166
|
+
}
|
|
167
|
+
onEditInDepth(event) {
|
|
168
|
+
event.preventDefault();
|
|
169
|
+
this.editInDepth.emit();
|
|
170
|
+
this.ref.close(ELEMENT_POPUP_EDIT_IN_DEPTH);
|
|
171
|
+
}
|
|
172
|
+
search(event) {
|
|
173
|
+
console.log(event);
|
|
174
|
+
this.searchElement.emit(event);
|
|
175
|
+
}
|
|
176
|
+
onHelp(event) {
|
|
177
|
+
if (this.helpUrl) {
|
|
178
|
+
event.preventDefault();
|
|
179
|
+
window.open(this.helpUrl, '_blank');
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
onVariableValueChange(variableName, value) {
|
|
183
|
+
console.log(value);
|
|
184
|
+
}
|
|
185
|
+
/** Update cached select options when labels change */
|
|
186
|
+
updateSelectOptions() {
|
|
187
|
+
// Limit options to prevent performance issues - only use unique labels, max 100
|
|
188
|
+
const uniqueLabels = Array.from(new Set(this.labels || [])).slice(0, 100);
|
|
189
|
+
this.cachedSelectOptions = uniqueLabels.map(label => ({
|
|
190
|
+
id: label,
|
|
191
|
+
name: label
|
|
192
|
+
}));
|
|
193
|
+
// Clear cached config to force recreation
|
|
194
|
+
this.cachedSelectConfig = undefined;
|
|
195
|
+
}
|
|
196
|
+
getSelectConfig() {
|
|
197
|
+
// Return cached config if available and labels haven't changed
|
|
198
|
+
if (this.cachedSelectConfig) {
|
|
199
|
+
return this.cachedSelectConfig;
|
|
200
|
+
}
|
|
201
|
+
// Create new config with cached options
|
|
202
|
+
this.cachedSelectConfig = {
|
|
203
|
+
key: "label",
|
|
204
|
+
label: "Label",
|
|
205
|
+
placeholder: "Select label",
|
|
206
|
+
options: this.cachedSelectOptions,
|
|
207
|
+
onChange: (value) => {
|
|
208
|
+
this.form.patchValue({ label: value });
|
|
209
|
+
this.onVariableValueChange("label", value);
|
|
210
|
+
}
|
|
211
|
+
};
|
|
212
|
+
return this.cachedSelectConfig;
|
|
213
|
+
}
|
|
214
|
+
getFormControl(controlName) {
|
|
215
|
+
return this.form.get(controlName);
|
|
216
|
+
}
|
|
217
|
+
getFormControlValue(controlName) {
|
|
218
|
+
return this.form.get(controlName)?.value || '';
|
|
219
|
+
}
|
|
220
|
+
onFormControlChange(controlName, value) {
|
|
221
|
+
this.form.patchValue({ [controlName]: value });
|
|
222
|
+
}
|
|
223
|
+
onElementClick(element) {
|
|
224
|
+
// Handle element click - you can customize this behavior
|
|
225
|
+
this.element = element;
|
|
226
|
+
// Optionally close the form if it's open
|
|
227
|
+
if (this.enableForm) {
|
|
228
|
+
this.enableForm = false;
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
ElementPopupComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ElementPopupComponent, deps: [{ token: CUSTOM_ELEMENT_POPUP_REF }, { token: ELEMENT_POPUP_DATA, optional: true }, { token: i1.FormBuilder, optional: true }, { token: i0.ChangeDetectorRef, optional: true }], target: i0.ɵɵFactoryTarget.Component });
|
|
233
|
+
ElementPopupComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: ElementPopupComponent, selector: "cqa-element-popup", inputs: { value: "value", helpUrl: "helpUrl", labels: "labels", element: "element", elements: "elements", enableForm: "enableForm", isOnRecord: "isOnRecord" }, outputs: { apply: "apply", cancel: "cancel", editInDepth: "editInDepth", searchElement: "searchElement" }, host: { classAttribute: "cqa-ui-root" }, usesOnChanges: true, 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 Element\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 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 <ng-container *ngIf=\"enableForm && !isOnRecord\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-px-1 cqa-pt-3\">\n <div class=\"cqa-flex cqa-gap-1.5\">\n <cqa-custom-input \n class=\"cqa-w-1/2\" \n label=\"Name\" \n placeholder=\"default-element\"\n [value]=\"getFormControlValue('name')\"\n [errors]=\"getFormControl('name')?.touched && getFormControl('name')?.invalid ? ['Name is required'] : []\"\n [required]=\"true\"\n (valueChange)=\"onFormControlChange('name', $event)\">\n </cqa-custom-input>\n <cqa-custom-input \n class=\"cqa-w-1/2\" \n label=\"Screen Name\" \n placeholder=\"default-screen\"\n [value]=\"getFormControlValue('screenName')\"\n [errors]=\"getFormControl('screenName')?.touched && getFormControl('screenName')?.invalid ? ['Screen Name is required'] : []\"\n [required]=\"true\"\n (valueChange)=\"onFormControlChange('screenName', $event)\">\n </cqa-custom-input>\n </div>\n <cqa-custom-input \n class=\"cqa-w-full\" \n label=\"Enter Value\" \n placeholder=\"#default_id\"\n [value]=\"getFormControlValue('value')\"\n [errors]=\"getFormControl('value')?.touched && getFormControl('value')?.invalid ? ['Value is required'] : []\"\n [required]=\"true\"\n (valueChange)=\"onFormControlChange('value', $event)\">\n </cqa-custom-input>\n <div class=\"cqa-w-full cqa-position-relative\">\n <cqa-dynamic-select [form]=\"form\" [config]=\"getSelectConfig()\"></cqa-dynamic-select>\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)=\"toggleForm()\"\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]=\"isEditMode ? 'Update' : 'Create'\" [fullWidth]=\"true\" (clicked)=\"onApply()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#3F43EE]'\"></cqa-button>\n </div>\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 </ng-container>\n\n<ng-container *ngIf=\"isOnRecord && !enableForm\">\n <div\n class=\"cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-text-center cqa-gap-3 cqa-pt-10\">\n\n <!-- Video Icon -->\n <div\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-[64px] cqa-h-[64px] cqa-rounded-full cqa-bg-[#FEE2E2]\">\n <svg width=\"32\" height=\"32\" viewBox=\"0 0 32 32\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M21.333 17.3333L28.297 21.9759C28.3974 22.0428 28.514 22.0811 28.6345 22.0868C28.7549 22.0926 28.8747 22.0656 28.981 22.0087C29.0873 21.9517 29.1762 21.8671 29.2382 21.7636C29.3002 21.6602 29.3329 21.5419 29.333 21.4213V10.4933C29.333 10.376 29.3021 10.2607 29.2434 10.1592C29.1846 10.0577 29.1001 9.97344 28.9984 9.91501C28.8967 9.85658 28.7814 9.82602 28.6641 9.82642C28.5468 9.82682 28.4317 9.85816 28.3303 9.91728L21.333 13.9999\" stroke=\"#E7000B\" stroke-width=\"2.66667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M18.667 8H5.33366C3.8609 8 2.66699 9.19391 2.66699 10.6667V21.3333C2.66699 22.8061 3.8609 24 5.33366 24H18.667C20.1398 24 21.3337 22.8061 21.3337 21.3333V10.6667C21.3337 9.19391 20.1398 8 18.667 8Z\" stroke=\"#E7000B\" stroke-width=\"2.66667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </svg>\n \n </div>\n\n <!-- Title -->\n <h4 class=\"cqa-m-0 cqa-text-[16px] cqa-leading-[24px] cqa-font-[600] cqa-text-[#0A0A0A]\">\n Recording Mode Active\n </h4>\n\n <!-- Subtitle -->\n <p class=\"cqa-m-0 cqa-text-[12px] cqa-leading-[18px] cqa-text-[#6B7280]\">\n Click on any element in the browser to capture it\n </p>\n </div>\n\n <!-- Footer -->\n <div class=\"cqa-flex cqa-justify-center cqa-pb-4\">\n <cqa-button\n variant=\"outlined\"\n btnSize=\"lg\"\n [text]=\"'Cancel Recording'\"\n (clicked)=\"toggleRecord()\"\n [customClass]=\"'cqa-text-[14px] cqa-px-16 cqa-py-[9px] cqa-border-[#414146]'\">\n </cqa-button>\n </div>\n</ng-container>\n\n\n <ng-container *ngIf=\"!enableForm && !isOnRecord\">\n<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-px-1 cqa-pt-3\">\n\n <!-- Selected -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <span class=\"cqa-text-[12px] cqa-text-[#6B7280]\">Selected</span>\n\n <div\n class=\"cqa-flex cqa-items-center cqa-justify-between\n cqa-bg-[#FFFBEB] cqa-border cqa-border-[#fadfba]\n cqa-rounded-lg cqa-p-3 cqa-border-solid cqa-gap-3\">\n\n <div class=\"cqa-flex cqa-px-3 cqa-py-2 cqa-rounded-lg cqa-flex-col cqa-gap-0.5 cqa-bg-[#f5f5f5] cqa-flex-1\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2\">\n <span class=\"cqa-text-[14px] cqa-font-[600] cqa-text-[#111827]\">\n {{element.title}}\n </span>\n\n <div class=\"cqa-flex cqa-gap-2\">\n <span *ngFor=\"let l of element.labels\"\n class=\"cqa-text-[10px] cqa-px-1.5 cqa-py-0.5\n cqa-rounded cqa-bg-[#EEF2FF] cqa-text-[#3F43EE] cqa-rounded-full cqa-px-2 cqa-bg-[#eff6ff] cqa-border cqa-border-solid cqa-border-[#c8e0ff]\">\n {{l}}\n </span>\n </div>\n </div>\n\n <span class=\"cqa-text-[11px] cqa-text-[#6B7280]\">\n {{element.selector}}\n </span>\n </div>\n <cqa-button variant=\"outlined\" icon=\"edit\" btnSize=\"lg\" [text]=\"'Edit'\" [fullWidth]=\"true\" (clicked)=\"toggleForm()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"></cqa-button>\n </div>\n </div>\n\n <!-- Recent -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <span class=\"cqa-text-[12px] cqa-text-[#6B7280]\">Recent</span>\n\n <div class=\"cqa-flex cqa-gap-2 cqa-overflow-x-auto cqa-scrollbar-thin cqa-scrollbar-track-transparent cqa-scrollbar-thumb-[#E5E7EB] cqa-scrollbar-thumb-rounded-full cqa-scrollbar-thumb-hover:cqa-bg-[#D1D5DB]\">\n\n <cqa-badge *ngFor=\"let label of labels\" class=\"cqa-element-badge cqa-mb-2 cqa-chip !cqa-bg-white !cqa-text-[12px] cqa-whitespace-nowrap\" [label]=\"label\"></cqa-badge>\n </div>\n </div>\n\n <!-- Element Library -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <span class=\"cqa-text-[12px] cqa-text-[#6B7280]\">Element Library</span>\n <cqa-search-bar [fullWidth]=\"true\" placeholder=\"Search library\" (valueChange)=\"search($event)\"></cqa-search-bar> \n </div>\n\n <cqa-element-list \n [items]=\"elements\"\n [titleKey]=\"'title'\"\n [selectorKey]=\"'selector'\"\n [labelsKey]=\"'labels'\"\n [maxHeight]=\"'200px'\"\n (itemClick)=\"onElementClick($event)\">\n </cqa-element-list>\n</div>\n\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\" icon=\"radio_button_checked\" btnSize=\"lg\" [text]=\"'Record'\" [fullWidth]=\"true\" (clicked)=\"toggleRecord()\"\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=\"outlined\" icon=\"add\" btnSize=\"lg\" [text]=\"'Create New'\" [fullWidth]=\"true\" (clicked)=\"openCreateForm()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"></cqa-button>\n </div>\n </div>\n </div>\n </ng-container>\n\n</div>\n", components: [{ type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i3.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: i4.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore"] }, { type: i5.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }, { type: i6.BadgeComponent, selector: "cqa-badge", inputs: ["type", "label", "icon", "iconLibrary", "variant", "size", "backgroundColor", "textColor", "borderColor", "iconBackgroundColor", "iconColor", "iconSize", "inlineStyles", "key", "value", "keyTextColor", "valueTextColor", "isLoading"] }, { type: i7.SearchBarComponent, selector: "cqa-search-bar", inputs: ["placeholder", "value", "disabled", "showClear", "ariaLabel", "autoFocus", "size", "fullWidth"], outputs: ["valueChange", "search", "cleared"] }, { type: i8.ElementListComponent, selector: "cqa-element-list", inputs: ["items", "titleKey", "selectorKey", "labelsKey", "maxHeight"], outputs: ["itemClick"] }], directives: [{ type: i9.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i9.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
234
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ElementPopupComponent, decorators: [{
|
|
235
|
+
type: Component,
|
|
236
|
+
args: [{ selector: 'cqa-element-popup', 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 Element\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 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 <ng-container *ngIf=\"enableForm && !isOnRecord\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-px-1 cqa-pt-3\">\n <div class=\"cqa-flex cqa-gap-1.5\">\n <cqa-custom-input \n class=\"cqa-w-1/2\" \n label=\"Name\" \n placeholder=\"default-element\"\n [value]=\"getFormControlValue('name')\"\n [errors]=\"getFormControl('name')?.touched && getFormControl('name')?.invalid ? ['Name is required'] : []\"\n [required]=\"true\"\n (valueChange)=\"onFormControlChange('name', $event)\">\n </cqa-custom-input>\n <cqa-custom-input \n class=\"cqa-w-1/2\" \n label=\"Screen Name\" \n placeholder=\"default-screen\"\n [value]=\"getFormControlValue('screenName')\"\n [errors]=\"getFormControl('screenName')?.touched && getFormControl('screenName')?.invalid ? ['Screen Name is required'] : []\"\n [required]=\"true\"\n (valueChange)=\"onFormControlChange('screenName', $event)\">\n </cqa-custom-input>\n </div>\n <cqa-custom-input \n class=\"cqa-w-full\" \n label=\"Enter Value\" \n placeholder=\"#default_id\"\n [value]=\"getFormControlValue('value')\"\n [errors]=\"getFormControl('value')?.touched && getFormControl('value')?.invalid ? ['Value is required'] : []\"\n [required]=\"true\"\n (valueChange)=\"onFormControlChange('value', $event)\">\n </cqa-custom-input>\n <div class=\"cqa-w-full cqa-position-relative\">\n <cqa-dynamic-select [form]=\"form\" [config]=\"getSelectConfig()\"></cqa-dynamic-select>\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)=\"toggleForm()\"\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]=\"isEditMode ? 'Update' : 'Create'\" [fullWidth]=\"true\" (clicked)=\"onApply()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#3F43EE]'\"></cqa-button>\n </div>\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 </ng-container>\n\n<ng-container *ngIf=\"isOnRecord && !enableForm\">\n <div\n class=\"cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-text-center cqa-gap-3 cqa-pt-10\">\n\n <!-- Video Icon -->\n <div\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-[64px] cqa-h-[64px] cqa-rounded-full cqa-bg-[#FEE2E2]\">\n <svg width=\"32\" height=\"32\" viewBox=\"0 0 32 32\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M21.333 17.3333L28.297 21.9759C28.3974 22.0428 28.514 22.0811 28.6345 22.0868C28.7549 22.0926 28.8747 22.0656 28.981 22.0087C29.0873 21.9517 29.1762 21.8671 29.2382 21.7636C29.3002 21.6602 29.3329 21.5419 29.333 21.4213V10.4933C29.333 10.376 29.3021 10.2607 29.2434 10.1592C29.1846 10.0577 29.1001 9.97344 28.9984 9.91501C28.8967 9.85658 28.7814 9.82602 28.6641 9.82642C28.5468 9.82682 28.4317 9.85816 28.3303 9.91728L21.333 13.9999\" stroke=\"#E7000B\" stroke-width=\"2.66667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n <path d=\"M18.667 8H5.33366C3.8609 8 2.66699 9.19391 2.66699 10.6667V21.3333C2.66699 22.8061 3.8609 24 5.33366 24H18.667C20.1398 24 21.3337 22.8061 21.3337 21.3333V10.6667C21.3337 9.19391 20.1398 8 18.667 8Z\" stroke=\"#E7000B\" stroke-width=\"2.66667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n </svg>\n \n </div>\n\n <!-- Title -->\n <h4 class=\"cqa-m-0 cqa-text-[16px] cqa-leading-[24px] cqa-font-[600] cqa-text-[#0A0A0A]\">\n Recording Mode Active\n </h4>\n\n <!-- Subtitle -->\n <p class=\"cqa-m-0 cqa-text-[12px] cqa-leading-[18px] cqa-text-[#6B7280]\">\n Click on any element in the browser to capture it\n </p>\n </div>\n\n <!-- Footer -->\n <div class=\"cqa-flex cqa-justify-center cqa-pb-4\">\n <cqa-button\n variant=\"outlined\"\n btnSize=\"lg\"\n [text]=\"'Cancel Recording'\"\n (clicked)=\"toggleRecord()\"\n [customClass]=\"'cqa-text-[14px] cqa-px-16 cqa-py-[9px] cqa-border-[#414146]'\">\n </cqa-button>\n </div>\n</ng-container>\n\n\n <ng-container *ngIf=\"!enableForm && !isOnRecord\">\n<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-px-1 cqa-pt-3\">\n\n <!-- Selected -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <span class=\"cqa-text-[12px] cqa-text-[#6B7280]\">Selected</span>\n\n <div\n class=\"cqa-flex cqa-items-center cqa-justify-between\n cqa-bg-[#FFFBEB] cqa-border cqa-border-[#fadfba]\n cqa-rounded-lg cqa-p-3 cqa-border-solid cqa-gap-3\">\n\n <div class=\"cqa-flex cqa-px-3 cqa-py-2 cqa-rounded-lg cqa-flex-col cqa-gap-0.5 cqa-bg-[#f5f5f5] cqa-flex-1\">\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-gap-2\">\n <span class=\"cqa-text-[14px] cqa-font-[600] cqa-text-[#111827]\">\n {{element.title}}\n </span>\n\n <div class=\"cqa-flex cqa-gap-2\">\n <span *ngFor=\"let l of element.labels\"\n class=\"cqa-text-[10px] cqa-px-1.5 cqa-py-0.5\n cqa-rounded cqa-bg-[#EEF2FF] cqa-text-[#3F43EE] cqa-rounded-full cqa-px-2 cqa-bg-[#eff6ff] cqa-border cqa-border-solid cqa-border-[#c8e0ff]\">\n {{l}}\n </span>\n </div>\n </div>\n\n <span class=\"cqa-text-[11px] cqa-text-[#6B7280]\">\n {{element.selector}}\n </span>\n </div>\n <cqa-button variant=\"outlined\" icon=\"edit\" btnSize=\"lg\" [text]=\"'Edit'\" [fullWidth]=\"true\" (clicked)=\"toggleForm()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"></cqa-button>\n </div>\n </div>\n\n <!-- Recent -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <span class=\"cqa-text-[12px] cqa-text-[#6B7280]\">Recent</span>\n\n <div class=\"cqa-flex cqa-gap-2 cqa-overflow-x-auto cqa-scrollbar-thin cqa-scrollbar-track-transparent cqa-scrollbar-thumb-[#E5E7EB] cqa-scrollbar-thumb-rounded-full cqa-scrollbar-thumb-hover:cqa-bg-[#D1D5DB]\">\n\n <cqa-badge *ngFor=\"let label of labels\" class=\"cqa-element-badge cqa-mb-2 cqa-chip !cqa-bg-white !cqa-text-[12px] cqa-whitespace-nowrap\" [label]=\"label\"></cqa-badge>\n </div>\n </div>\n\n <!-- Element Library -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <span class=\"cqa-text-[12px] cqa-text-[#6B7280]\">Element Library</span>\n <cqa-search-bar [fullWidth]=\"true\" placeholder=\"Search library\" (valueChange)=\"search($event)\"></cqa-search-bar> \n </div>\n\n <cqa-element-list \n [items]=\"elements\"\n [titleKey]=\"'title'\"\n [selectorKey]=\"'selector'\"\n [labelsKey]=\"'labels'\"\n [maxHeight]=\"'200px'\"\n (itemClick)=\"onElementClick($event)\">\n </cqa-element-list>\n</div>\n\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\" icon=\"radio_button_checked\" btnSize=\"lg\" [text]=\"'Record'\" [fullWidth]=\"true\" (clicked)=\"toggleRecord()\"\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=\"outlined\" icon=\"add\" btnSize=\"lg\" [text]=\"'Create New'\" [fullWidth]=\"true\" (clicked)=\"openCreateForm()\"\n [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"></cqa-button>\n </div>\n </div>\n </div>\n </ng-container>\n\n</div>\n" }]
|
|
237
|
+
}], ctorParameters: function () { return [{ type: i10.ElementPopupRef, decorators: [{
|
|
238
|
+
type: Inject,
|
|
239
|
+
args: [CUSTOM_ELEMENT_POPUP_REF]
|
|
240
|
+
}] }, { type: undefined, decorators: [{
|
|
241
|
+
type: Optional
|
|
242
|
+
}, {
|
|
243
|
+
type: Inject,
|
|
244
|
+
args: [ELEMENT_POPUP_DATA]
|
|
245
|
+
}] }, { type: i1.FormBuilder, decorators: [{
|
|
246
|
+
type: Optional
|
|
247
|
+
}] }, { type: i0.ChangeDetectorRef, decorators: [{
|
|
248
|
+
type: Optional
|
|
249
|
+
}] }]; }, propDecorators: { value: [{
|
|
250
|
+
type: Input
|
|
251
|
+
}], helpUrl: [{
|
|
252
|
+
type: Input
|
|
253
|
+
}], labels: [{
|
|
254
|
+
type: Input
|
|
255
|
+
}], element: [{
|
|
256
|
+
type: Input
|
|
257
|
+
}], elements: [{
|
|
258
|
+
type: Input
|
|
259
|
+
}], enableForm: [{
|
|
260
|
+
type: Input
|
|
261
|
+
}], isOnRecord: [{
|
|
262
|
+
type: Input
|
|
263
|
+
}], apply: [{
|
|
264
|
+
type: Output
|
|
265
|
+
}], cancel: [{
|
|
266
|
+
type: Output
|
|
267
|
+
}], editInDepth: [{
|
|
268
|
+
type: Output
|
|
269
|
+
}], searchElement: [{
|
|
270
|
+
type: Output
|
|
271
|
+
}] } });
|
|
272
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"element-popup.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/element-popup/element-popup.component.ts","../../../../../../src/lib/test-case-details/element-popup/element-popup.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Inject,\n  Input,\n  OnChanges,\n  OnInit,\n  Optional,\n  Output,\n  SimpleChanges,\n} from '@angular/core';\nimport {\n  ELEMENT_POPUP_DATA,\n  ELEMENT_POPUP_EDIT_IN_DEPTH,\n  ElementPopupData,\n  ElementPopupDataElements,\n} from './element-popup-data';\nimport { CUSTOM_ELEMENT_POPUP_REF, ElementPopupRef } from './../element-popup/element-popup-ref';\nimport { FormBuilder, FormGroup, FormControl, Validators } from '@angular/forms';\nimport { DynamicSelectFieldConfig } from '../../dynamic-select/dynamic-select-field.component';\n\n@Component({\n  selector: 'cqa-element-popup',\n  templateUrl: './element-popup.component.html',\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ElementPopupComponent implements OnInit, OnChanges {\n  @Input() value: string = '';\n  @Input() helpUrl: string = '';\n  @Input() labels: string[] = [];\n  @Input() element: ElementPopupDataElements = {title: '', selector: '', labels: []};\n  @Input() elements: ElementPopupDataElements[] = [];\n  @Input() enableForm: boolean = false;\n  @Input() isOnRecord: boolean = false;\n\n  @Output() apply = new EventEmitter<string>();\n  @Output() cancel = new EventEmitter<void>();\n  @Output() editInDepth = new EventEmitter<void>();\n  @Output() searchElement = new EventEmitter<void>();\n\n  form!: FormGroup;\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  /** Whether we're in edit mode (true) or create mode (false) */\n  isEditMode: boolean = false;\n\n  private injectedData?: ElementPopupData;\n  private fb: FormBuilder;\n  /** Cached select options to avoid recomputing on every change detection */\n  private cachedSelectOptions: any[] = [];\n  /** Cached select config to avoid recreating on every change detection */\n  private cachedSelectConfig?: DynamicSelectFieldConfig;\n\n  constructor(\n    @Inject(CUSTOM_ELEMENT_POPUP_REF) private ref: ElementPopupRef,\n    @Optional() @Inject(ELEMENT_POPUP_DATA) data?: ElementPopupData,\n    @Optional() fb?: FormBuilder,\n    @Optional() private readonly cdr?: ChangeDetectorRef\n  ) {\n    this.injectedData = data;\n    // Initialize FormBuilder if not injected (for Storybook/testing scenarios)\n    this.fb = fb || new FormBuilder();\n    this.initializeForm();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // Update cached options when labels input changes\n    if (changes['labels'] && !changes['labels'].firstChange) {\n      this.updateSelectOptions();\n    }\n  }\n\n  ngOnInit(): void {\n    // Use injected data if inputs are not provided (for backward compatibility)\n    if (this.injectedData) {\n      if (!this.value) {\n        this.value = this.injectedData.description ?? '';\n      }\n      if (!this.helpUrl) {\n        this.helpUrl = this.injectedData.helpUrl ?? '';\n      }\n      if (this.labels.length === 0) {\n        this.labels = this.injectedData.labels ?? [];\n      }\n      if (this.elements.length === 0) {\n        this.elements = this.injectedData.elements ?? [];\n      }\n      if (!this.element || !this.element.title) {\n        this.element = this.injectedData.element ?? {title: '', selector: '', labels: []};\n      }\n    }\n    // Update select options after labels are potentially set\n    this.updateSelectOptions();\n  }\n\n  private initializeForm(): void {\n    this.form = this.fb.group({\n      name: ['', [Validators.required]],\n      screenName: ['', [Validators.required]],\n      value: ['', [Validators.required]],\n      label: [null]\n    });\n  }\n\n  private populateFormForEdit(): void {\n    if (this.element && this.element.title) {\n      this.isEditMode = true;\n      this.form.patchValue({\n        name: this.element.title || '',\n        screenName: this.element.selector || '',\n        value: this.element.selector || '',\n        label: this.element.labels && this.element.labels.length > 0 ? this.element.labels[0] : null\n      });\n    } else {\n      this.isEditMode = false;\n      this.form.reset();\n    }\n  }\n\n  private resetForm(): void {\n    this.isEditMode = false;\n    this.form.reset({\n      name: '',\n      screenName: '',\n      value: '',\n      label: null\n    });\n  }\n\n  onApply(): void {\n    if (this.form.invalid) {\n      // Mark all fields as touched to show validation errors\n      Object.keys(this.form.controls).forEach(key => {\n        this.form.get(key)?.markAsTouched();\n      });\n      return;\n    }\n\n    const formValue = this.form.value;\n    const elementData: ElementPopupDataElements = {\n      title: formValue.name,\n      selector: formValue.value,\n      labels: formValue.label ? [formValue.label] : []\n    };\n\n    if (this.isEditMode) {\n      // Update existing element\n      this.element = elementData;\n    } else {\n      // Create new element\n      this.elements = [...this.elements, elementData];\n      this.element = elementData;\n    }\n\n    // Emit the selector value (for backward compatibility)\n    this.value = formValue.value;\n    this.apply.emit(formValue.value);\n    this.ref.close(formValue.value);\n    \n    // Close form after applying\n    this.enableForm = false;\n    this.resetForm();\n  }\n\n  toggleForm(): void {\n    if (!this.enableForm) {\n      // Opening form - check if we're editing existing element\n      this.populateFormForEdit();\n    } else {\n      // Closing form - reset\n      this.resetForm();\n    }\n    this.enableForm = !this.enableForm;\n  }\n\n  /** Called when \"Create New\" button is clicked - explicitly sets create mode */\n  openCreateForm(): void {\n    this.isEditMode = false;\n    this.resetForm();\n    this.enableForm = true;\n  }\n\n  toggleRecord():void{\n    this.isOnRecord= !this.isOnRecord;\n  }\n\n  onCancel(): void {\n    // Reset form when canceling\n    if (this.enableForm) {\n      this.resetForm();\n      this.enableForm = false;\n    }\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(ELEMENT_POPUP_EDIT_IN_DEPTH as any);\n  }\n\n  search(event:any){\n    console.log(event)\n    this.searchElement.emit(event);\n  }\n\n  onHelp(event: Event): void {\n    if (this.helpUrl) {\n      event.preventDefault();\n      window.open(this.helpUrl, '_blank');\n    }\n  }\n  onVariableValueChange(variableName: string, value: any): void {\n    console.log(value)\n  }\n\n  /** Update cached select options when labels change */\n  private updateSelectOptions(): void {\n    // Limit options to prevent performance issues - only use unique labels, max 100\n    const uniqueLabels = Array.from(new Set(this.labels || [])).slice(0, 100);\n    this.cachedSelectOptions = uniqueLabels.map(label => ({ \n      id: label, \n      name: label \n    }));\n    // Clear cached config to force recreation\n    this.cachedSelectConfig = undefined;\n  }\n\n  getSelectConfig(): DynamicSelectFieldConfig {\n    // Return cached config if available and labels haven't changed\n    if (this.cachedSelectConfig) {\n      return this.cachedSelectConfig;\n    }\n\n    // Create new config with cached options\n    this.cachedSelectConfig = {\n      key: \"label\",\n      label: \"Label\",\n      placeholder: \"Select label\",\n      options: this.cachedSelectOptions,\n      onChange: (value: any) => {\n        this.form.patchValue({ label: value });\n        this.onVariableValueChange(\"label\", value);\n      }\n    };\n\n    return this.cachedSelectConfig;\n  }\n\n  getFormControl(controlName: string): FormControl {\n    return this.form.get(controlName) as FormControl;\n  }\n\n  getFormControlValue(controlName: string): string {\n    return this.form.get(controlName)?.value || '';\n  }\n\n  onFormControlChange(controlName: string, value: string): void {\n    this.form.patchValue({ [controlName]: value });\n  }\n\n  onElementClick(element: ElementPopupDataElements): void {\n    // Handle element click - you can customize this behavior\n    this.element = element;\n    // Optionally close the form if it's open\n    if (this.enableForm) {\n      this.enableForm = false;\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      Element\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 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  <ng-container *ngIf=\"enableForm && !isOnRecord\">\n      <div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-px-1 cqa-pt-3\">\n        <div class=\"cqa-flex cqa-gap-1.5\">\n          <cqa-custom-input \n            class=\"cqa-w-1/2\" \n            label=\"Name\" \n            placeholder=\"default-element\"\n            [value]=\"getFormControlValue('name')\"\n            [errors]=\"getFormControl('name')?.touched && getFormControl('name')?.invalid ? ['Name is required'] : []\"\n            [required]=\"true\"\n            (valueChange)=\"onFormControlChange('name', $event)\">\n          </cqa-custom-input>\n          <cqa-custom-input \n            class=\"cqa-w-1/2\"  \n            label=\"Screen Name\" \n            placeholder=\"default-screen\"\n            [value]=\"getFormControlValue('screenName')\"\n            [errors]=\"getFormControl('screenName')?.touched && getFormControl('screenName')?.invalid ? ['Screen Name is required'] : []\"\n            [required]=\"true\"\n            (valueChange)=\"onFormControlChange('screenName', $event)\">\n          </cqa-custom-input>\n        </div>\n        <cqa-custom-input \n          class=\"cqa-w-full\"  \n          label=\"Enter Value\" \n          placeholder=\"#default_id\"\n          [value]=\"getFormControlValue('value')\"\n          [errors]=\"getFormControl('value')?.touched && getFormControl('value')?.invalid ? ['Value is required'] : []\"\n          [required]=\"true\"\n          (valueChange)=\"onFormControlChange('value', $event)\">\n        </cqa-custom-input>\n        <div class=\"cqa-w-full cqa-position-relative\">\n          <cqa-dynamic-select [form]=\"form\" [config]=\"getSelectConfig()\"></cqa-dynamic-select>\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)=\"toggleForm()\"\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]=\"isEditMode ? 'Update' : 'Create'\" [fullWidth]=\"true\" (clicked)=\"onApply()\"\n                [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#3F43EE]'\"></cqa-button>\n            </div>\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  </ng-container>\n\n<ng-container *ngIf=\"isOnRecord && !enableForm\">\n  <div\n    class=\"cqa-flex cqa-flex-col cqa-items-center cqa-justify-center cqa-text-center cqa-gap-3 cqa-pt-10\">\n\n    <!-- Video Icon -->\n    <div\n      class=\"cqa-flex cqa-items-center cqa-justify-center cqa-w-[64px] cqa-h-[64px] cqa-rounded-full cqa-bg-[#FEE2E2]\">\n      <svg width=\"32\" height=\"32\" viewBox=\"0 0 32 32\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n        <path d=\"M21.333 17.3333L28.297 21.9759C28.3974 22.0428 28.514 22.0811 28.6345 22.0868C28.7549 22.0926 28.8747 22.0656 28.981 22.0087C29.0873 21.9517 29.1762 21.8671 29.2382 21.7636C29.3002 21.6602 29.3329 21.5419 29.333 21.4213V10.4933C29.333 10.376 29.3021 10.2607 29.2434 10.1592C29.1846 10.0577 29.1001 9.97344 28.9984 9.91501C28.8967 9.85658 28.7814 9.82602 28.6641 9.82642C28.5468 9.82682 28.4317 9.85816 28.3303 9.91728L21.333 13.9999\" stroke=\"#E7000B\" stroke-width=\"2.66667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n        <path d=\"M18.667 8H5.33366C3.8609 8 2.66699 9.19391 2.66699 10.6667V21.3333C2.66699 22.8061 3.8609 24 5.33366 24H18.667C20.1398 24 21.3337 22.8061 21.3337 21.3333V10.6667C21.3337 9.19391 20.1398 8 18.667 8Z\" stroke=\"#E7000B\" stroke-width=\"2.66667\" stroke-linecap=\"round\" stroke-linejoin=\"round\"/>\n        </svg>\n        \n    </div>\n\n    <!-- Title -->\n    <h4 class=\"cqa-m-0 cqa-text-[16px] cqa-leading-[24px] cqa-font-[600] cqa-text-[#0A0A0A]\">\n      Recording Mode Active\n    </h4>\n\n    <!-- Subtitle -->\n    <p class=\"cqa-m-0 cqa-text-[12px] cqa-leading-[18px] cqa-text-[#6B7280]\">\n      Click on any element in the browser to capture it\n    </p>\n  </div>\n\n  <!-- Footer -->\n  <div class=\"cqa-flex cqa-justify-center cqa-pb-4\">\n    <cqa-button\n      variant=\"outlined\"\n      btnSize=\"lg\"\n      [text]=\"'Cancel Recording'\"\n      (clicked)=\"toggleRecord()\"\n      [customClass]=\"'cqa-text-[14px] cqa-px-16 cqa-py-[9px] cqa-border-[#414146]'\">\n    </cqa-button>\n  </div>\n</ng-container>\n\n\n  <ng-container *ngIf=\"!enableForm && !isOnRecord\">\n<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-px-1 cqa-pt-3\">\n\n  <!-- Selected -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n    <span class=\"cqa-text-[12px] cqa-text-[#6B7280]\">Selected</span>\n\n    <div\n      class=\"cqa-flex cqa-items-center cqa-justify-between\n             cqa-bg-[#FFFBEB] cqa-border cqa-border-[#fadfba]\n             cqa-rounded-lg cqa-p-3 cqa-border-solid cqa-gap-3\">\n\n      <div class=\"cqa-flex cqa-px-3 cqa-py-2 cqa-rounded-lg cqa-flex-col cqa-gap-0.5 cqa-bg-[#f5f5f5] cqa-flex-1\">\n        <div class=\"cqa-flex cqa-items-center cqa-justify-between  cqa-gap-2\">\n          <span class=\"cqa-text-[14px] cqa-font-[600] cqa-text-[#111827]\">\n            {{element.title}}\n          </span>\n\n          <div class=\"cqa-flex cqa-gap-2\">\n          <span *ngFor=\"let l of element.labels\"\n            class=\"cqa-text-[10px] cqa-px-1.5 cqa-py-0.5\n                   cqa-rounded cqa-bg-[#EEF2FF] cqa-text-[#3F43EE] cqa-rounded-full cqa-px-2 cqa-bg-[#eff6ff] cqa-border cqa-border-solid cqa-border-[#c8e0ff]\">\n             {{l}}\n          </span>\n          </div>\n        </div>\n\n        <span class=\"cqa-text-[11px] cqa-text-[#6B7280]\">\n          {{element.selector}}\n        </span>\n      </div>\n      <cqa-button variant=\"outlined\" icon=\"edit\" btnSize=\"lg\" [text]=\"'Edit'\" [fullWidth]=\"true\" (clicked)=\"toggleForm()\"\n              [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"></cqa-button>\n    </div>\n  </div>\n\n  <!-- Recent -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n    <span class=\"cqa-text-[12px] cqa-text-[#6B7280]\">Recent</span>\n\n    <div class=\"cqa-flex cqa-gap-2 cqa-overflow-x-auto cqa-scrollbar-thin cqa-scrollbar-track-transparent cqa-scrollbar-thumb-[#E5E7EB] cqa-scrollbar-thumb-rounded-full cqa-scrollbar-thumb-hover:cqa-bg-[#D1D5DB]\">\n\n      <cqa-badge *ngFor=\"let label of labels\"  class=\"cqa-element-badge cqa-mb-2 cqa-chip !cqa-bg-white !cqa-text-[12px] cqa-whitespace-nowrap\" [label]=\"label\"></cqa-badge>\n    </div>\n  </div>\n\n  <!-- Element Library -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n    <span class=\"cqa-text-[12px] cqa-text-[#6B7280]\">Element Library</span>\n    <cqa-search-bar [fullWidth]=\"true\" placeholder=\"Search library\" (valueChange)=\"search($event)\"></cqa-search-bar>        \n  </div>\n\n  <cqa-element-list \n    [items]=\"elements\"\n    [titleKey]=\"'title'\"\n    [selectorKey]=\"'selector'\"\n    [labelsKey]=\"'labels'\"\n    [maxHeight]=\"'200px'\"\n    (itemClick)=\"onElementClick($event)\">\n  </cqa-element-list>\n</div>\n\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\" icon=\"radio_button_checked\" btnSize=\"lg\" [text]=\"'Record'\" [fullWidth]=\"true\" (clicked)=\"toggleRecord()\"\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=\"outlined\" icon=\"add\" btnSize=\"lg\" [text]=\"'Create New'\" [fullWidth]=\"true\" (clicked)=\"openCreateForm()\"\n              [customClass]=\"'cqa-text-[14px] cqa-py-[9px] cqa-border-[#414146]'\"></cqa-button>\n          </div>\n        </div>\n      </div>\n  </ng-container>\n\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 { ElementPopupComponent } from './element-popup.component';
|
|
6
|
+
import { ElementPopupRef } from './element-popup-ref';
|
|
7
|
+
import { ELEMENT_POPUP_DATA, } from './element-popup-data';
|
|
8
|
+
import { CUSTOM_ELEMENT_POPUP_REF } from './element-popup-ref';
|
|
9
|
+
import * as i0 from "@angular/core";
|
|
10
|
+
import * as i1 from "@angular/cdk/overlay";
|
|
11
|
+
export class ElementPopupService {
|
|
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-element-popup-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 ElementPopupRef(overlayRef);
|
|
61
|
+
const injector = Injector.create({
|
|
62
|
+
parent: this.injector,
|
|
63
|
+
providers: [
|
|
64
|
+
{ provide: CUSTOM_ELEMENT_POPUP_REF, useValue: editStepRef },
|
|
65
|
+
{ provide: ELEMENT_POPUP_DATA, useValue: data },
|
|
66
|
+
],
|
|
67
|
+
});
|
|
68
|
+
updatePosition();
|
|
69
|
+
const portal = new ComponentPortal(ElementPopupComponent, 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
|
+
ElementPopupService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ElementPopupService, deps: [{ token: i1.Overlay }, { token: i0.Injector }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
90
|
+
ElementPopupService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ElementPopupService, providedIn: 'root' });
|
|
91
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ElementPopupService, 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":"element-popup.service.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/element-popup/element-popup.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 { ElementPopupComponent } from './element-popup.component';\nimport { ElementPopupRef } from './element-popup-ref';\nimport {\n  ELEMENT_POPUP_DATA,\n  ELEMENT_POPUP_EDIT_IN_DEPTH,\n  ElementPopupData,\n} from './element-popup-data';\nimport { CUSTOM_ELEMENT_POPUP_REF } from './element-popup-ref';\n\nexport type ElementPopupResult =\n  | string\n  | undefined\n  | typeof ELEMENT_POPUP_EDIT_IN_DEPTH;\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class ElementPopupService {\n  /** Currently open Step Description overlay ref; only one panel is allowed at a time. */\n  private currentRef: ElementPopupRef<ElementPopupResult> | 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: ElementPopupData\n  ): ElementPopupRef<ElementPopupResult> {\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-element-popup-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 ElementPopupRef<ElementPopupResult>(overlayRef);\n\n    const injector = Injector.create({\n      parent: this.injector,\n      providers: [\n        { provide: CUSTOM_ELEMENT_POPUP_REF, useValue: editStepRef },\n        { provide: ELEMENT_POPUP_DATA, useValue: data },\n      ],\n    });\n\n    updatePosition();\n\n    const portal = new ComponentPortal(ElementPopupComponent, 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"]}
|