@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 TEST_DATA_MODAL_EDIT_IN_DEPTH = Symbol('TestDataModalEditInDepth');
|
|
4
|
+
export const TEST_DATA_MODAL_DATA = new InjectionToken('TEST_DATA_MODAL_DATA');
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVzdC1kYXRhLW1vZGFsLWRhdGEuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3Rlc3QtY2FzZS1kZXRhaWxzL3Rlc3QtZGF0YS1tb2RhbC90ZXN0LWRhdGEtbW9kYWwtZGF0YS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBK0IvQyw4RUFBOEU7QUFDOUUsTUFBTSxDQUFDLE1BQU0sNkJBQTZCLEdBQUcsTUFBTSxDQUFDLDBCQUEwQixDQUFDLENBQUM7QUFFaEYsTUFBTSxDQUFDLE1BQU0sb0JBQW9CLEdBQUcsSUFBSSxjQUFjLENBQ3BELHNCQUFzQixDQUN2QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0aW9uVG9rZW4gfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB0eXBlIHsgUGFyYW1ldGVyU2NvcGVGaWx0ZXIgfSBmcm9tICcuL3Rlc3QtZGF0YS1tb2RhbC1wYXJhbWV0ZXIubW9kZWwnO1xuXG5leHBvcnQgdHlwZSBUZXN0RGF0YU1vZGFsVGFiID0gJ3BsYWluLXRleHQnIHwgJ3BhcmFtZXRlcicgfCAnZW52aXJvbm1lbnQnIHwgJ3J1bnRpbWUnO1xuXG4vKipcbiAqIENvbmZpZy9kYXRhIGZvciBUZXN0IERhdGEgTW9kYWwgY29udHJvbHMuXG4gKiBNaXJyb3JzIExvb3AgU3RlcCBwYXR0ZXJuOiBzaW5nbGUgY29uZmlnIG9iamVjdCBob2xkcyBhbGwgY29udHJvbCBzdGF0ZTtcbiAqIHZhbHVlcyBhcmUgaW5pdGlhbGl6ZWQgZnJvbSBjb25maWcgYW5kIGJvdW5kIHRvIGVkaXRGb3JtLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFRlc3REYXRhTW9kYWxEYXRhIHtcbiAgLyoqIEluaXRpYWwgdmFsdWUgZm9yIFBsYWluIFRleHQgdGFiIChlLmcuIFVSTCBmb3IgbmF2aWdhdGUgc3RlcCkuICovXG4gIHZhbHVlPzogc3RyaW5nO1xuICAvKiogT3B0aW9uYWwgaGVscCBVUkwgZm9yIFwiTmVlZCBoZWxwP1wiIGxpbmsuICovXG4gIGhlbHBVcmw/OiBzdHJpbmc7XG4gIC8qKiBJbml0aWFsbHkgc2VsZWN0ZWQgdGFiLiBEZWZhdWx0cyB0byAncGxhaW4tdGV4dCcuICovXG4gIGFjdGl2ZVRhYj86IFRlc3REYXRhTW9kYWxUYWI7XG4gIC8qKiBQYXJhbWV0ZXIgdGFiOiBzY29wZSBmaWx0ZXIgKEFsbCAvIEdsb2JhbCAvIExvY2FsKS4gKi9cbiAgcGFyYW1ldGVyU2NvcGVGaWx0ZXI/OiBQYXJhbWV0ZXJTY29wZUZpbHRlcjtcbiAgLyoqIFBhcmFtZXRlciB0YWI6IHNlYXJjaCBxdWVyeS4gKi9cbiAgcGFyYW1ldGVyU2VhcmNoUXVlcnk/OiBzdHJpbmc7XG4gIC8qKiBQYXJhbWV0ZXIgdGFiOiBzZWxlY3RlZCBwYXJhbWV0ZXIgaWQgKGZvciBoaWdobGlnaHQpLiAqL1xuICBzZWxlY3RlZFBhcmFtZXRlcklkPzogc3RyaW5nIHwgbnVsbDtcbiAgLyoqIEVudmlyb25tZW50IHRhYjogc2VhcmNoIHF1ZXJ5LiAqL1xuICBlbnZpcm9ubWVudFNlYXJjaFF1ZXJ5Pzogc3RyaW5nO1xuICAvKiogRW52aXJvbm1lbnQgdGFiOiBzZWxlY3RlZCBlbnZpcm9ubWVudCBpZCAoZm9yIGhpZ2hsaWdodCkuICovXG4gIHNlbGVjdGVkRW52aXJvbm1lbnRJZD86IHN0cmluZyB8IG51bGw7XG4gIC8qKiBSdW50aW1lIHRhYjogdmFsdWUgKGluZGVwZW5kZW50IGZyb20gUGxhaW4gVGV4dCAvIFBhcmFtZXRlciAvIEVudmlyb25tZW50KS4gKi9cbiAgcnVudGltZVZhbHVlPzogc3RyaW5nO1xufVxuXG4vKiogU2VudGluZWwgcmV0dXJuZWQgZnJvbSBhZnRlckNsb3NlZCgpIHdoZW4gdXNlciBjbGlja2VkIFwiRWRpdCBpbiBkZXB0aFwiLiAqL1xuZXhwb3J0IGNvbnN0IFRFU1RfREFUQV9NT0RBTF9FRElUX0lOX0RFUFRIID0gU3ltYm9sKCdUZXN0RGF0YU1vZGFsRWRpdEluRGVwdGgnKTtcblxuZXhwb3J0IGNvbnN0IFRFU1RfREFUQV9NT0RBTF9EQVRBID0gbmV3IEluamVjdGlvblRva2VuPFRlc3REYXRhTW9kYWxEYXRhPihcbiAgJ1RFU1RfREFUQV9NT0RBTF9EQVRBJ1xuKTtcbiJdfQ==
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
/** Mock data for Environment tab (no API). Replace with API call when integrating. */
|
|
2
|
+
export const MOCK_ENVIRONMENT_ITEMS = [
|
|
3
|
+
{ id: 'e1', name: 'BASE_URL', value: 'https://ohrm.example.com', environment: 'OHRM login', editable: true },
|
|
4
|
+
{ id: 'e2', name: 'USERNAME', value: 'admin', environment: 'OHRM login', editable: true },
|
|
5
|
+
{ id: 'e3', name: 'PASSWORD', value: '••••••••', environment: 'OHRM login', editable: true },
|
|
6
|
+
{ id: 'e4', name: 'API_BASE', value: 'https://api.halight.com/v1', environment: 'Halight Env', editable: true },
|
|
7
|
+
{ id: 'e5', name: 'AUTH_TOKEN', value: 'Bearer eyJ...', environment: 'Halight Env', editable: true },
|
|
8
|
+
{ id: 'e6', name: 'TENANT_ID', value: 'tenant-abc-123', environment: 'Halight Env', editable: false },
|
|
9
|
+
];
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVzdC1kYXRhLW1vZGFsLWVudmlyb25tZW50Lm1vZGVsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi90ZXN0LWNhc2UtZGV0YWlscy90ZXN0LWRhdGEtbW9kYWwvdGVzdC1kYXRhLW1vZGFsLWVudmlyb25tZW50Lm1vZGVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQVdBLHNGQUFzRjtBQUN0RixNQUFNLENBQUMsTUFBTSxzQkFBc0IsR0FBc0I7SUFDdkQsRUFBRSxFQUFFLEVBQUUsSUFBSSxFQUFFLElBQUksRUFBRSxVQUFVLEVBQUUsS0FBSyxFQUFFLDBCQUEwQixFQUFFLFdBQVcsRUFBRSxZQUFZLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRTtJQUM1RyxFQUFFLEVBQUUsRUFBRSxJQUFJLEVBQUUsSUFBSSxFQUFFLFVBQVUsRUFBRSxLQUFLLEVBQUUsT0FBTyxFQUFFLFdBQVcsRUFBRSxZQUFZLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRTtJQUN6RixFQUFFLEVBQUUsRUFBRSxJQUFJLEVBQUUsSUFBSSxFQUFFLFVBQVUsRUFBRSxLQUFLLEVBQUUsVUFBVSxFQUFFLFdBQVcsRUFBRSxZQUFZLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRTtJQUM1RixFQUFFLEVBQUUsRUFBRSxJQUFJLEVBQUUsSUFBSSxFQUFFLFVBQVUsRUFBRSxLQUFLLEVBQUUsNEJBQTRCLEVBQUUsV0FBVyxFQUFFLGFBQWEsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFO0lBQy9HLEVBQUUsRUFBRSxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxlQUFlLEVBQUUsV0FBVyxFQUFFLGFBQWEsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFO0lBQ3BHLEVBQUUsRUFBRSxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsV0FBVyxFQUFFLEtBQUssRUFBRSxnQkFBZ0IsRUFBRSxXQUFXLEVBQUUsYUFBYSxFQUFFLFFBQVEsRUFBRSxLQUFLLEVBQUU7Q0FDdEcsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKiBFbnZpcm9ubWVudCB2YXJpYWJsZSBpdGVtIGZvciB0aGUgRW52aXJvbm1lbnQgdGFiICh2YXJpYWJsZSBsaWJyYXJ5IGJ5IGVudmlyb25tZW50KS4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgRW52aXJvbm1lbnRJdGVtIHtcbiAgaWQ6IHN0cmluZztcbiAgbmFtZTogc3RyaW5nO1xuICB2YWx1ZTogc3RyaW5nO1xuICAvKiogRGlzcGxheSBuYW1lIG9mIHRoZSBlbnZpcm9ubWVudCBncm91cCAoZS5nLiBcIk9IUk0gbG9naW5cIiwgXCJIYWxpZ2h0IEVudlwiKS4gKi9cbiAgZW52aXJvbm1lbnQ6IHN0cmluZztcbiAgLyoqIFdoZW4gdHJ1ZSwgc2hvdyBlZGl0IGljb24gd2hlbiBzZWxlY3RlZC4gKi9cbiAgZWRpdGFibGU/OiBib29sZWFuO1xufVxuXG4vKiogTW9jayBkYXRhIGZvciBFbnZpcm9ubWVudCB0YWIgKG5vIEFQSSkuIFJlcGxhY2Ugd2l0aCBBUEkgY2FsbCB3aGVuIGludGVncmF0aW5nLiAqL1xuZXhwb3J0IGNvbnN0IE1PQ0tfRU5WSVJPTk1FTlRfSVRFTVM6IEVudmlyb25tZW50SXRlbVtdID0gW1xuICB7IGlkOiAnZTEnLCBuYW1lOiAnQkFTRV9VUkwnLCB2YWx1ZTogJ2h0dHBzOi8vb2hybS5leGFtcGxlLmNvbScsIGVudmlyb25tZW50OiAnT0hSTSBsb2dpbicsIGVkaXRhYmxlOiB0cnVlIH0sXG4gIHsgaWQ6ICdlMicsIG5hbWU6ICdVU0VSTkFNRScsIHZhbHVlOiAnYWRtaW4nLCBlbnZpcm9ubWVudDogJ09IUk0gbG9naW4nLCBlZGl0YWJsZTogdHJ1ZSB9LFxuICB7IGlkOiAnZTMnLCBuYW1lOiAnUEFTU1dPUkQnLCB2YWx1ZTogJ+KAouKAouKAouKAouKAouKAouKAouKAoicsIGVudmlyb25tZW50OiAnT0hSTSBsb2dpbicsIGVkaXRhYmxlOiB0cnVlIH0sXG4gIHsgaWQ6ICdlNCcsIG5hbWU6ICdBUElfQkFTRScsIHZhbHVlOiAnaHR0cHM6Ly9hcGkuaGFsaWdodC5jb20vdjEnLCBlbnZpcm9ubWVudDogJ0hhbGlnaHQgRW52JywgZWRpdGFibGU6IHRydWUgfSxcbiAgeyBpZDogJ2U1JywgbmFtZTogJ0FVVEhfVE9LRU4nLCB2YWx1ZTogJ0JlYXJlciBleUouLi4nLCBlbnZpcm9ubWVudDogJ0hhbGlnaHQgRW52JywgZWRpdGFibGU6IHRydWUgfSxcbiAgeyBpZDogJ2U2JywgbmFtZTogJ1RFTkFOVF9JRCcsIHZhbHVlOiAndGVuYW50LWFiYy0xMjMnLCBlbnZpcm9ubWVudDogJ0hhbGlnaHQgRW52JywgZWRpdGFibGU6IGZhbHNlIH0sXG5dO1xuIl19
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
/** Mock data for Variable Library (no API). Replace with API call when integrating. */
|
|
2
|
+
export const MOCK_PARAMETERS = [
|
|
3
|
+
{ id: '1', name: 'user_email', value: 'test@example.com', scope: 'Local' },
|
|
4
|
+
{ id: '2', name: 'API_BASE_URL', value: 'https://api.example.com', scope: 'Global' },
|
|
5
|
+
{ id: '3', name: 'auth_token', value: 'eyJhbGciOiJIUzI1...', scope: 'Global' },
|
|
6
|
+
{ id: '4', name: 'session_id', value: 'abc123xvz456...', scope: 'Local' },
|
|
7
|
+
];
|
|
8
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVzdC1kYXRhLW1vZGFsLXBhcmFtZXRlci5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3NyYy9saWIvdGVzdC1jYXNlLWRldGFpbHMvdGVzdC1kYXRhLW1vZGFsL3Rlc3QtZGF0YS1tb2RhbC1wYXJhbWV0ZXIubW9kZWwudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBVUEsdUZBQXVGO0FBQ3ZGLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBb0I7SUFDOUMsRUFBRSxFQUFFLEVBQUUsR0FBRyxFQUFFLElBQUksRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLGtCQUFrQixFQUFFLEtBQUssRUFBRSxPQUFPLEVBQUU7SUFDMUUsRUFBRSxFQUFFLEVBQUUsR0FBRyxFQUFFLElBQUksRUFBRSxjQUFjLEVBQUUsS0FBSyxFQUFFLHlCQUF5QixFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUU7SUFDcEYsRUFBRSxFQUFFLEVBQUUsR0FBRyxFQUFFLElBQUksRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLHFCQUFxQixFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUU7SUFDOUUsRUFBRSxFQUFFLEVBQUUsR0FBRyxFQUFFLElBQUksRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLGlCQUFpQixFQUFFLEtBQUssRUFBRSxPQUFPLEVBQUU7Q0FDMUUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKiBTY29wZSBvZiBhIHZhcmlhYmxlL3BhcmFtZXRlciBpbiB0aGUgbGlicmFyeS4gKi9cbmV4cG9ydCB0eXBlIFBhcmFtZXRlclNjb3BlID0gJ0dsb2JhbCcgfCAnTG9jYWwnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFBhcmFtZXRlckl0ZW0ge1xuICBpZDogc3RyaW5nO1xuICBuYW1lOiBzdHJpbmc7XG4gIHZhbHVlOiBzdHJpbmc7XG4gIHNjb3BlOiBQYXJhbWV0ZXJTY29wZTtcbn1cblxuLyoqIE1vY2sgZGF0YSBmb3IgVmFyaWFibGUgTGlicmFyeSAobm8gQVBJKS4gUmVwbGFjZSB3aXRoIEFQSSBjYWxsIHdoZW4gaW50ZWdyYXRpbmcuICovXG5leHBvcnQgY29uc3QgTU9DS19QQVJBTUVURVJTOiBQYXJhbWV0ZXJJdGVtW10gPSBbXG4gIHsgaWQ6ICcxJywgbmFtZTogJ3VzZXJfZW1haWwnLCB2YWx1ZTogJ3Rlc3RAZXhhbXBsZS5jb20nLCBzY29wZTogJ0xvY2FsJyB9LFxuICB7IGlkOiAnMicsIG5hbWU6ICdBUElfQkFTRV9VUkwnLCB2YWx1ZTogJ2h0dHBzOi8vYXBpLmV4YW1wbGUuY29tJywgc2NvcGU6ICdHbG9iYWwnIH0sXG4gIHsgaWQ6ICczJywgbmFtZTogJ2F1dGhfdG9rZW4nLCB2YWx1ZTogJ2V5SmhiR2NpT2lKSVV6STEuLi4nLCBzY29wZTogJ0dsb2JhbCcgfSxcbiAgeyBpZDogJzQnLCBuYW1lOiAnc2Vzc2lvbl9pZCcsIHZhbHVlOiAnYWJjMTIzeHZ6NDU2Li4uJywgc2NvcGU6ICdMb2NhbCcgfSxcbl07XG5cbmV4cG9ydCB0eXBlIFBhcmFtZXRlclNjb3BlRmlsdGVyID0gJ2FsbCcgfCAnZ2xvYmFsJyB8ICdsb2NhbCc7XG4iXX0=
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { InjectionToken } from '@angular/core';
|
|
2
|
+
import { Subject } from 'rxjs';
|
|
3
|
+
export class TestDataModalRef {
|
|
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 TEST_DATA_MODAL_REF = new InjectionToken('TEST_DATA_MODAL_REF');
|
|
32
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVzdC1kYXRhLW1vZGFsLXJlZi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3NyYy9saWIvdGVzdC1jYXNlLWRldGFpbHMvdGVzdC1kYXRhLW1vZGFsL3Rlc3QtZGF0YS1tb2RhbC1yZWYudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUUvQyxPQUFPLEVBQWMsT0FBTyxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBRTNDLE1BQU0sT0FBTyxnQkFBZ0I7SUFJM0IsWUFBNkIsVUFBc0I7UUFBdEIsZUFBVSxHQUFWLFVBQVUsQ0FBWTtRQUhsQyxZQUFPLEdBQUcsSUFBSSxPQUFPLEVBQXVCLENBQUM7UUFDdEQsYUFBUSxHQUFHLEtBQUssQ0FBQztRQUd2QixJQUFJLENBQUMsVUFBVSxDQUFDLFdBQVcsRUFBRSxDQUFDLFNBQVMsQ0FBQyxHQUFHLEVBQUU7WUFDM0MsSUFBSSxDQUFDLFdBQVcsQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUM5QixDQUFDLENBQUMsQ0FBQztJQUNMLENBQUM7SUFFRCxLQUFLLENBQUMsTUFBZ0I7UUFDcEIsSUFBSSxJQUFJLENBQUMsUUFBUSxFQUFFO1lBQ2pCLE9BQU87U0FDUjtRQUNELElBQUksQ0FBQyxXQUFXLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDekIsSUFBSSxDQUFDLFVBQVUsQ0FBQyxPQUFPLEVBQUUsQ0FBQztJQUM1QixDQUFDO0lBRUQsV0FBVztRQUNULE9BQU8sSUFBSSxDQUFDLE9BQU8sQ0FBQyxZQUFZLEVBQUUsQ0FBQztJQUNyQyxDQUFDO0lBRU8sV0FBVyxDQUFDLE1BQTJCO1FBQzdDLElBQUksSUFBSSxDQUFDLFFBQVEsRUFBRTtZQUNqQixPQUFPO1NBQ1I7UUFDRCxJQUFJLENBQUMsUUFBUSxHQUFHLElBQUksQ0FBQztRQUNyQixJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUMxQixJQUFJLENBQUMsT0FBTyxDQUFDLFFBQVEsRUFBRSxDQUFDO0lBQzFCLENBQUM7Q0FDRjtBQUVELE1BQU0sQ0FBQyxNQUFNLG1CQUFtQixHQUFHLElBQUksY0FBYyxDQUNuRCxxQkFBcUIsQ0FDdEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGlvblRva2VuIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBPdmVybGF5UmVmIH0gZnJvbSAnQGFuZ3VsYXIvY2RrL292ZXJsYXknO1xuaW1wb3J0IHsgT2JzZXJ2YWJsZSwgU3ViamVjdCB9IGZyb20gJ3J4anMnO1xuXG5leHBvcnQgY2xhc3MgVGVzdERhdGFNb2RhbFJlZjxUUmVzdWx0ID0gc3RyaW5nIHwgdW5kZWZpbmVkPiB7XG4gIHByaXZhdGUgcmVhZG9ubHkgY2xvc2VkJCA9IG5ldyBTdWJqZWN0PFRSZXN1bHQgfCB1bmRlZmluZWQ+KCk7XG4gIHByaXZhdGUgaXNDbG9zZWQgPSBmYWxzZTtcblxuICBjb25zdHJ1Y3Rvcihwcml2YXRlIHJlYWRvbmx5IG92ZXJsYXlSZWY6IE92ZXJsYXlSZWYpIHtcbiAgICB0aGlzLm92ZXJsYXlSZWYuZGV0YWNobWVudHMoKS5zdWJzY3JpYmUoKCkgPT4ge1xuICAgICAgdGhpcy5maW5pc2hDbG9zZSh1bmRlZmluZWQpO1xuICAgIH0pO1xuICB9XG5cbiAgY2xvc2UocmVzdWx0PzogVFJlc3VsdCk6IHZvaWQge1xuICAgIGlmICh0aGlzLmlzQ2xvc2VkKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIHRoaXMuZmluaXNoQ2xvc2UocmVzdWx0KTtcbiAgICB0aGlzLm92ZXJsYXlSZWYuZGlzcG9zZSgpO1xuICB9XG5cbiAgYWZ0ZXJDbG9zZWQoKTogT2JzZXJ2YWJsZTxUUmVzdWx0IHwgdW5kZWZpbmVkPiB7XG4gICAgcmV0dXJuIHRoaXMuY2xvc2VkJC5hc09ic2VydmFibGUoKTtcbiAgfVxuXG4gIHByaXZhdGUgZmluaXNoQ2xvc2UocmVzdWx0OiBUUmVzdWx0IHwgdW5kZWZpbmVkKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuaXNDbG9zZWQpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhpcy5pc0Nsb3NlZCA9IHRydWU7XG4gICAgdGhpcy5jbG9zZWQkLm5leHQocmVzdWx0KTtcbiAgICB0aGlzLmNsb3NlZCQuY29tcGxldGUoKTtcbiAgfVxufVxuXG5leHBvcnQgY29uc3QgVEVTVF9EQVRBX01PREFMX1JFRiA9IG5ldyBJbmplY3Rpb25Ub2tlbjxUZXN0RGF0YU1vZGFsUmVmPihcbiAgJ1RFU1RfREFUQV9NT0RBTF9SRUYnXG4pO1xuIl19
|
|
@@ -0,0 +1,301 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Inject, Optional, Output, } from '@angular/core';
|
|
2
|
+
import { TEST_DATA_MODAL_DATA, TEST_DATA_MODAL_EDIT_IN_DEPTH, } from './test-data-modal-data';
|
|
3
|
+
import { TEST_DATA_MODAL_REF } from './test-data-modal-ref';
|
|
4
|
+
import { MOCK_ENVIRONMENT_ITEMS, } from './test-data-modal-environment.model';
|
|
5
|
+
import { MOCK_PARAMETERS, } from './test-data-modal-parameter.model';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/forms";
|
|
8
|
+
import * as i2 from "@angular/material/icon";
|
|
9
|
+
import * as i3 from "../../segment-control/segment-control.component";
|
|
10
|
+
import * as i4 from "../../custom-input/custom-input.component";
|
|
11
|
+
import * as i5 from "../../item-list/item-list.component";
|
|
12
|
+
import * as i6 from "../../button/button.component";
|
|
13
|
+
import * as i7 from "@angular/common";
|
|
14
|
+
import * as i8 from "./test-data-modal-ref";
|
|
15
|
+
const TAB_SEGMENTS = [
|
|
16
|
+
{ label: 'Plain Text', value: 'plain-text' },
|
|
17
|
+
{ label: 'Parameter', value: 'parameter' },
|
|
18
|
+
{ label: 'Environment', value: 'environment' },
|
|
19
|
+
{ label: 'Runtime', value: 'runtime' },
|
|
20
|
+
];
|
|
21
|
+
const PARAMETER_FILTER_TABS = [
|
|
22
|
+
{ label: 'All', value: 'all' },
|
|
23
|
+
{ label: 'Global', value: 'global' },
|
|
24
|
+
{ label: 'Local', value: 'local' },
|
|
25
|
+
];
|
|
26
|
+
/** Mock runtime variable suggestions (replace with API when integrating). */
|
|
27
|
+
const RUNTIME_SUGGESTED_VALUES = [
|
|
28
|
+
'{{runtime.MaxWaitTime}}',
|
|
29
|
+
'{{runtime.session_id}}',
|
|
30
|
+
'{{runtime.auth_token}}',
|
|
31
|
+
'{{runtime.step_index}}',
|
|
32
|
+
'{{runtime.timestamp}}',
|
|
33
|
+
'{{runtime.test_name}}',
|
|
34
|
+
];
|
|
35
|
+
export class TestDataModalComponent {
|
|
36
|
+
constructor(fb, cdr, ref, data) {
|
|
37
|
+
this.fb = fb;
|
|
38
|
+
this.cdr = cdr;
|
|
39
|
+
this.ref = ref;
|
|
40
|
+
this.apply = new EventEmitter();
|
|
41
|
+
this.cancel = new EventEmitter();
|
|
42
|
+
this.editInDepth = new EventEmitter();
|
|
43
|
+
this.tabSegments = TAB_SEGMENTS;
|
|
44
|
+
this.parameterFilterTabs = PARAMETER_FILTER_TABS;
|
|
45
|
+
/** Mock parameter list (replace with API when integrating). */
|
|
46
|
+
this.parameters = MOCK_PARAMETERS;
|
|
47
|
+
/** Mock environment list (replace with API when integrating). */
|
|
48
|
+
this.environmentItems = MOCK_ENVIRONMENT_ITEMS;
|
|
49
|
+
/** Runtime tab: suggested variable chips. */
|
|
50
|
+
this.runtimeSuggestedValues = RUNTIME_SUGGESTED_VALUES;
|
|
51
|
+
/** Local copy of form value for Cancel revert (same pattern as Loop Step editSnapshot). */
|
|
52
|
+
this.editSnapshot = {};
|
|
53
|
+
/** State derived from config/editForm for template and getters (mirrors Loop Step inputs). */
|
|
54
|
+
this.activeTab = 'plain-text';
|
|
55
|
+
this.value = '';
|
|
56
|
+
this.helpUrl = '';
|
|
57
|
+
this.helpTooltipText = 'Not sure what to do? Click to go to our detailed step creation documentation';
|
|
58
|
+
this.showHelpTooltip = false;
|
|
59
|
+
this.parameterScopeFilter = 'all';
|
|
60
|
+
this.parameterSearchQuery = '';
|
|
61
|
+
this.selectedParameterId = null;
|
|
62
|
+
this.environmentSearchQuery = '';
|
|
63
|
+
this.selectedEnvironmentId = null;
|
|
64
|
+
this.runtimeValue = '';
|
|
65
|
+
if (data) {
|
|
66
|
+
this.value = data.value ?? '';
|
|
67
|
+
this.helpUrl = data.helpUrl ?? '';
|
|
68
|
+
this.activeTab = (data.activeTab ?? 'plain-text');
|
|
69
|
+
this.parameterScopeFilter = data.parameterScopeFilter ?? 'all';
|
|
70
|
+
this.parameterSearchQuery = data.parameterSearchQuery ?? '';
|
|
71
|
+
this.selectedParameterId = data.selectedParameterId ?? null;
|
|
72
|
+
this.environmentSearchQuery = data.environmentSearchQuery ?? '';
|
|
73
|
+
this.selectedEnvironmentId = data.selectedEnvironmentId ?? null;
|
|
74
|
+
this.runtimeValue = data.runtimeValue ?? '';
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
ngOnInit() {
|
|
78
|
+
this.buildEditForm();
|
|
79
|
+
this.buildSelectConfigs();
|
|
80
|
+
this.editSnapshot = { ...this.editForm.value };
|
|
81
|
+
this.cdr.markForCheck();
|
|
82
|
+
}
|
|
83
|
+
/** Build form with all control values (same pattern as Loop Step buildEditForm). */
|
|
84
|
+
buildEditForm() {
|
|
85
|
+
this.editForm = this.fb.group({
|
|
86
|
+
value: [this.value ?? ''],
|
|
87
|
+
activeTab: [this.activeTab ?? 'plain-text'],
|
|
88
|
+
parameterScopeFilter: [this.parameterScopeFilter ?? 'all'],
|
|
89
|
+
parameterSearchQuery: [this.parameterSearchQuery ?? ''],
|
|
90
|
+
selectedParameterId: [this.selectedParameterId ?? null],
|
|
91
|
+
environmentSearchQuery: [this.environmentSearchQuery ?? ''],
|
|
92
|
+
selectedEnvironmentId: [this.selectedEnvironmentId ?? null],
|
|
93
|
+
runtimeValue: [this.runtimeValue ?? ''],
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
/** Build select configs once (same pattern as Loop Step buildSelectConfigs). */
|
|
97
|
+
buildSelectConfigs() {
|
|
98
|
+
const tabOptions = TAB_SEGMENTS.map((s) => ({
|
|
99
|
+
id: s.value,
|
|
100
|
+
value: s.value,
|
|
101
|
+
name: s.label,
|
|
102
|
+
label: s.label,
|
|
103
|
+
}));
|
|
104
|
+
const filterOptions = PARAMETER_FILTER_TABS.map((s) => ({
|
|
105
|
+
id: s.value,
|
|
106
|
+
value: s.value,
|
|
107
|
+
name: s.label,
|
|
108
|
+
label: s.label,
|
|
109
|
+
}));
|
|
110
|
+
this.activeTabSelectConfig = {
|
|
111
|
+
key: 'activeTab',
|
|
112
|
+
placeholder: 'Tab',
|
|
113
|
+
searchable: false,
|
|
114
|
+
options: tabOptions,
|
|
115
|
+
};
|
|
116
|
+
this.parameterScopeFilterSelectConfig = {
|
|
117
|
+
key: 'parameterScopeFilter',
|
|
118
|
+
placeholder: 'Scope',
|
|
119
|
+
searchable: false,
|
|
120
|
+
options: filterOptions,
|
|
121
|
+
};
|
|
122
|
+
}
|
|
123
|
+
/** Update form control and sync component state (same pattern as Loop Step onEditFormFieldChange). */
|
|
124
|
+
onEditFormFieldChange(controlName, value) {
|
|
125
|
+
this.editForm.get(controlName)?.setValue(value);
|
|
126
|
+
this.syncStateFromForm();
|
|
127
|
+
this.cdr.markForCheck();
|
|
128
|
+
}
|
|
129
|
+
/** Sync component state from editForm so getters (e.g. filteredParameters) stay correct. */
|
|
130
|
+
syncStateFromForm() {
|
|
131
|
+
const v = this.editForm.value;
|
|
132
|
+
this.value = v.value ?? '';
|
|
133
|
+
this.activeTab = (v.activeTab ?? 'plain-text');
|
|
134
|
+
this.parameterScopeFilter = (v.parameterScopeFilter ?? 'all');
|
|
135
|
+
this.parameterSearchQuery = v.parameterSearchQuery ?? '';
|
|
136
|
+
this.selectedParameterId = v.selectedParameterId ?? null;
|
|
137
|
+
this.environmentSearchQuery = v.environmentSearchQuery ?? '';
|
|
138
|
+
this.selectedEnvironmentId = v.selectedEnvironmentId ?? null;
|
|
139
|
+
this.runtimeValue = v.runtimeValue ?? '';
|
|
140
|
+
}
|
|
141
|
+
/** Filtered parameters by scope and search (Parameter tab). */
|
|
142
|
+
get filteredParameters() {
|
|
143
|
+
let list = this.parameters;
|
|
144
|
+
if (this.parameterScopeFilter === 'global') {
|
|
145
|
+
list = list.filter((p) => p.scope === 'Global');
|
|
146
|
+
}
|
|
147
|
+
else if (this.parameterScopeFilter === 'local') {
|
|
148
|
+
list = list.filter((p) => p.scope === 'Local');
|
|
149
|
+
}
|
|
150
|
+
const q = this.parameterSearchQuery.trim().toLowerCase();
|
|
151
|
+
if (q) {
|
|
152
|
+
list = list.filter((p) => p.name.toLowerCase().includes(q));
|
|
153
|
+
}
|
|
154
|
+
return list;
|
|
155
|
+
}
|
|
156
|
+
/** Filtered environment items by search (Environment tab). */
|
|
157
|
+
get filteredEnvironments() {
|
|
158
|
+
const q = this.environmentSearchQuery.trim().toLowerCase();
|
|
159
|
+
if (!q) {
|
|
160
|
+
return this.environmentItems;
|
|
161
|
+
}
|
|
162
|
+
return this.environmentItems.filter((e) => e.name.toLowerCase().includes(q) || e.value.toLowerCase().includes(q));
|
|
163
|
+
}
|
|
164
|
+
/** Parameter tab: list config for cqa-item-list (search-friendly: bind filtered list). */
|
|
165
|
+
get parameterListConfig() {
|
|
166
|
+
return this.filteredParameters.map((p) => ({
|
|
167
|
+
id: p.id,
|
|
168
|
+
title: p.name,
|
|
169
|
+
subtitle: p.value,
|
|
170
|
+
badge: p.scope,
|
|
171
|
+
showEdit: this.selectedParameterId === p.id,
|
|
172
|
+
}));
|
|
173
|
+
}
|
|
174
|
+
/** Environment tab: list config for cqa-item-list. */
|
|
175
|
+
get environmentListConfig() {
|
|
176
|
+
return this.filteredEnvironments.map((e) => ({
|
|
177
|
+
id: e.id,
|
|
178
|
+
title: e.name,
|
|
179
|
+
subtitle: `Value: ${e.value}`,
|
|
180
|
+
badge: e.environment,
|
|
181
|
+
showEdit: this.selectedEnvironmentId === e.id && (e.editable ?? true),
|
|
182
|
+
}));
|
|
183
|
+
}
|
|
184
|
+
onTabChange(tabValue) {
|
|
185
|
+
this.onEditFormFieldChange('activeTab', tabValue);
|
|
186
|
+
}
|
|
187
|
+
onParameterScopeFilterChange(filter) {
|
|
188
|
+
this.onEditFormFieldChange('parameterScopeFilter', filter);
|
|
189
|
+
}
|
|
190
|
+
onParameterSearchInput(value) {
|
|
191
|
+
this.onEditFormFieldChange('parameterSearchQuery', value);
|
|
192
|
+
}
|
|
193
|
+
onParameterSelect(paramId) {
|
|
194
|
+
const param = this.parameters.find((p) => p.id === paramId);
|
|
195
|
+
if (param) {
|
|
196
|
+
this.selectedParameterId = this.selectedParameterId === param.id ? null : param.id;
|
|
197
|
+
this.onEditFormFieldChange('selectedParameterId', this.selectedParameterId);
|
|
198
|
+
this.onEditFormFieldChange('value', param.name);
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
onParameterEdit(_paramId) {
|
|
202
|
+
// Placeholder for future edit action (e.g. open edit drawer).
|
|
203
|
+
}
|
|
204
|
+
onCreateNewVariable(event) {
|
|
205
|
+
event.preventDefault();
|
|
206
|
+
// Placeholder for future "Create New" action (no API yet).
|
|
207
|
+
}
|
|
208
|
+
onEnvironmentSearchInput(value) {
|
|
209
|
+
this.onEditFormFieldChange('environmentSearchQuery', value);
|
|
210
|
+
}
|
|
211
|
+
onEnvironmentSelect(itemId) {
|
|
212
|
+
const item = this.environmentItems.find((e) => e.id === itemId);
|
|
213
|
+
if (item) {
|
|
214
|
+
this.selectedEnvironmentId =
|
|
215
|
+
this.selectedEnvironmentId === item.id ? null : item.id;
|
|
216
|
+
this.onEditFormFieldChange('selectedEnvironmentId', this.selectedEnvironmentId);
|
|
217
|
+
this.onEditFormFieldChange('value', item.name);
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
onEnvironmentEdit(_itemId) {
|
|
221
|
+
// Placeholder for future edit action (e.g. open edit drawer).
|
|
222
|
+
}
|
|
223
|
+
onCreateNewEnvironment(event) {
|
|
224
|
+
event.preventDefault();
|
|
225
|
+
// Placeholder for future "Create New" action (no API yet).
|
|
226
|
+
}
|
|
227
|
+
/** Runtime tab: insert suggested value into the runtime input field. */
|
|
228
|
+
onRuntimeChipInsert(snippet) {
|
|
229
|
+
const next = this.editForm.get('runtimeValue')?.value
|
|
230
|
+
? this.editForm.get('runtimeValue')?.value + ' ' + snippet
|
|
231
|
+
: snippet;
|
|
232
|
+
this.onEditFormFieldChange('runtimeValue', next);
|
|
233
|
+
}
|
|
234
|
+
onApply() {
|
|
235
|
+
this.syncStateFromForm();
|
|
236
|
+
let valueToApply = this.activeTab === 'runtime' ? this.runtimeValue : this.value;
|
|
237
|
+
if (this.activeTab === 'parameter') {
|
|
238
|
+
valueToApply = this.selectedParameterId ? this.parameters.find((p) => p.id === this.selectedParameterId)?.name ?? '' : '';
|
|
239
|
+
}
|
|
240
|
+
else if (this.activeTab === 'environment') {
|
|
241
|
+
valueToApply = this.selectedEnvironmentId ? this.environmentItems.find((e) => e.id === this.selectedEnvironmentId)?.name ?? '' : '';
|
|
242
|
+
}
|
|
243
|
+
// Build result with only the active tab's data, clear other tabs' data
|
|
244
|
+
const result = {
|
|
245
|
+
value: valueToApply,
|
|
246
|
+
activeTab: this.activeTab,
|
|
247
|
+
// Only include parameter data if parameter tab is active
|
|
248
|
+
parameterScopeFilter: this.activeTab === 'parameter' ? this.parameterScopeFilter : undefined,
|
|
249
|
+
parameterSearchQuery: this.activeTab === 'parameter' ? this.parameterSearchQuery : undefined,
|
|
250
|
+
selectedParameterId: this.activeTab === 'parameter' ? this.selectedParameterId : undefined,
|
|
251
|
+
// Only include environment data if environment tab is active
|
|
252
|
+
environmentSearchQuery: this.activeTab === 'environment' ? this.environmentSearchQuery : undefined,
|
|
253
|
+
selectedEnvironmentId: this.activeTab === 'environment' ? this.selectedEnvironmentId : undefined,
|
|
254
|
+
// Only include runtime data if runtime tab is active
|
|
255
|
+
runtimeValue: this.activeTab === 'runtime' ? this.runtimeValue : undefined,
|
|
256
|
+
};
|
|
257
|
+
this.apply.emit({ value: valueToApply, activeTab: this.activeTab });
|
|
258
|
+
this.ref.close(result);
|
|
259
|
+
}
|
|
260
|
+
onCancel() {
|
|
261
|
+
this.editForm.patchValue(this.editSnapshot);
|
|
262
|
+
this.syncStateFromForm();
|
|
263
|
+
this.cancel.emit();
|
|
264
|
+
this.ref.close(undefined);
|
|
265
|
+
}
|
|
266
|
+
onClose() {
|
|
267
|
+
this.onCancel();
|
|
268
|
+
}
|
|
269
|
+
onEditInDepth(event) {
|
|
270
|
+
event.preventDefault();
|
|
271
|
+
this.editInDepth.emit();
|
|
272
|
+
this.ref.close(TEST_DATA_MODAL_EDIT_IN_DEPTH);
|
|
273
|
+
}
|
|
274
|
+
onHelp(event) {
|
|
275
|
+
if (this.helpUrl) {
|
|
276
|
+
event.preventDefault();
|
|
277
|
+
window.open(this.helpUrl, '_blank');
|
|
278
|
+
}
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
TestDataModalComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TestDataModalComponent, deps: [{ token: i1.FormBuilder }, { token: i0.ChangeDetectorRef }, { token: TEST_DATA_MODAL_REF }, { token: TEST_DATA_MODAL_DATA, optional: true }], target: i0.ɵɵFactoryTarget.Component });
|
|
282
|
+
TestDataModalComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: TestDataModalComponent, selector: "cqa-test-data-modal", 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 cqa-min-h-0\">\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 Test Data\n </h2>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-min-h-[28px]\">\n <!-- Need help? with custom tooltip (works inside overlay) -->\n <div class=\"cqa-relative cqa-inline-flex cqa-items-center\"\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(#test-data-help-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=\"test-data-help-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(#test-data-help-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=\"test-data-help-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\"\n class=\"cqa-absolute cqa-pointer-events-none cqa-z-[100] cqa-top-[-24px] cqa-left-[-125px]\"\n role=\"tooltip\">\n <div\n class=\"cqa-text-white cqa-text-center cqa-whitespace-nowrap cqa-w-[306px] cqa-min-h-[20px] cqa-rounded-[6px] cqa-py-1 cqa-px-2 cqa-bg-[#0A0A0A] cqa-leading-[20px] cqa-text-[8px]\">\n {{ helpTooltipText }}\n </div>\n </div>\n </div>\n <button type=\"button\" (click)=\"onClose()\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-[28px] cqa-min-w-[28px] cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0\"\n title=\"Close\" aria-label=\"Close\">\n <mat-icon class=\"!cqa-w-5 !cqa-h-5 !cqa-text-[20px] !cqa-block !cqa-leading-none\">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 <!-- Tabs: Plain Text, Parameter, Environment, Runtime (bound to editForm like Loop Step) -->\n <div class=\"cqa-w-full cqa-test-data-modal-tabs\">\n <cqa-segment-control\n [segments]=\"tabSegments\"\n [value]=\"editForm.get('activeTab')?.value ?? activeTab\"\n (valueChange)=\"onTabChange($event)\">\n </cqa-segment-control>\n </div>\n\n <!-- Content per tab: flex-1 min-h-0 so Parameter/Environment/Runtime content can scroll -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-min-h-0\"\n [class.cqa-flex-1]=\"activeTab === 'parameter' || activeTab === 'environment' || activeTab === 'runtime'\">\n <!-- Plain Text tab (form-bound like Loop Step editForm + onEditFormFieldChange) -->\n <ng-container *ngIf=\"activeTab === 'plain-text'\">\n <cqa-custom-input\n label=\"Value\"\n [value]=\"editForm.get('value')?.value ?? ''\"\n (valueChange)=\"onEditFormFieldChange('value', $event)\"\n placeholder=\"https://example.com/dashboard\"\n [fullWidth]=\"true\"\n size=\"md\">\n </cqa-custom-input>\n <p class=\"cqa-text-[12px] cqa-leading-[18px] cqa-text-[#0A0A0A] cqa-m-0\">\n Enter a static text value that won't change during execution.\n </p>\n </ng-container>\n\n <!-- Parameter tab: Variable Library -->\n <ng-container *ngIf=\"activeTab === 'parameter'\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-min-h-0 cqa-flex-1 cqa-min-h-[18rem]\">\n <!-- Variable Library header -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-flex-shrink-0\">\n <h3 class=\"cqa-text-[12px] cqa-leading-[20px] cqa-font-semibold cqa-text-[#111827] cqa-m-0\">\n Variable Library\n </h3>\n <a href=\"#\" (click)=\"onCreateNewVariable($event)\"\n class=\"cqa-text-[#3F43EE] cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-no-underline hover:cqa-opacity-90 cqa-cursor-pointer\">\n Create New +\n </a>\n </div>\n <!-- Filter tabs: All, Global, Local (blue underline under active label only) -->\n <div class=\"cqa-flex cqa-w-full cqa-flex-shrink-0\">\n <button *ngFor=\"let tab of parameterFilterTabs\" type=\"button\"\n (click)=\"onParameterScopeFilterChange(tab.value)\"\n class=\"cqa-flex-1 cqa-py-0 cqa-px-3 cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-bg-transparent cqa-cursor-pointer cqa-transition-colors cqa-flex cqa-justify-center cqa-items-center\"\n [class.cqa-text-[#6B7280]]=\"parameterScopeFilter !== tab.value\">\n <span class=\"cqa-test-data-param-filter-label cqa-inline-block cqa-pb-0.5\"\n [class.cqa-test-data-param-filter-active]=\"parameterScopeFilter === tab.value\">\n {{ tab.label }}\n </span>\n </button>\n </div>\n <!-- Search: bound to editForm (same pattern as Loop Step control binding) -->\n <div class=\"cqa-flex-shrink-0 cqa-relative cqa-flex cqa-items-center\">\n <span class=\"cqa-absolute cqa-left-4 cqa-top-0 cqa-bottom-0 cqa-flex cqa-items-center cqa-justify-center cqa-pointer-events-none cqa-text-[#6B7280] cqa-leading-none\" aria-hidden=\"true\">\n <mat-icon class=\"!cqa-w-4 !cqa-h-4 !cqa-text-[16px]\">search</mat-icon>\n </span>\n <input\n type=\"text\"\n [value]=\"editForm.get('parameterSearchQuery')?.value ?? ''\"\n (input)=\"onParameterSearchInput($any($event.target).value)\"\n placeholder=\"Search library\"\n class=\"cqa-test-data-search-input cqa-w-full cqa-pl-10 cqa-pr-4 cqa-py-2.5 cqa-text-sm cqa-leading-5 cqa-text-[#0A0A0A] placeholder:cqa-text-[#9CA3AF] focus:cqa-outline-none focus:cqa-border-[#3B82F6] focus:cqa-ring-1 focus:cqa-ring-[#3B82F6]\"\n aria-label=\"Search library\"\n />\n </div>\n <!-- Parameter list (scrollable, search-integrated via filtered list binding) -->\n <div class=\"cqa-min-h-[200px] cqa-flex-1 cqa-flex cqa-flex-col\">\n <cqa-item-list\n [items]=\"parameterListConfig\"\n [selectedId]=\"selectedParameterId\"\n emptyMessage=\"No parameters match your search.\"\n [scrollable]=\"true\"\n (itemSelect)=\"onParameterSelect($event)\"\n (itemEdit)=\"onParameterEdit($event)\">\n </cqa-item-list>\n </div>\n </div>\n </ng-container>\n\n <!-- Environment tab: Variable library by environment -->\n <ng-container *ngIf=\"activeTab === 'environment'\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-min-h-0 cqa-flex-1 cqa-min-h-[18rem]\">\n <!-- Section header -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-flex-shrink-0\">\n <h3 class=\"cqa-text-[12px] cqa-leading-[20px] cqa-font-semibold cqa-text-[#111827] cqa-m-0\">\n Environment\n </h3>\n <a href=\"#\" (click)=\"onCreateNewEnvironment($event)\"\n class=\"cqa-text-[#3F43EE] cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-no-underline hover:cqa-opacity-90 cqa-cursor-pointer\">\n Create New +\n </a>\n </div>\n <!-- Search: bound to editForm (same pattern as Loop Step control binding) -->\n <div class=\"cqa-flex-shrink-0 cqa-relative cqa-flex cqa-items-center\">\n <span class=\"cqa-absolute cqa-left-4 cqa-top-0 cqa-bottom-0 cqa-flex cqa-items-center cqa-justify-center cqa-pointer-events-none cqa-text-[#6B7280] cqa-leading-none\" aria-hidden=\"true\">\n <mat-icon class=\"!cqa-w-4 !cqa-h-4 !cqa-text-[16px]\">search</mat-icon>\n </span>\n <input\n type=\"text\"\n [value]=\"editForm.get('environmentSearchQuery')?.value ?? ''\"\n (input)=\"onEnvironmentSearchInput($any($event.target).value)\"\n placeholder=\"Search Environment\"\n class=\"cqa-test-data-search-input cqa-w-full cqa-pl-10 cqa-pr-4 cqa-py-2.5 cqa-text-sm cqa-leading-5 cqa-text-[#0A0A0A] placeholder:cqa-text-[#9CA3AF] focus:cqa-outline-none focus:cqa-border-[#3B82F6] focus:cqa-ring-1 focus:cqa-ring-[#3B82F6]\"\n aria-label=\"Search Environment\"\n />\n </div>\n <!-- Environment list (scrollable, search-integrated via filtered list binding) -->\n <div class=\"cqa-min-h-[220px] cqa-flex-1 cqa-flex cqa-flex-col\">\n <cqa-item-list\n [items]=\"environmentListConfig\"\n [selectedId]=\"selectedEnvironmentId\"\n emptyMessage=\"No environment variables match your search.\"\n [scrollable]=\"true\"\n (itemSelect)=\"onEnvironmentSelect($event)\"\n (itemEdit)=\"onEnvironmentEdit($event)\">\n </cqa-item-list>\n </div>\n </div>\n </ng-container>\n\n <!-- Runtime tab: dynamic value injector -->\n <ng-container *ngIf=\"activeTab === 'runtime'\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-3 cqa-min-h-0 cqa-h-full cqa-overflow-y-auto\">\n <h3 class=\"cqa-text-[12px] cqa-leading-[20px] cqa-font-semibold cqa-text-[#111827] cqa-m-0 cqa-flex-shrink-0\">\n Runtime\n </h3>\n <div class=\"cqa-flex-shrink-0\">\n <cqa-custom-input\n [value]=\"editForm.get('runtimeValue')?.value ?? ''\"\n (valueChange)=\"onEditFormFieldChange('runtimeValue', $event)\"\n placeholder=\"input text\"\n [fullWidth]=\"true\"\n size=\"md\">\n </cqa-custom-input>\n <p class=\"cqa-text-[12px] cqa-leading-[18px] cqa-text-[#6B7280] cqa-mt-1.5 cqa-mb-0\">\n Values resolved during test execution based on session and step state.\n </p>\n </div>\n <!-- Recommended: suggestion chips -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-flex-shrink-0\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-1.5\">\n <mat-icon class=\"!cqa-w-4 !cqa-h-4 !cqa-text-[16px] cqa-text-[#9CA3AF]\" aria-hidden=\"true\">schedule</mat-icon>\n <span class=\"cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-text-[#374151]\">Recommended</span>\n </div>\n <div class=\"cqa-test-data-runtime-chips-scroll cqa-flex cqa-gap-2 cqa-overflow-x-auto cqa-min-w-0 cqa-pb-1\">\n <button\n *ngFor=\"let snippet of runtimeSuggestedValues\"\n type=\"button\"\n (click)=\"onRuntimeChipInsert(snippet)\"\n class=\"cqa-flex-shrink-0 cqa-px-3 cqa-py-1.5 cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-text-[#374151] cqa-bg-[#F3F4F6] cqa-rounded-full hover:cqa-bg-[#E5E7EB] focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] cqa-transition-colors cqa-whitespace-nowrap\">\n {{ snippet }}\n </button>\n </div>\n </div>\n </div>\n </ng-container>\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-[12px] 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-[12px] 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: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i3.SegmentControlComponent, selector: "cqa-segment-control", inputs: ["segments", "value", "disabled", "containerBgColor"], outputs: ["valueChange"] }, { type: i4.CustomInputComponent, selector: "cqa-custom-input", inputs: ["label", "type", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "inputInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused", "enterPressed"] }, { type: i5.ItemListComponent, selector: "cqa-item-list", inputs: ["items", "selectedId", "emptyMessage", "scrollable"], outputs: ["itemSelect", "itemEdit"] }, { type: i6.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }], directives: [{ type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
283
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: TestDataModalComponent, decorators: [{
|
|
284
|
+
type: Component,
|
|
285
|
+
args: [{ selector: 'cqa-test-data-modal', 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 cqa-min-h-0\">\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 Test Data\n </h2>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-min-h-[28px]\">\n <!-- Need help? with custom tooltip (works inside overlay) -->\n <div class=\"cqa-relative cqa-inline-flex cqa-items-center\"\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(#test-data-help-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=\"test-data-help-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(#test-data-help-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=\"test-data-help-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\"\n class=\"cqa-absolute cqa-pointer-events-none cqa-z-[100] cqa-top-[-24px] cqa-left-[-125px]\"\n role=\"tooltip\">\n <div\n class=\"cqa-text-white cqa-text-center cqa-whitespace-nowrap cqa-w-[306px] cqa-min-h-[20px] cqa-rounded-[6px] cqa-py-1 cqa-px-2 cqa-bg-[#0A0A0A] cqa-leading-[20px] cqa-text-[8px]\">\n {{ helpTooltipText }}\n </div>\n </div>\n </div>\n <button type=\"button\" (click)=\"onClose()\"\n class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-[28px] cqa-min-w-[28px] cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0\"\n title=\"Close\" aria-label=\"Close\">\n <mat-icon class=\"!cqa-w-5 !cqa-h-5 !cqa-text-[20px] !cqa-block !cqa-leading-none\">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 <!-- Tabs: Plain Text, Parameter, Environment, Runtime (bound to editForm like Loop Step) -->\n <div class=\"cqa-w-full cqa-test-data-modal-tabs\">\n <cqa-segment-control\n [segments]=\"tabSegments\"\n [value]=\"editForm.get('activeTab')?.value ?? activeTab\"\n (valueChange)=\"onTabChange($event)\">\n </cqa-segment-control>\n </div>\n\n <!-- Content per tab: flex-1 min-h-0 so Parameter/Environment/Runtime content can scroll -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-min-h-0\"\n [class.cqa-flex-1]=\"activeTab === 'parameter' || activeTab === 'environment' || activeTab === 'runtime'\">\n <!-- Plain Text tab (form-bound like Loop Step editForm + onEditFormFieldChange) -->\n <ng-container *ngIf=\"activeTab === 'plain-text'\">\n <cqa-custom-input\n label=\"Value\"\n [value]=\"editForm.get('value')?.value ?? ''\"\n (valueChange)=\"onEditFormFieldChange('value', $event)\"\n placeholder=\"https://example.com/dashboard\"\n [fullWidth]=\"true\"\n size=\"md\">\n </cqa-custom-input>\n <p class=\"cqa-text-[12px] cqa-leading-[18px] cqa-text-[#0A0A0A] cqa-m-0\">\n Enter a static text value that won't change during execution.\n </p>\n </ng-container>\n\n <!-- Parameter tab: Variable Library -->\n <ng-container *ngIf=\"activeTab === 'parameter'\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-min-h-0 cqa-flex-1 cqa-min-h-[18rem]\">\n <!-- Variable Library header -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-flex-shrink-0\">\n <h3 class=\"cqa-text-[12px] cqa-leading-[20px] cqa-font-semibold cqa-text-[#111827] cqa-m-0\">\n Variable Library\n </h3>\n <a href=\"#\" (click)=\"onCreateNewVariable($event)\"\n class=\"cqa-text-[#3F43EE] cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-no-underline hover:cqa-opacity-90 cqa-cursor-pointer\">\n Create New +\n </a>\n </div>\n <!-- Filter tabs: All, Global, Local (blue underline under active label only) -->\n <div class=\"cqa-flex cqa-w-full cqa-flex-shrink-0\">\n <button *ngFor=\"let tab of parameterFilterTabs\" type=\"button\"\n (click)=\"onParameterScopeFilterChange(tab.value)\"\n class=\"cqa-flex-1 cqa-py-0 cqa-px-3 cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-bg-transparent cqa-cursor-pointer cqa-transition-colors cqa-flex cqa-justify-center cqa-items-center\"\n [class.cqa-text-[#6B7280]]=\"parameterScopeFilter !== tab.value\">\n <span class=\"cqa-test-data-param-filter-label cqa-inline-block cqa-pb-0.5\"\n [class.cqa-test-data-param-filter-active]=\"parameterScopeFilter === tab.value\">\n {{ tab.label }}\n </span>\n </button>\n </div>\n <!-- Search: bound to editForm (same pattern as Loop Step control binding) -->\n <div class=\"cqa-flex-shrink-0 cqa-relative cqa-flex cqa-items-center\">\n <span class=\"cqa-absolute cqa-left-4 cqa-top-0 cqa-bottom-0 cqa-flex cqa-items-center cqa-justify-center cqa-pointer-events-none cqa-text-[#6B7280] cqa-leading-none\" aria-hidden=\"true\">\n <mat-icon class=\"!cqa-w-4 !cqa-h-4 !cqa-text-[16px]\">search</mat-icon>\n </span>\n <input\n type=\"text\"\n [value]=\"editForm.get('parameterSearchQuery')?.value ?? ''\"\n (input)=\"onParameterSearchInput($any($event.target).value)\"\n placeholder=\"Search library\"\n class=\"cqa-test-data-search-input cqa-w-full cqa-pl-10 cqa-pr-4 cqa-py-2.5 cqa-text-sm cqa-leading-5 cqa-text-[#0A0A0A] placeholder:cqa-text-[#9CA3AF] focus:cqa-outline-none focus:cqa-border-[#3B82F6] focus:cqa-ring-1 focus:cqa-ring-[#3B82F6]\"\n aria-label=\"Search library\"\n />\n </div>\n <!-- Parameter list (scrollable, search-integrated via filtered list binding) -->\n <div class=\"cqa-min-h-[200px] cqa-flex-1 cqa-flex cqa-flex-col\">\n <cqa-item-list\n [items]=\"parameterListConfig\"\n [selectedId]=\"selectedParameterId\"\n emptyMessage=\"No parameters match your search.\"\n [scrollable]=\"true\"\n (itemSelect)=\"onParameterSelect($event)\"\n (itemEdit)=\"onParameterEdit($event)\">\n </cqa-item-list>\n </div>\n </div>\n </ng-container>\n\n <!-- Environment tab: Variable library by environment -->\n <ng-container *ngIf=\"activeTab === 'environment'\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-min-h-0 cqa-flex-1 cqa-min-h-[18rem]\">\n <!-- Section header -->\n <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-flex-shrink-0\">\n <h3 class=\"cqa-text-[12px] cqa-leading-[20px] cqa-font-semibold cqa-text-[#111827] cqa-m-0\">\n Environment\n </h3>\n <a href=\"#\" (click)=\"onCreateNewEnvironment($event)\"\n class=\"cqa-text-[#3F43EE] cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-no-underline hover:cqa-opacity-90 cqa-cursor-pointer\">\n Create New +\n </a>\n </div>\n <!-- Search: bound to editForm (same pattern as Loop Step control binding) -->\n <div class=\"cqa-flex-shrink-0 cqa-relative cqa-flex cqa-items-center\">\n <span class=\"cqa-absolute cqa-left-4 cqa-top-0 cqa-bottom-0 cqa-flex cqa-items-center cqa-justify-center cqa-pointer-events-none cqa-text-[#6B7280] cqa-leading-none\" aria-hidden=\"true\">\n <mat-icon class=\"!cqa-w-4 !cqa-h-4 !cqa-text-[16px]\">search</mat-icon>\n </span>\n <input\n type=\"text\"\n [value]=\"editForm.get('environmentSearchQuery')?.value ?? ''\"\n (input)=\"onEnvironmentSearchInput($any($event.target).value)\"\n placeholder=\"Search Environment\"\n class=\"cqa-test-data-search-input cqa-w-full cqa-pl-10 cqa-pr-4 cqa-py-2.5 cqa-text-sm cqa-leading-5 cqa-text-[#0A0A0A] placeholder:cqa-text-[#9CA3AF] focus:cqa-outline-none focus:cqa-border-[#3B82F6] focus:cqa-ring-1 focus:cqa-ring-[#3B82F6]\"\n aria-label=\"Search Environment\"\n />\n </div>\n <!-- Environment list (scrollable, search-integrated via filtered list binding) -->\n <div class=\"cqa-min-h-[220px] cqa-flex-1 cqa-flex cqa-flex-col\">\n <cqa-item-list\n [items]=\"environmentListConfig\"\n [selectedId]=\"selectedEnvironmentId\"\n emptyMessage=\"No environment variables match your search.\"\n [scrollable]=\"true\"\n (itemSelect)=\"onEnvironmentSelect($event)\"\n (itemEdit)=\"onEnvironmentEdit($event)\">\n </cqa-item-list>\n </div>\n </div>\n </ng-container>\n\n <!-- Runtime tab: dynamic value injector -->\n <ng-container *ngIf=\"activeTab === 'runtime'\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-3 cqa-min-h-0 cqa-h-full cqa-overflow-y-auto\">\n <h3 class=\"cqa-text-[12px] cqa-leading-[20px] cqa-font-semibold cqa-text-[#111827] cqa-m-0 cqa-flex-shrink-0\">\n Runtime\n </h3>\n <div class=\"cqa-flex-shrink-0\">\n <cqa-custom-input\n [value]=\"editForm.get('runtimeValue')?.value ?? ''\"\n (valueChange)=\"onEditFormFieldChange('runtimeValue', $event)\"\n placeholder=\"input text\"\n [fullWidth]=\"true\"\n size=\"md\">\n </cqa-custom-input>\n <p class=\"cqa-text-[12px] cqa-leading-[18px] cqa-text-[#6B7280] cqa-mt-1.5 cqa-mb-0\">\n Values resolved during test execution based on session and step state.\n </p>\n </div>\n <!-- Recommended: suggestion chips -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-flex-shrink-0\">\n <div class=\"cqa-flex cqa-items-center cqa-gap-1.5\">\n <mat-icon class=\"!cqa-w-4 !cqa-h-4 !cqa-text-[16px] cqa-text-[#9CA3AF]\" aria-hidden=\"true\">schedule</mat-icon>\n <span class=\"cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-text-[#374151]\">Recommended</span>\n </div>\n <div class=\"cqa-test-data-runtime-chips-scroll cqa-flex cqa-gap-2 cqa-overflow-x-auto cqa-min-w-0 cqa-pb-1\">\n <button\n *ngFor=\"let snippet of runtimeSuggestedValues\"\n type=\"button\"\n (click)=\"onRuntimeChipInsert(snippet)\"\n class=\"cqa-flex-shrink-0 cqa-px-3 cqa-py-1.5 cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-text-[#374151] cqa-bg-[#F3F4F6] cqa-rounded-full hover:cqa-bg-[#E5E7EB] focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] cqa-transition-colors cqa-whitespace-nowrap\">\n {{ snippet }}\n </button>\n </div>\n </div>\n </div>\n </ng-container>\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-[12px] 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-[12px] 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" }]
|
|
286
|
+
}], ctorParameters: function () { return [{ type: i1.FormBuilder }, { type: i0.ChangeDetectorRef }, { type: i8.TestDataModalRef, decorators: [{
|
|
287
|
+
type: Inject,
|
|
288
|
+
args: [TEST_DATA_MODAL_REF]
|
|
289
|
+
}] }, { type: undefined, decorators: [{
|
|
290
|
+
type: Optional
|
|
291
|
+
}, {
|
|
292
|
+
type: Inject,
|
|
293
|
+
args: [TEST_DATA_MODAL_DATA]
|
|
294
|
+
}] }]; }, propDecorators: { apply: [{
|
|
295
|
+
type: Output
|
|
296
|
+
}], cancel: [{
|
|
297
|
+
type: Output
|
|
298
|
+
}], editInDepth: [{
|
|
299
|
+
type: Output
|
|
300
|
+
}] } });
|
|
301
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test-data-modal.component.js","sourceRoot":"","sources":["../../../../../../src/lib/test-case-details/test-data-modal/test-data-modal.component.ts","../../../../../../src/lib/test-case-details/test-data-modal/test-data-modal.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Inject,\n  OnInit,\n  Optional,\n  Output,\n} from '@angular/core';\nimport { FormBuilder, FormGroup } from '@angular/forms';\nimport { DynamicSelectFieldConfig, SelectOption } from '../../dynamic-select/dynamic-select-field.component';\nimport {\n  TEST_DATA_MODAL_DATA,\n  TEST_DATA_MODAL_EDIT_IN_DEPTH,\n  TestDataModalData,\n  TestDataModalTab,\n} from './test-data-modal-data';\nimport { TEST_DATA_MODAL_REF, TestDataModalRef } from './test-data-modal-ref';\nimport { TestDataModalResult } from './test-data-modal.service';\nimport {\n  MOCK_ENVIRONMENT_ITEMS,\n  EnvironmentItem,\n} from './test-data-modal-environment.model';\nimport {\n  MOCK_PARAMETERS,\n  ParameterItem,\n  ParameterScopeFilter,\n} from './test-data-modal-parameter.model';\nimport { CqaListItemConfig } from '../../item-list/item-list.model';\n\nconst TAB_SEGMENTS = [\n  { label: 'Plain Text', value: 'plain-text' },\n  { label: 'Parameter', value: 'parameter' },\n  { label: 'Environment', value: 'environment' },\n  { label: 'Runtime', value: 'runtime' },\n];\n\nconst PARAMETER_FILTER_TABS: { label: string; value: ParameterScopeFilter }[] = [\n  { label: 'All', value: 'all' },\n  { label: 'Global', value: 'global' },\n  { label: 'Local', value: 'local' },\n];\n\n/** Mock runtime variable suggestions (replace with API when integrating). */\nconst RUNTIME_SUGGESTED_VALUES = [\n  '{{runtime.MaxWaitTime}}',\n  '{{runtime.session_id}}',\n  '{{runtime.auth_token}}',\n  '{{runtime.step_index}}',\n  '{{runtime.timestamp}}',\n  '{{runtime.test_name}}',\n];\n\n@Component({\n  selector: 'cqa-test-data-modal',\n  templateUrl: './test-data-modal.component.html',\n  host: { class: 'cqa-ui-root' },\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class TestDataModalComponent implements OnInit {\n  @Output() apply = new EventEmitter<{value: string, activeTab: TestDataModalTab}>();\n  @Output() cancel = new EventEmitter<void>();\n  @Output() editInDepth = new EventEmitter<void>();\n\n  readonly tabSegments = TAB_SEGMENTS;\n  readonly parameterFilterTabs = PARAMETER_FILTER_TABS;\n  /** Mock parameter list (replace with API when integrating). */\n  readonly parameters: ParameterItem[] = MOCK_PARAMETERS;\n  /** Mock environment list (replace with API when integrating). */\n  readonly environmentItems: EnvironmentItem[] = MOCK_ENVIRONMENT_ITEMS;\n  /** Runtime tab: suggested variable chips. */\n  readonly runtimeSuggestedValues: readonly string[] = RUNTIME_SUGGESTED_VALUES;\n\n  /** Form for all modal controls (same pattern as Loop Step editForm). */\n  editForm!: FormGroup;\n\n  /** Cached select configs (stable refs, same naming as Loop Step: xxxSelectConfig). */\n  activeTabSelectConfig!: DynamicSelectFieldConfig;\n  parameterScopeFilterSelectConfig!: DynamicSelectFieldConfig;\n\n  /** Local copy of form value for Cancel revert (same pattern as Loop Step editSnapshot). */\n  private editSnapshot: Record<string, unknown> = {};\n\n  /** State derived from config/editForm for template and getters (mirrors Loop Step inputs). */\n  activeTab: TestDataModalTab = 'plain-text';\n  value = '';\n  helpUrl = '';\n  helpTooltipText = 'Not sure what to do? Click to go to our detailed step creation documentation';\n  showHelpTooltip = false;\n\n  parameterScopeFilter: ParameterScopeFilter = 'all';\n  parameterSearchQuery = '';\n  selectedParameterId: string | null = null;\n\n  environmentSearchQuery = '';\n  selectedEnvironmentId: string | null = null;\n\n  runtimeValue = '';\n\n  constructor(\n    private readonly fb: FormBuilder,\n    private readonly cdr: ChangeDetectorRef,\n    @Inject(TEST_DATA_MODAL_REF) private ref: TestDataModalRef<TestDataModalResult>,\n    @Optional() @Inject(TEST_DATA_MODAL_DATA) data?: TestDataModalData\n  ) {\n    if (data) {\n      this.value = data.value ?? '';\n      this.helpUrl = data.helpUrl ?? '';\n      this.activeTab = (data.activeTab ?? 'plain-text') as TestDataModalTab;\n      this.parameterScopeFilter = data.parameterScopeFilter ?? 'all';\n      this.parameterSearchQuery = data.parameterSearchQuery ?? '';\n      this.selectedParameterId = data.selectedParameterId ?? null;\n      this.environmentSearchQuery = data.environmentSearchQuery ?? '';\n      this.selectedEnvironmentId = data.selectedEnvironmentId ?? null;\n      this.runtimeValue = data.runtimeValue ?? '';\n    }\n  }\n\n  ngOnInit(): void {\n    this.buildEditForm();\n    this.buildSelectConfigs();\n    this.editSnapshot = { ...this.editForm.value };\n    this.cdr.markForCheck();\n  }\n\n  /** Build form with all control values (same pattern as Loop Step buildEditForm). */\n  private buildEditForm(): void {\n    this.editForm = this.fb.group({\n      value: [this.value ?? ''],\n      activeTab: [this.activeTab ?? 'plain-text'],\n      parameterScopeFilter: [this.parameterScopeFilter ?? 'all'],\n      parameterSearchQuery: [this.parameterSearchQuery ?? ''],\n      selectedParameterId: [this.selectedParameterId ?? null],\n      environmentSearchQuery: [this.environmentSearchQuery ?? ''],\n      selectedEnvironmentId: [this.selectedEnvironmentId ?? null],\n      runtimeValue: [this.runtimeValue ?? ''],\n    });\n  }\n\n  /** Build select configs once (same pattern as Loop Step buildSelectConfigs). */\n  private buildSelectConfigs(): void {\n    const tabOptions: SelectOption[] = TAB_SEGMENTS.map((s) => ({\n      id: s.value,\n      value: s.value,\n      name: s.label,\n      label: s.label,\n    }));\n    const filterOptions: SelectOption[] = PARAMETER_FILTER_TABS.map((s) => ({\n      id: s.value,\n      value: s.value,\n      name: s.label,\n      label: s.label,\n    }));\n    this.activeTabSelectConfig = {\n      key: 'activeTab',\n      placeholder: 'Tab',\n      searchable: false,\n      options: tabOptions,\n    };\n    this.parameterScopeFilterSelectConfig = {\n      key: 'parameterScopeFilter',\n      placeholder: 'Scope',\n      searchable: false,\n      options: filterOptions,\n    };\n  }\n\n  /** Update form control and sync component state (same pattern as Loop Step onEditFormFieldChange). */\n  onEditFormFieldChange(controlName: string, value: string | number | null): void {\n    this.editForm.get(controlName)?.setValue(value);\n    this.syncStateFromForm();\n    this.cdr.markForCheck();\n  }\n\n  /** Sync component state from editForm so getters (e.g. filteredParameters) stay correct. */\n  private syncStateFromForm(): void {\n    const v = this.editForm.value;\n    this.value = v.value ?? '';\n    this.activeTab = (v.activeTab ?? 'plain-text') as TestDataModalTab;\n    this.parameterScopeFilter = (v.parameterScopeFilter ?? 'all') as ParameterScopeFilter;\n    this.parameterSearchQuery = v.parameterSearchQuery ?? '';\n    this.selectedParameterId = v.selectedParameterId ?? null;\n    this.environmentSearchQuery = v.environmentSearchQuery ?? '';\n    this.selectedEnvironmentId = v.selectedEnvironmentId ?? null;\n    this.runtimeValue = v.runtimeValue ?? '';\n  }\n\n  /** Filtered parameters by scope and search (Parameter tab). */\n  get filteredParameters(): ParameterItem[] {\n    let list = this.parameters;\n    if (this.parameterScopeFilter === 'global') {\n      list = list.filter((p) => p.scope === 'Global');\n    } else if (this.parameterScopeFilter === 'local') {\n      list = list.filter((p) => p.scope === 'Local');\n    }\n    const q = this.parameterSearchQuery.trim().toLowerCase();\n    if (q) {\n      list = list.filter((p) => p.name.toLowerCase().includes(q));\n    }\n    return list;\n  }\n\n  /** Filtered environment items by search (Environment tab). */\n  get filteredEnvironments(): EnvironmentItem[] {\n    const q = this.environmentSearchQuery.trim().toLowerCase();\n    if (!q) {\n      return this.environmentItems;\n    }\n    return this.environmentItems.filter(\n      (e) =>\n        e.name.toLowerCase().includes(q) || e.value.toLowerCase().includes(q)\n    );\n  }\n\n  /** Parameter tab: list config for cqa-item-list (search-friendly: bind filtered list). */\n  get parameterListConfig(): CqaListItemConfig[] {\n    return this.filteredParameters.map((p) => ({\n      id: p.id,\n      title: p.name,\n      subtitle: p.value,\n      badge: p.scope,\n      showEdit: this.selectedParameterId === p.id,\n    }));\n  }\n\n  /** Environment tab: list config for cqa-item-list. */\n  get environmentListConfig(): CqaListItemConfig[] {\n    return this.filteredEnvironments.map((e) => ({\n      id: e.id,\n      title: e.name,\n      subtitle: `Value: ${e.value}`,\n      badge: e.environment,\n      showEdit: this.selectedEnvironmentId === e.id && (e.editable ?? true),\n    }));\n  }\n\n  onTabChange(tabValue: string): void {\n    this.onEditFormFieldChange('activeTab', tabValue);\n  }\n\n  onParameterScopeFilterChange(filter: ParameterScopeFilter): void {\n    this.onEditFormFieldChange('parameterScopeFilter', filter);\n  }\n\n  onParameterSearchInput(value: string): void {\n    this.onEditFormFieldChange('parameterSearchQuery', value);\n  }\n\n  onParameterSelect(paramId: string): void {\n    const param = this.parameters.find((p) => p.id === paramId);\n    if (param) {\n      this.selectedParameterId = this.selectedParameterId === param.id ? null : param.id;\n      this.onEditFormFieldChange('selectedParameterId', this.selectedParameterId);\n      this.onEditFormFieldChange('value', param.name);\n    }\n  }\n\n  onParameterEdit(_paramId: string): void {\n    // Placeholder for future edit action (e.g. open edit drawer).\n  }\n\n  onCreateNewVariable(event: Event): void {\n    event.preventDefault();\n    // Placeholder for future \"Create New\" action (no API yet).\n  }\n\n  onEnvironmentSearchInput(value: string): void {\n    this.onEditFormFieldChange('environmentSearchQuery', value);\n  }\n\n  onEnvironmentSelect(itemId: string): void {\n    const item = this.environmentItems.find((e) => e.id === itemId);\n    if (item) {\n      this.selectedEnvironmentId =\n        this.selectedEnvironmentId === item.id ? null : item.id;\n      this.onEditFormFieldChange('selectedEnvironmentId', this.selectedEnvironmentId);\n      this.onEditFormFieldChange('value', item.name);\n    }\n  }\n\n  onEnvironmentEdit(_itemId: string): void {\n    // Placeholder for future edit action (e.g. open edit drawer).\n  }\n\n  onCreateNewEnvironment(event: Event): void {\n    event.preventDefault();\n    // Placeholder for future \"Create New\" action (no API yet).\n  }\n\n  /** Runtime tab: insert suggested value into the runtime input field. */\n  onRuntimeChipInsert(snippet: string): void {\n    const next = this.editForm.get('runtimeValue')?.value\n      ? this.editForm.get('runtimeValue')?.value + ' ' + snippet\n      : snippet;\n    this.onEditFormFieldChange('runtimeValue', next);\n  }\n\n  onApply(): void {\n    this.syncStateFromForm();\n    let valueToApply = this.activeTab === 'runtime' ? this.runtimeValue : this.value;\n\n    if (this.activeTab === 'parameter') {\n      valueToApply = this.selectedParameterId ? this.parameters.find((p) => p.id === this.selectedParameterId)?.name ?? '' : '';\n    } else if (this.activeTab === 'environment') {\n      valueToApply = this.selectedEnvironmentId ? this.environmentItems.find((e) => e.id === this.selectedEnvironmentId)?.name ?? '' : '';\n    }\n\n    // Build result with only the active tab's data, clear other tabs' data\n    const result: TestDataModalData = {\n      value: valueToApply,\n      activeTab: this.activeTab,\n      // Only include parameter data if parameter tab is active\n      parameterScopeFilter: this.activeTab === 'parameter' ? this.parameterScopeFilter : undefined,\n      parameterSearchQuery: this.activeTab === 'parameter' ? this.parameterSearchQuery : undefined,\n      selectedParameterId: this.activeTab === 'parameter' ? this.selectedParameterId : undefined,\n      // Only include environment data if environment tab is active\n      environmentSearchQuery: this.activeTab === 'environment' ? this.environmentSearchQuery : undefined,\n      selectedEnvironmentId: this.activeTab === 'environment' ? this.selectedEnvironmentId : undefined,\n      // Only include runtime data if runtime tab is active\n      runtimeValue: this.activeTab === 'runtime' ? this.runtimeValue : undefined,\n    };\n\n    this.apply.emit({value: valueToApply, activeTab: this.activeTab as TestDataModalTab});\n    this.ref.close(result);\n  }\n\n  onCancel(): void {\n    this.editForm.patchValue(this.editSnapshot);\n    this.syncStateFromForm();\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(TEST_DATA_MODAL_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 cqa-min-h-0\">\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      Test Data\n    </h2>\n    <div class=\"cqa-flex cqa-items-center cqa-gap-2 cqa-min-h-[28px]\">\n      <!-- Need help? with custom tooltip (works inside overlay) -->\n      <div class=\"cqa-relative cqa-inline-flex cqa-items-center\"\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(#test-data-help-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=\"test-data-help-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(#test-data-help-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=\"test-data-help-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\"\n          class=\"cqa-absolute cqa-pointer-events-none cqa-z-[100] cqa-top-[-24px] cqa-left-[-125px]\"\n          role=\"tooltip\">\n          <div\n            class=\"cqa-text-white cqa-text-center cqa-whitespace-nowrap cqa-w-[306px] cqa-min-h-[20px] cqa-rounded-[6px] cqa-py-1 cqa-px-2 cqa-bg-[#0A0A0A] cqa-leading-[20px] cqa-text-[8px]\">\n            {{ helpTooltipText }}\n          </div>\n        </div>\n      </div>\n      <button type=\"button\" (click)=\"onClose()\"\n        class=\"cqa-flex cqa-items-center cqa-justify-center cqa-min-h-[28px] cqa-min-w-[28px] cqa-rounded cqa-text-[#6B7280] hover:cqa-bg-[#F3F4F6] cqa-p-0\"\n        title=\"Close\" aria-label=\"Close\">\n        <mat-icon class=\"!cqa-w-5 !cqa-h-5 !cqa-text-[20px] !cqa-block !cqa-leading-none\">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  <!-- Tabs: Plain Text, Parameter, Environment, Runtime (bound to editForm like Loop Step) -->\n  <div class=\"cqa-w-full cqa-test-data-modal-tabs\">\n    <cqa-segment-control\n      [segments]=\"tabSegments\"\n      [value]=\"editForm.get('activeTab')?.value ?? activeTab\"\n      (valueChange)=\"onTabChange($event)\">\n    </cqa-segment-control>\n  </div>\n\n  <!-- Content per tab: flex-1 min-h-0 so Parameter/Environment/Runtime content can scroll -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-1.5 cqa-min-h-0\"\n    [class.cqa-flex-1]=\"activeTab === 'parameter' || activeTab === 'environment' || activeTab === 'runtime'\">\n    <!-- Plain Text tab (form-bound like Loop Step editForm + onEditFormFieldChange) -->\n    <ng-container *ngIf=\"activeTab === 'plain-text'\">\n      <cqa-custom-input\n        label=\"Value\"\n        [value]=\"editForm.get('value')?.value ?? ''\"\n        (valueChange)=\"onEditFormFieldChange('value', $event)\"\n        placeholder=\"https://example.com/dashboard\"\n        [fullWidth]=\"true\"\n        size=\"md\">\n      </cqa-custom-input>\n      <p class=\"cqa-text-[12px] cqa-leading-[18px] cqa-text-[#0A0A0A] cqa-m-0\">\n        Enter a static text value that won't change during execution.\n      </p>\n    </ng-container>\n\n    <!-- Parameter tab: Variable Library -->\n    <ng-container *ngIf=\"activeTab === 'parameter'\">\n      <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-min-h-0 cqa-flex-1 cqa-min-h-[18rem]\">\n        <!-- Variable Library header -->\n        <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-flex-shrink-0\">\n          <h3 class=\"cqa-text-[12px] cqa-leading-[20px] cqa-font-semibold cqa-text-[#111827] cqa-m-0\">\n            Variable Library\n          </h3>\n          <a href=\"#\" (click)=\"onCreateNewVariable($event)\"\n            class=\"cqa-text-[#3F43EE] cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-no-underline hover:cqa-opacity-90 cqa-cursor-pointer\">\n            Create New +\n          </a>\n        </div>\n        <!-- Filter tabs: All, Global, Local (blue underline under active label only) -->\n        <div class=\"cqa-flex cqa-w-full cqa-flex-shrink-0\">\n          <button *ngFor=\"let tab of parameterFilterTabs\" type=\"button\"\n            (click)=\"onParameterScopeFilterChange(tab.value)\"\n            class=\"cqa-flex-1 cqa-py-0 cqa-px-3 cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-bg-transparent cqa-cursor-pointer cqa-transition-colors cqa-flex cqa-justify-center cqa-items-center\"\n            [class.cqa-text-[#6B7280]]=\"parameterScopeFilter !== tab.value\">\n            <span class=\"cqa-test-data-param-filter-label cqa-inline-block cqa-pb-0.5\"\n              [class.cqa-test-data-param-filter-active]=\"parameterScopeFilter === tab.value\">\n              {{ tab.label }}\n            </span>\n          </button>\n        </div>\n        <!-- Search: bound to editForm (same pattern as Loop Step control binding) -->\n        <div class=\"cqa-flex-shrink-0 cqa-relative cqa-flex cqa-items-center\">\n          <span class=\"cqa-absolute cqa-left-4 cqa-top-0 cqa-bottom-0 cqa-flex cqa-items-center cqa-justify-center cqa-pointer-events-none cqa-text-[#6B7280] cqa-leading-none\" aria-hidden=\"true\">\n            <mat-icon class=\"!cqa-w-4 !cqa-h-4 !cqa-text-[16px]\">search</mat-icon>\n          </span>\n          <input\n            type=\"text\"\n            [value]=\"editForm.get('parameterSearchQuery')?.value ?? ''\"\n            (input)=\"onParameterSearchInput($any($event.target).value)\"\n            placeholder=\"Search library\"\n            class=\"cqa-test-data-search-input cqa-w-full cqa-pl-10 cqa-pr-4 cqa-py-2.5 cqa-text-sm cqa-leading-5 cqa-text-[#0A0A0A] placeholder:cqa-text-[#9CA3AF] focus:cqa-outline-none focus:cqa-border-[#3B82F6] focus:cqa-ring-1 focus:cqa-ring-[#3B82F6]\"\n            aria-label=\"Search library\"\n          />\n        </div>\n        <!-- Parameter list (scrollable, search-integrated via filtered list binding) -->\n        <div class=\"cqa-min-h-[200px] cqa-flex-1 cqa-flex cqa-flex-col\">\n          <cqa-item-list\n            [items]=\"parameterListConfig\"\n            [selectedId]=\"selectedParameterId\"\n            emptyMessage=\"No parameters match your search.\"\n            [scrollable]=\"true\"\n            (itemSelect)=\"onParameterSelect($event)\"\n            (itemEdit)=\"onParameterEdit($event)\">\n          </cqa-item-list>\n        </div>\n      </div>\n    </ng-container>\n\n    <!-- Environment tab: Variable library by environment -->\n    <ng-container *ngIf=\"activeTab === 'environment'\">\n      <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-min-h-0 cqa-flex-1 cqa-min-h-[18rem]\">\n        <!-- Section header -->\n        <div class=\"cqa-flex cqa-items-center cqa-justify-between cqa-flex-shrink-0\">\n          <h3 class=\"cqa-text-[12px] cqa-leading-[20px] cqa-font-semibold cqa-text-[#111827] cqa-m-0\">\n            Environment\n          </h3>\n          <a href=\"#\" (click)=\"onCreateNewEnvironment($event)\"\n            class=\"cqa-text-[#3F43EE] cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-no-underline hover:cqa-opacity-90 cqa-cursor-pointer\">\n            Create New +\n          </a>\n        </div>\n        <!-- Search: bound to editForm (same pattern as Loop Step control binding) -->\n        <div class=\"cqa-flex-shrink-0 cqa-relative cqa-flex cqa-items-center\">\n          <span class=\"cqa-absolute cqa-left-4 cqa-top-0 cqa-bottom-0 cqa-flex cqa-items-center cqa-justify-center cqa-pointer-events-none cqa-text-[#6B7280] cqa-leading-none\" aria-hidden=\"true\">\n            <mat-icon class=\"!cqa-w-4 !cqa-h-4 !cqa-text-[16px]\">search</mat-icon>\n          </span>\n          <input\n            type=\"text\"\n            [value]=\"editForm.get('environmentSearchQuery')?.value ?? ''\"\n            (input)=\"onEnvironmentSearchInput($any($event.target).value)\"\n            placeholder=\"Search Environment\"\n            class=\"cqa-test-data-search-input cqa-w-full cqa-pl-10 cqa-pr-4 cqa-py-2.5 cqa-text-sm cqa-leading-5 cqa-text-[#0A0A0A] placeholder:cqa-text-[#9CA3AF] focus:cqa-outline-none focus:cqa-border-[#3B82F6] focus:cqa-ring-1 focus:cqa-ring-[#3B82F6]\"\n            aria-label=\"Search Environment\"\n          />\n        </div>\n        <!-- Environment list (scrollable, search-integrated via filtered list binding) -->\n        <div class=\"cqa-min-h-[220px] cqa-flex-1 cqa-flex cqa-flex-col\">\n          <cqa-item-list\n            [items]=\"environmentListConfig\"\n            [selectedId]=\"selectedEnvironmentId\"\n            emptyMessage=\"No environment variables match your search.\"\n            [scrollable]=\"true\"\n            (itemSelect)=\"onEnvironmentSelect($event)\"\n            (itemEdit)=\"onEnvironmentEdit($event)\">\n          </cqa-item-list>\n        </div>\n      </div>\n    </ng-container>\n\n    <!-- Runtime tab: dynamic value injector -->\n    <ng-container *ngIf=\"activeTab === 'runtime'\">\n      <div class=\"cqa-flex cqa-flex-col cqa-gap-3 cqa-min-h-0 cqa-h-full cqa-overflow-y-auto\">\n        <h3 class=\"cqa-text-[12px] cqa-leading-[20px] cqa-font-semibold cqa-text-[#111827] cqa-m-0 cqa-flex-shrink-0\">\n          Runtime\n        </h3>\n        <div class=\"cqa-flex-shrink-0\">\n          <cqa-custom-input\n            [value]=\"editForm.get('runtimeValue')?.value ?? ''\"\n            (valueChange)=\"onEditFormFieldChange('runtimeValue', $event)\"\n            placeholder=\"input text\"\n            [fullWidth]=\"true\"\n            size=\"md\">\n          </cqa-custom-input>\n          <p class=\"cqa-text-[12px] cqa-leading-[18px] cqa-text-[#6B7280] cqa-mt-1.5 cqa-mb-0\">\n            Values resolved during test execution based on session and step state.\n          </p>\n        </div>\n        <!-- Recommended: suggestion chips -->\n        <div class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-flex-shrink-0\">\n          <div class=\"cqa-flex cqa-items-center cqa-gap-1.5\">\n            <mat-icon class=\"!cqa-w-4 !cqa-h-4 !cqa-text-[16px] cqa-text-[#9CA3AF]\" aria-hidden=\"true\">schedule</mat-icon>\n            <span class=\"cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-text-[#374151]\">Recommended</span>\n          </div>\n          <div class=\"cqa-test-data-runtime-chips-scroll cqa-flex cqa-gap-2 cqa-overflow-x-auto cqa-min-w-0 cqa-pb-1\">\n            <button\n              *ngFor=\"let snippet of runtimeSuggestedValues\"\n              type=\"button\"\n              (click)=\"onRuntimeChipInsert(snippet)\"\n              class=\"cqa-flex-shrink-0 cqa-px-3 cqa-py-1.5 cqa-text-[12px] cqa-leading-[18px] cqa-font-medium cqa-text-[#374151] cqa-bg-[#F3F4F6] cqa-rounded-full hover:cqa-bg-[#E5E7EB] focus:cqa-outline-none focus-visible:cqa-ring-2 focus-visible:cqa-ring-[#3F43EE] cqa-transition-colors cqa-whitespace-nowrap\">\n              {{ snippet }}\n            </button>\n          </div>\n        </div>\n      </div>\n    </ng-container>\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-[12px] 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-[12px] 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"]}
|