@cqa-lib/cqa-ui 1.1.522 → 1.1.524

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.
Files changed (26) hide show
  1. package/esm2020/lib/assign-environments-dialog/assign-environments-dialog.component.mjs +155 -0
  2. package/esm2020/lib/assign-environments-dialog/assign-environments-dialog.models.mjs +2 -0
  3. package/esm2020/lib/new-environment-dialog/new-environment-dialog.component.mjs +123 -0
  4. package/esm2020/lib/new-environment-dialog/new-environment-dialog.models.mjs +9 -0
  5. package/esm2020/lib/new-environment-variable-dialog/new-environment-variable-dialog.component.mjs +190 -0
  6. package/esm2020/lib/new-environment-variable-dialog/new-environment-variable-dialog.models.mjs +2 -0
  7. package/esm2020/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.component.mjs +188 -0
  8. package/esm2020/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.models.mjs +2 -0
  9. package/esm2020/lib/ui-kit.module.mjs +21 -1
  10. package/esm2020/public-api.mjs +9 -1
  11. package/fesm2015/cqa-lib-cqa-ui.mjs +669 -1
  12. package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
  13. package/fesm2020/cqa-lib-cqa-ui.mjs +657 -1
  14. package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
  15. package/lib/assign-environments-dialog/assign-environments-dialog.component.d.ts +36 -0
  16. package/lib/assign-environments-dialog/assign-environments-dialog.models.d.ts +19 -0
  17. package/lib/new-environment-dialog/new-environment-dialog.component.d.ts +37 -0
  18. package/lib/new-environment-dialog/new-environment-dialog.models.d.ts +14 -0
  19. package/lib/new-environment-variable-dialog/new-environment-variable-dialog.component.d.ts +54 -0
  20. package/lib/new-environment-variable-dialog/new-environment-variable-dialog.models.d.ts +14 -0
  21. package/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.component.d.ts +43 -0
  22. package/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.models.d.ts +20 -0
  23. package/lib/ui-kit.module.d.ts +92 -88
  24. package/package.json +1 -1
  25. package/public-api.d.ts +8 -0
  26. package/styles.css +1 -1
@@ -0,0 +1,155 @@
1
+ import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
2
+ import { FormControl, FormGroup } from '@angular/forms';
3
+ import * as i0 from "@angular/core";
4
+ import * as i1 from "../dynamic-select/dynamic-select-field.component";
5
+ import * as i2 from "@angular/common";
6
+ export class AssignEnvironmentsDialogComponent {
7
+ constructor(cdr) {
8
+ this.cdr = cdr;
9
+ this.mode = 'assign';
10
+ this.profileName = '';
11
+ this.environments = [];
12
+ this.assignedEnvironments = [];
13
+ this.selected = new Set();
14
+ this.sourceForm = new FormGroup({
15
+ sourceEnvId: new FormControl(null),
16
+ });
17
+ this.sourceConfig = this.buildSourceConfig([]);
18
+ this.sourceError = null;
19
+ }
20
+ ngOnInit() {
21
+ // Source picker appears in both modes when there are already-attached envs
22
+ // to copy from. In clone mode it's required; in assign mode it's optional
23
+ // (falling back to the base profile rows when no source is picked).
24
+ if (this.showSourcePicker) {
25
+ const initial = this.defaultSourceEnvId != null
26
+ ? this.defaultSourceEnvId
27
+ : (this.mode === 'clone' && this.assignedEnvironments && this.assignedEnvironments.length
28
+ ? this.assignedEnvironments[0].id
29
+ : null);
30
+ this.sourceForm.get('sourceEnvId').setValue(initial);
31
+ this.sourceConfig = this.buildSourceConfig(this.assignedEnvironments ?? []);
32
+ }
33
+ }
34
+ get showSourcePicker() {
35
+ return (this.assignedEnvironments && this.assignedEnvironments.length > 0);
36
+ }
37
+ get sourceLabel() {
38
+ return this.mode === 'clone' ? 'Copy from which environment' : 'Copy rows from (optional)';
39
+ }
40
+ get sourceHelper() {
41
+ if (this.sourceError) {
42
+ return this.sourceError;
43
+ }
44
+ return this.mode === 'clone'
45
+ ? "The selected environment's rows will be copied into each target environment below."
46
+ : 'Pick an environment to copy its rows into the new ones. Leave blank to seed with the base profile rows.';
47
+ }
48
+ get title() {
49
+ return this.mode === 'clone' ? 'Clone to other environments' : 'Assign to environments';
50
+ }
51
+ get subtitle() {
52
+ const name = this.profileName ? `"${this.profileName}"` : 'this profile';
53
+ return this.mode === 'clone'
54
+ ? `Duplicate ${name} into other environments. Column structure is shared; row values are independent per environment.`
55
+ : `Make ${name} available in the selected environments. Columns are shared — each environment gets an independent row set.`;
56
+ }
57
+ get primaryButtonLabel() {
58
+ const count = this.selected.size;
59
+ const verb = this.mode === 'clone' ? 'Clone to' : 'Assign to';
60
+ return `${verb} ${count} environment${count === 1 ? '' : 's'}`;
61
+ }
62
+ get primaryDisabled() {
63
+ if (this.selected.size === 0) {
64
+ return true;
65
+ }
66
+ if (this.mode === 'clone' && this.sourceForm.get('sourceEnvId').value == null) {
67
+ return true;
68
+ }
69
+ return false;
70
+ }
71
+ get selectableEnvironments() {
72
+ return this.environments ?? [];
73
+ }
74
+ get hasAnySelectable() {
75
+ return (this.environments ?? []).some(e => !e.alreadyAssigned);
76
+ }
77
+ get sourceValue() {
78
+ return this.sourceForm.get('sourceEnvId').value ?? null;
79
+ }
80
+ isSelected(id) {
81
+ return this.selected.has(id);
82
+ }
83
+ toggle(env) {
84
+ if (env.alreadyAssigned) {
85
+ return;
86
+ }
87
+ if (this.selected.has(env.id)) {
88
+ this.selected.delete(env.id);
89
+ }
90
+ else {
91
+ this.selected.add(env.id);
92
+ }
93
+ this.cdr.markForCheck();
94
+ }
95
+ helperFor(env) {
96
+ if (env.alreadyAssigned) {
97
+ return 'Already assigned to this environment';
98
+ }
99
+ return env.description || '';
100
+ }
101
+ getValue() {
102
+ if (this.selected.size === 0) {
103
+ return null;
104
+ }
105
+ const src = this.sourceValue;
106
+ if (this.mode === 'clone') {
107
+ if (src == null) {
108
+ this.sourceError = 'Pick the environment whose rows should be copied.';
109
+ this.cdr.markForCheck();
110
+ return null;
111
+ }
112
+ this.sourceError = null;
113
+ return { selectedIds: Array.from(this.selected), sourceEnvId: Number(src) };
114
+ }
115
+ // Assign mode: source is optional — include it when the user picked one.
116
+ this.sourceError = null;
117
+ return src != null
118
+ ? { selectedIds: Array.from(this.selected), sourceEnvId: Number(src) }
119
+ : { selectedIds: Array.from(this.selected) };
120
+ }
121
+ buildSourceConfig(envs) {
122
+ const options = (envs || []).map(e => ({
123
+ id: e.id,
124
+ value: e.id,
125
+ name: e.name,
126
+ label: e.name,
127
+ statusColor: e.color,
128
+ }));
129
+ return {
130
+ key: 'sourceEnvId',
131
+ label: '',
132
+ placeholder: 'Select source environment…',
133
+ multiple: false,
134
+ searchable: true,
135
+ options,
136
+ };
137
+ }
138
+ }
139
+ AssignEnvironmentsDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AssignEnvironmentsDialogComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
140
+ AssignEnvironmentsDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: AssignEnvironmentsDialogComponent, selector: "cqa-assign-environments-dialog", inputs: { mode: "mode", profileName: "profileName", environments: "environments", assignedEnvironments: "assignedEnvironments", defaultSourceEnvId: "defaultSourceEnvId" }, host: { styleAttribute: "display:block;width:100%;", classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n <!-- Source env picker: required in clone mode, optional in assign mode -->\n <div *ngIf=\"showSourcePicker\" class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">{{ sourceLabel }}</label>\n <cqa-dynamic-select\n [form]=\"sourceForm\"\n [config]=\"sourceConfig\">\n </cqa-dynamic-select>\n <span [class.cqa-text-red-600]=\"sourceError\" [class.cqa-text-gray-500]=\"!sourceError\" class=\"cqa-text-xs\">\n {{ sourceHelper }}\n </span>\n </div>\n\n <!-- Selectable envs -->\n <div\n *ngIf=\"hasAnySelectable || (environments?.length || 0) > 0\"\n class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-max-h-[340px] cqa-overflow-y-auto\">\n <label\n *ngFor=\"let env of selectableEnvironments\"\n class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-px-3 cqa-py-2.5 cqa-border cqa-rounded-[10px] cqa-bg-white cqa-transition-colors\"\n [class.cqa-border-gray-200]=\"!isSelected(env.id) && !env.alreadyAssigned\"\n [class.hover:cqa-border-indigo-200]=\"!env.alreadyAssigned && !isSelected(env.id)\"\n [class.cqa-border-indigo-400]=\"isSelected(env.id)\"\n [class.cqa-bg-indigo-50]=\"isSelected(env.id)\"\n [class.cqa-opacity-50]=\"env.alreadyAssigned\"\n [class.cqa-cursor-pointer]=\"!env.alreadyAssigned\"\n [class.cqa-cursor-not-allowed]=\"env.alreadyAssigned\"\n (click)=\"toggle(env)\">\n <input\n type=\"checkbox\"\n class=\"cqa-w-4 cqa-h-4 cqa-cursor-pointer\"\n [checked]=\"isSelected(env.id)\"\n [disabled]=\"!!env.alreadyAssigned\"\n (click)=\"$event.stopPropagation()\"\n (change)=\"toggle(env)\" />\n\n <span\n class=\"cqa-inline-block cqa-w-2 cqa-h-2 cqa-rounded-full cqa-flex-none\"\n [style.background]=\"env.color || '#3F43EE'\"></span>\n\n <div class=\"cqa-flex cqa-flex-col cqa-min-w-0 cqa-flex-1\">\n <div class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">{{ env.name }}</div>\n <div *ngIf=\"helperFor(env)\" class=\"cqa-text-xs cqa-text-gray-500 cqa-leading-[1.4]\">{{ helperFor(env) }}</div>\n </div>\n </label>\n </div>\n\n <div\n *ngIf=\"(environments?.length || 0) === 0\"\n class=\"cqa-py-8 cqa-px-4 cqa-text-center cqa-text-sm cqa-text-gray-500\">\n All environments are already assigned to this profile.\n </div>\n</div>\n", components: [{ type: i1.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
141
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: AssignEnvironmentsDialogComponent, decorators: [{
142
+ type: Component,
143
+ args: [{ selector: 'cqa-assign-environments-dialog', changeDetection: ChangeDetectionStrategy.OnPush, host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n <!-- Source env picker: required in clone mode, optional in assign mode -->\n <div *ngIf=\"showSourcePicker\" class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">{{ sourceLabel }}</label>\n <cqa-dynamic-select\n [form]=\"sourceForm\"\n [config]=\"sourceConfig\">\n </cqa-dynamic-select>\n <span [class.cqa-text-red-600]=\"sourceError\" [class.cqa-text-gray-500]=\"!sourceError\" class=\"cqa-text-xs\">\n {{ sourceHelper }}\n </span>\n </div>\n\n <!-- Selectable envs -->\n <div\n *ngIf=\"hasAnySelectable || (environments?.length || 0) > 0\"\n class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-max-h-[340px] cqa-overflow-y-auto\">\n <label\n *ngFor=\"let env of selectableEnvironments\"\n class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-px-3 cqa-py-2.5 cqa-border cqa-rounded-[10px] cqa-bg-white cqa-transition-colors\"\n [class.cqa-border-gray-200]=\"!isSelected(env.id) && !env.alreadyAssigned\"\n [class.hover:cqa-border-indigo-200]=\"!env.alreadyAssigned && !isSelected(env.id)\"\n [class.cqa-border-indigo-400]=\"isSelected(env.id)\"\n [class.cqa-bg-indigo-50]=\"isSelected(env.id)\"\n [class.cqa-opacity-50]=\"env.alreadyAssigned\"\n [class.cqa-cursor-pointer]=\"!env.alreadyAssigned\"\n [class.cqa-cursor-not-allowed]=\"env.alreadyAssigned\"\n (click)=\"toggle(env)\">\n <input\n type=\"checkbox\"\n class=\"cqa-w-4 cqa-h-4 cqa-cursor-pointer\"\n [checked]=\"isSelected(env.id)\"\n [disabled]=\"!!env.alreadyAssigned\"\n (click)=\"$event.stopPropagation()\"\n (change)=\"toggle(env)\" />\n\n <span\n class=\"cqa-inline-block cqa-w-2 cqa-h-2 cqa-rounded-full cqa-flex-none\"\n [style.background]=\"env.color || '#3F43EE'\"></span>\n\n <div class=\"cqa-flex cqa-flex-col cqa-min-w-0 cqa-flex-1\">\n <div class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">{{ env.name }}</div>\n <div *ngIf=\"helperFor(env)\" class=\"cqa-text-xs cqa-text-gray-500 cqa-leading-[1.4]\">{{ helperFor(env) }}</div>\n </div>\n </label>\n </div>\n\n <div\n *ngIf=\"(environments?.length || 0) === 0\"\n class=\"cqa-py-8 cqa-px-4 cqa-text-center cqa-text-sm cqa-text-gray-500\">\n All environments are already assigned to this profile.\n </div>\n</div>\n" }]
144
+ }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { mode: [{
145
+ type: Input
146
+ }], profileName: [{
147
+ type: Input
148
+ }], environments: [{
149
+ type: Input
150
+ }], assignedEnvironments: [{
151
+ type: Input
152
+ }], defaultSourceEnvId: [{
153
+ type: Input
154
+ }] } });
155
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"assign-environments-dialog.component.js","sourceRoot":"","sources":["../../../../../src/lib/assign-environments-dialog/assign-environments-dialog.component.ts","../../../../../src/lib/assign-environments-dialog/assign-environments-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Input, OnInit } from '@angular/core';\nimport { FormControl, FormGroup } from '@angular/forms';\n\nimport {\n  DynamicSelectFieldConfig,\n  SelectOption,\n} from '../dynamic-select/dynamic-select-field.component';\nimport {\n  AssignEnvironmentOption,\n  AssignEnvironmentsDialogValue,\n  AssignEnvironmentsMode,\n} from './assign-environments-dialog.models';\n\n@Component({\n  selector: 'cqa-assign-environments-dialog',\n  templateUrl: './assign-environments-dialog.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' },\n})\nexport class AssignEnvironmentsDialogComponent implements OnInit {\n  @Input() mode: AssignEnvironmentsMode = 'assign';\n  @Input() profileName: string = '';\n  @Input() environments: AssignEnvironmentOption[] = [];\n  @Input() assignedEnvironments: AssignEnvironmentOption[] = [];\n  @Input() defaultSourceEnvId?: number;\n\n  public readonly selected = new Set<number>();\n\n  public readonly sourceForm = new FormGroup({\n    sourceEnvId: new FormControl(null),\n  });\n  public sourceConfig: DynamicSelectFieldConfig = this.buildSourceConfig([]);\n\n  public sourceError: string | null = null;\n\n  constructor(private readonly cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    // Source picker appears in both modes when there are already-attached envs\n    // to copy from. In clone mode it's required; in assign mode it's optional\n    // (falling back to the base profile rows when no source is picked).\n    if (this.showSourcePicker) {\n      const initial = this.defaultSourceEnvId != null\n        ? this.defaultSourceEnvId\n        : (this.mode === 'clone' && this.assignedEnvironments && this.assignedEnvironments.length\n          ? this.assignedEnvironments[0].id\n          : null);\n      this.sourceForm.get('sourceEnvId')!.setValue(initial);\n      this.sourceConfig = this.buildSourceConfig(this.assignedEnvironments ?? []);\n    }\n  }\n\n  public get showSourcePicker(): boolean {\n    return (this.assignedEnvironments && this.assignedEnvironments.length > 0);\n  }\n\n  public get sourceLabel(): string {\n    return this.mode === 'clone' ? 'Copy from which environment' : 'Copy rows from (optional)';\n  }\n\n  public get sourceHelper(): string {\n    if (this.sourceError) { return this.sourceError; }\n    return this.mode === 'clone'\n      ? \"The selected environment's rows will be copied into each target environment below.\"\n      : 'Pick an environment to copy its rows into the new ones. Leave blank to seed with the base profile rows.';\n  }\n\n  public get title(): string {\n    return this.mode === 'clone' ? 'Clone to other environments' : 'Assign to environments';\n  }\n\n  public get subtitle(): string {\n    const name = this.profileName ? `\"${this.profileName}\"` : 'this profile';\n    return this.mode === 'clone'\n      ? `Duplicate ${name} into other environments. Column structure is shared; row values are independent per environment.`\n      : `Make ${name} available in the selected environments. Columns are shared — each environment gets an independent row set.`;\n  }\n\n  public get primaryButtonLabel(): string {\n    const count = this.selected.size;\n    const verb = this.mode === 'clone' ? 'Clone to' : 'Assign to';\n    return `${verb} ${count} environment${count === 1 ? '' : 's'}`;\n  }\n\n  public get primaryDisabled(): boolean {\n    if (this.selected.size === 0) { return true; }\n    if (this.mode === 'clone' && this.sourceForm.get('sourceEnvId')!.value == null) { return true; }\n    return false;\n  }\n\n  public get selectableEnvironments(): AssignEnvironmentOption[] {\n    return this.environments ?? [];\n  }\n\n  public get hasAnySelectable(): boolean {\n    return (this.environments ?? []).some(e => !e.alreadyAssigned);\n  }\n\n  public get sourceValue(): number | null {\n    return this.sourceForm.get('sourceEnvId')!.value ?? null;\n  }\n\n  public isSelected(id: number): boolean {\n    return this.selected.has(id);\n  }\n\n  public toggle(env: AssignEnvironmentOption): void {\n    if (env.alreadyAssigned) { return; }\n    if (this.selected.has(env.id)) {\n      this.selected.delete(env.id);\n    } else {\n      this.selected.add(env.id);\n    }\n    this.cdr.markForCheck();\n  }\n\n  public helperFor(env: AssignEnvironmentOption): string {\n    if (env.alreadyAssigned) { return 'Already assigned to this environment'; }\n    return env.description || '';\n  }\n\n  public getValue(): AssignEnvironmentsDialogValue | null {\n    if (this.selected.size === 0) { return null; }\n    const src = this.sourceValue;\n    if (this.mode === 'clone') {\n      if (src == null) {\n        this.sourceError = 'Pick the environment whose rows should be copied.';\n        this.cdr.markForCheck();\n        return null;\n      }\n      this.sourceError = null;\n      return { selectedIds: Array.from(this.selected), sourceEnvId: Number(src) };\n    }\n    // Assign mode: source is optional — include it when the user picked one.\n    this.sourceError = null;\n    return src != null\n      ? { selectedIds: Array.from(this.selected), sourceEnvId: Number(src) }\n      : { selectedIds: Array.from(this.selected) };\n  }\n\n  private buildSourceConfig(envs: AssignEnvironmentOption[]): DynamicSelectFieldConfig {\n    const options: SelectOption[] = (envs || []).map(e => ({\n      id: e.id,\n      value: e.id,\n      name: e.name,\n      label: e.name,\n      statusColor: e.color,\n    }));\n    return {\n      key: 'sourceEnvId',\n      label: '',\n      placeholder: 'Select source environment…',\n      multiple: false,\n      searchable: true,\n      options,\n    };\n  }\n}\n","<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n  <!-- Source env picker: required in clone mode, optional in assign mode -->\n  <div *ngIf=\"showSourcePicker\" class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n    <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">{{ sourceLabel }}</label>\n    <cqa-dynamic-select\n      [form]=\"sourceForm\"\n      [config]=\"sourceConfig\">\n    </cqa-dynamic-select>\n    <span [class.cqa-text-red-600]=\"sourceError\" [class.cqa-text-gray-500]=\"!sourceError\" class=\"cqa-text-xs\">\n      {{ sourceHelper }}\n    </span>\n  </div>\n\n  <!-- Selectable envs -->\n  <div\n    *ngIf=\"hasAnySelectable || (environments?.length || 0) > 0\"\n    class=\"cqa-flex cqa-flex-col cqa-gap-2 cqa-max-h-[340px] cqa-overflow-y-auto\">\n    <label\n      *ngFor=\"let env of selectableEnvironments\"\n      class=\"cqa-flex cqa-items-center cqa-gap-3 cqa-px-3 cqa-py-2.5 cqa-border cqa-rounded-[10px] cqa-bg-white cqa-transition-colors\"\n      [class.cqa-border-gray-200]=\"!isSelected(env.id) && !env.alreadyAssigned\"\n      [class.hover:cqa-border-indigo-200]=\"!env.alreadyAssigned && !isSelected(env.id)\"\n      [class.cqa-border-indigo-400]=\"isSelected(env.id)\"\n      [class.cqa-bg-indigo-50]=\"isSelected(env.id)\"\n      [class.cqa-opacity-50]=\"env.alreadyAssigned\"\n      [class.cqa-cursor-pointer]=\"!env.alreadyAssigned\"\n      [class.cqa-cursor-not-allowed]=\"env.alreadyAssigned\"\n      (click)=\"toggle(env)\">\n      <input\n        type=\"checkbox\"\n        class=\"cqa-w-4 cqa-h-4 cqa-cursor-pointer\"\n        [checked]=\"isSelected(env.id)\"\n        [disabled]=\"!!env.alreadyAssigned\"\n        (click)=\"$event.stopPropagation()\"\n        (change)=\"toggle(env)\" />\n\n      <span\n        class=\"cqa-inline-block cqa-w-2 cqa-h-2 cqa-rounded-full cqa-flex-none\"\n        [style.background]=\"env.color || '#3F43EE'\"></span>\n\n      <div class=\"cqa-flex cqa-flex-col cqa-min-w-0 cqa-flex-1\">\n        <div class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">{{ env.name }}</div>\n        <div *ngIf=\"helperFor(env)\" class=\"cqa-text-xs cqa-text-gray-500 cqa-leading-[1.4]\">{{ helperFor(env) }}</div>\n      </div>\n    </label>\n  </div>\n\n  <div\n    *ngIf=\"(environments?.length || 0) === 0\"\n    class=\"cqa-py-8 cqa-px-4 cqa-text-center cqa-text-sm cqa-text-gray-500\">\n    All environments are already assigned to this profile.\n  </div>\n</div>\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXNzaWduLWVudmlyb25tZW50cy1kaWFsb2cubW9kZWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9hc3NpZ24tZW52aXJvbm1lbnRzLWRpYWxvZy9hc3NpZ24tZW52aXJvbm1lbnRzLWRpYWxvZy5tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIEFzc2lnbkVudmlyb25tZW50c01vZGUgPSAnYXNzaWduJyB8ICdjbG9uZSc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgQXNzaWduRW52aXJvbm1lbnRPcHRpb24ge1xuICBpZDogbnVtYmVyO1xuICBuYW1lOiBzdHJpbmc7XG4gIGNvbG9yPzogc3RyaW5nO1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgYWxyZWFkeUFzc2lnbmVkPzogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBBc3NpZ25FbnZpcm9ubWVudHNEaWFsb2dWYWx1ZSB7XG4gIHNlbGVjdGVkSWRzOiBudW1iZXJbXTtcbiAgc291cmNlRW52SWQ/OiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQXNzaWduRW52aXJvbm1lbnRzRGlhbG9nSW5wdXRzIHtcbiAgbW9kZT86IEFzc2lnbkVudmlyb25tZW50c01vZGU7XG4gIHByb2ZpbGVOYW1lPzogc3RyaW5nO1xuICBlbnZpcm9ubWVudHM/OiBBc3NpZ25FbnZpcm9ubWVudE9wdGlvbltdO1xuICBhc3NpZ25lZEVudmlyb25tZW50cz86IEFzc2lnbkVudmlyb25tZW50T3B0aW9uW107XG4gIGRlZmF1bHRTb3VyY2VFbnZJZD86IG51bWJlcjtcbn1cbiJdfQ==
@@ -0,0 +1,123 @@
1
+ import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
2
+ import { ENVIRONMENT_ACCENT_COLORS, } from './new-environment-dialog.models';
3
+ import * as i0 from "@angular/core";
4
+ import * as i1 from "../custom-input/custom-input.component";
5
+ import * as i2 from "../custom-toggle/custom-toggle.component";
6
+ import * as i3 from "@angular/common";
7
+ export class NewEnvironmentDialogComponent {
8
+ constructor(cdr) {
9
+ this.cdr = cdr;
10
+ this.mode = 'create';
11
+ this.existingNames = [];
12
+ this.allowDefaultToggle = true;
13
+ this.defaultAlreadySet = false;
14
+ this.colors = ENVIRONMENT_ACCENT_COLORS;
15
+ this.name = '';
16
+ this.description = '';
17
+ this.color = ENVIRONMENT_ACCENT_COLORS[0];
18
+ this.isDefault = false;
19
+ this.nameError = null;
20
+ }
21
+ ngOnInit() {
22
+ if (this.initialValue) {
23
+ this.name = this.initialValue.name ?? '';
24
+ this.description = this.initialValue.description ?? '';
25
+ this.color = this.initialValue.color || ENVIRONMENT_ACCENT_COLORS[0];
26
+ this.isDefault = this.initialValue.isDefault ?? false;
27
+ }
28
+ }
29
+ get title() {
30
+ return this.mode === 'edit' ? 'Edit environment' : 'New environment';
31
+ }
32
+ get subtitle() {
33
+ return 'Create an environment to scope variables, DB configs, and data profiles.';
34
+ }
35
+ get primaryButtonLabel() {
36
+ return this.mode === 'edit' ? 'Save changes' : 'Create environment';
37
+ }
38
+ get nameErrorsArray() {
39
+ return this.nameError ? [this.nameError] : [];
40
+ }
41
+ get defaultHelperText() {
42
+ if (!this.allowDefaultToggle) {
43
+ return 'Default promotion is handled from the environment detail page.';
44
+ }
45
+ if (this.defaultAlreadySet && !this.isDefault) {
46
+ return 'Another environment is currently the default. Toggling this on will demote it.';
47
+ }
48
+ return 'Test cases without an explicit env selected will resolve against this one.';
49
+ }
50
+ onNameChange(next) {
51
+ this.name = next;
52
+ this.nameError = null;
53
+ this.cdr.markForCheck();
54
+ }
55
+ onDescriptionChange(next) {
56
+ this.description = next;
57
+ this.cdr.markForCheck();
58
+ }
59
+ onColorSelect(next) {
60
+ this.color = next;
61
+ this.cdr.markForCheck();
62
+ }
63
+ onDefaultChange(next) {
64
+ this.isDefault = next;
65
+ this.cdr.markForCheck();
66
+ }
67
+ /**
68
+ * Called by the host (via DialogRef.getComponentInstance()) when the primary
69
+ * button is clicked. Returns the captured value when valid, or null when the
70
+ * form has errors — host should leave the dialog open.
71
+ */
72
+ getValue() {
73
+ const trimmedName = this.name.trim();
74
+ if (!trimmedName) {
75
+ this.nameError = 'Name is required.';
76
+ }
77
+ else if (trimmedName.length < 2) {
78
+ this.nameError = 'Name must be at least 2 characters.';
79
+ }
80
+ else if (this.isDuplicateName(trimmedName)) {
81
+ this.nameError = 'An environment with this name already exists — names must be unique.';
82
+ }
83
+ else {
84
+ this.nameError = null;
85
+ }
86
+ if (this.nameError) {
87
+ this.cdr.markForCheck();
88
+ return null;
89
+ }
90
+ return {
91
+ name: trimmedName,
92
+ description: this.description ? this.description.trim() : null,
93
+ color: this.color,
94
+ isDefault: this.allowDefaultToggle ? this.isDefault : false,
95
+ };
96
+ }
97
+ isDuplicateName(candidate) {
98
+ const existing = (this.existingNames ?? []).map(n => (n ?? '').trim().toLowerCase());
99
+ const lowered = candidate.toLowerCase();
100
+ if (this.mode === 'edit') {
101
+ const original = (this.initialValue?.name ?? '').trim().toLowerCase();
102
+ return lowered !== original && existing.includes(lowered);
103
+ }
104
+ return existing.includes(lowered);
105
+ }
106
+ }
107
+ NewEnvironmentDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NewEnvironmentDialogComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
108
+ NewEnvironmentDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: NewEnvironmentDialogComponent, selector: "cqa-new-environment-dialog", inputs: { mode: "mode", initialValue: "initialValue", existingNames: "existingNames", allowDefaultToggle: "allowDefaultToggle", defaultAlreadySet: "defaultAlreadySet" }, host: { styleAttribute: "display:block;width:100%;", classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n <!-- Name -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Environment name\"\n placeholder=\"e.g. QA, SIT, UAT, Prod, Demo\"\n type=\"text\"\n [value]=\"name\"\n [required]=\"true\"\n [fullWidth]=\"true\"\n [errors]=\"nameErrorsArray\"\n (valueChange)=\"onNameChange($event)\">\n </cqa-custom-input>\n <span *ngIf=\"!nameError\"\n class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Names are user-defined. Be concise \u2014 you'll see this on every test run.\n </span>\n </div>\n\n <!-- Description -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Description\"\n placeholder=\"What is this environment for?\"\n type=\"text\"\n [value]=\"description\"\n [fullWidth]=\"true\"\n (valueChange)=\"onDescriptionChange($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- Accent color -->\n <div class=\"cqa-flex cqa-flex-col\">\n <label class=\"cqa-text-sm cqa-mb-2 cqa-font-medium cqa-text-gray-700\">Accent color</label>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <button\n *ngFor=\"let swatch of colors\"\n type=\"button\"\n class=\"cqa-w-7 cqa-h-7 cqa-rounded-lg cqa-cursor-pointer cqa-transition-all\"\n [attr.aria-label]=\"'Select ' + swatch\"\n [attr.aria-pressed]=\"color === swatch\"\n [style.background]=\"swatch\"\n [style.border]=\"color === swatch ? '2px solid #0F172A' : '2px solid transparent'\"\n [style.boxShadow]=\"color === swatch ? '0 0 0 2px rgba(15,23,42,0.12)' : '0 0 0 1px rgba(0,0,0,0.05)'\"\n (click)=\"onColorSelect(swatch)\">\n </button>\n </div>\n </div>\n\n <!-- Make default -->\n <div\n *ngIf=\"allowDefaultToggle\"\n class=\"cqa-flex cqa-items-start cqa-gap-3 cqa-p-3 cqa-rounded-lg cqa-border cqa-border-indigo-100 cqa-bg-indigo-50\">\n <cqa-custom-toggle\n [checked]=\"isDefault\"\n ariaLabel=\"Make default environment\"\n (checkedChange)=\"onDefaultChange($event)\">\n </cqa-custom-toggle>\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1\">\n <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">\n Make this the default environment\n </span>\n <span class=\"cqa-text-xs cqa-text-gray-600 cqa-mt-0.5\">\n {{ defaultHelperText }}\n </span>\n </div>\n </div>\n\n</div>\n", components: [{ type: i1.CustomInputComponent, selector: "cqa-custom-input", inputs: ["inputId", "label", "type", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "inputInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused", "enterPressed"] }, { type: i2.CustomToggleComponent, selector: "cqa-custom-toggle", inputs: ["checked", "disabled", "ariaLabel"], outputs: ["checkedChange", "change"] }], directives: [{ type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
109
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NewEnvironmentDialogComponent, decorators: [{
110
+ type: Component,
111
+ args: [{ selector: 'cqa-new-environment-dialog', changeDetection: ChangeDetectionStrategy.OnPush, host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n <!-- Name -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Environment name\"\n placeholder=\"e.g. QA, SIT, UAT, Prod, Demo\"\n type=\"text\"\n [value]=\"name\"\n [required]=\"true\"\n [fullWidth]=\"true\"\n [errors]=\"nameErrorsArray\"\n (valueChange)=\"onNameChange($event)\">\n </cqa-custom-input>\n <span *ngIf=\"!nameError\"\n class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n Names are user-defined. Be concise \u2014 you'll see this on every test run.\n </span>\n </div>\n\n <!-- Description -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Description\"\n placeholder=\"What is this environment for?\"\n type=\"text\"\n [value]=\"description\"\n [fullWidth]=\"true\"\n (valueChange)=\"onDescriptionChange($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- Accent color -->\n <div class=\"cqa-flex cqa-flex-col\">\n <label class=\"cqa-text-sm cqa-mb-2 cqa-font-medium cqa-text-gray-700\">Accent color</label>\n <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <button\n *ngFor=\"let swatch of colors\"\n type=\"button\"\n class=\"cqa-w-7 cqa-h-7 cqa-rounded-lg cqa-cursor-pointer cqa-transition-all\"\n [attr.aria-label]=\"'Select ' + swatch\"\n [attr.aria-pressed]=\"color === swatch\"\n [style.background]=\"swatch\"\n [style.border]=\"color === swatch ? '2px solid #0F172A' : '2px solid transparent'\"\n [style.boxShadow]=\"color === swatch ? '0 0 0 2px rgba(15,23,42,0.12)' : '0 0 0 1px rgba(0,0,0,0.05)'\"\n (click)=\"onColorSelect(swatch)\">\n </button>\n </div>\n </div>\n\n <!-- Make default -->\n <div\n *ngIf=\"allowDefaultToggle\"\n class=\"cqa-flex cqa-items-start cqa-gap-3 cqa-p-3 cqa-rounded-lg cqa-border cqa-border-indigo-100 cqa-bg-indigo-50\">\n <cqa-custom-toggle\n [checked]=\"isDefault\"\n ariaLabel=\"Make default environment\"\n (checkedChange)=\"onDefaultChange($event)\">\n </cqa-custom-toggle>\n <div class=\"cqa-flex cqa-flex-col cqa-flex-1\">\n <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">\n Make this the default environment\n </span>\n <span class=\"cqa-text-xs cqa-text-gray-600 cqa-mt-0.5\">\n {{ defaultHelperText }}\n </span>\n </div>\n </div>\n\n</div>\n" }]
112
+ }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { mode: [{
113
+ type: Input
114
+ }], initialValue: [{
115
+ type: Input
116
+ }], existingNames: [{
117
+ type: Input
118
+ }], allowDefaultToggle: [{
119
+ type: Input
120
+ }], defaultAlreadySet: [{
121
+ type: Input
122
+ }] } });
123
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"new-environment-dialog.component.js","sourceRoot":"","sources":["../../../../../src/lib/new-environment-dialog/new-environment-dialog.component.ts","../../../../../src/lib/new-environment-dialog/new-environment-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Input, OnInit } from '@angular/core';\nimport {\n  ENVIRONMENT_ACCENT_COLORS,\n  EnvironmentDialogValue,\n} from './new-environment-dialog.models';\n\n@Component({\n  selector: 'cqa-new-environment-dialog',\n  templateUrl: './new-environment-dialog.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' },\n})\nexport class NewEnvironmentDialogComponent implements OnInit {\n  @Input() mode: 'create' | 'edit' = 'create';\n  @Input() initialValue?: Partial<EnvironmentDialogValue>;\n  @Input() existingNames: string[] = [];\n  @Input() allowDefaultToggle: boolean = true;\n  @Input() defaultAlreadySet: boolean = false;\n\n  readonly colors: readonly string[] = ENVIRONMENT_ACCENT_COLORS;\n\n  name = '';\n  description = '';\n  color: string = ENVIRONMENT_ACCENT_COLORS[0];\n  isDefault = false;\n\n  nameError: string | null = null;\n\n  constructor(private readonly cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    if (this.initialValue) {\n      this.name = this.initialValue.name ?? '';\n      this.description = this.initialValue.description ?? '';\n      this.color = this.initialValue.color || ENVIRONMENT_ACCENT_COLORS[0];\n      this.isDefault = this.initialValue.isDefault ?? false;\n    }\n  }\n\n  get title(): string {\n    return this.mode === 'edit' ? 'Edit environment' : 'New environment';\n  }\n\n  get subtitle(): string {\n    return 'Create an environment to scope variables, DB configs, and data profiles.';\n  }\n\n  get primaryButtonLabel(): string {\n    return this.mode === 'edit' ? 'Save changes' : 'Create environment';\n  }\n\n  get nameErrorsArray(): string[] {\n    return this.nameError ? [this.nameError] : [];\n  }\n\n  get defaultHelperText(): string {\n    if (!this.allowDefaultToggle) {\n      return 'Default promotion is handled from the environment detail page.';\n    }\n    if (this.defaultAlreadySet && !this.isDefault) {\n      return 'Another environment is currently the default. Toggling this on will demote it.';\n    }\n    return 'Test cases without an explicit env selected will resolve against this one.';\n  }\n\n  onNameChange(next: string): void {\n    this.name = next;\n    this.nameError = null;\n    this.cdr.markForCheck();\n  }\n\n  onDescriptionChange(next: string): void {\n    this.description = next;\n    this.cdr.markForCheck();\n  }\n\n  onColorSelect(next: string): void {\n    this.color = next;\n    this.cdr.markForCheck();\n  }\n\n  onDefaultChange(next: boolean): void {\n    this.isDefault = next;\n    this.cdr.markForCheck();\n  }\n\n  /**\n   * Called by the host (via DialogRef.getComponentInstance()) when the primary\n   * button is clicked. Returns the captured value when valid, or null when the\n   * form has errors — host should leave the dialog open.\n   */\n  getValue(): EnvironmentDialogValue | null {\n    const trimmedName = this.name.trim();\n\n    if (!trimmedName) {\n      this.nameError = 'Name is required.';\n    } else if (trimmedName.length < 2) {\n      this.nameError = 'Name must be at least 2 characters.';\n    } else if (this.isDuplicateName(trimmedName)) {\n      this.nameError = 'An environment with this name already exists — names must be unique.';\n    } else {\n      this.nameError = null;\n    }\n\n    if (this.nameError) {\n      this.cdr.markForCheck();\n      return null;\n    }\n\n    return {\n      name: trimmedName,\n      description: this.description ? this.description.trim() : null,\n      color: this.color,\n      isDefault: this.allowDefaultToggle ? this.isDefault : false,\n    };\n  }\n\n  private isDuplicateName(candidate: string): boolean {\n    const existing = (this.existingNames ?? []).map(n => (n ?? '').trim().toLowerCase());\n    const lowered = candidate.toLowerCase();\n    if (this.mode === 'edit') {\n      const original = (this.initialValue?.name ?? '').trim().toLowerCase();\n      return lowered !== original && existing.includes(lowered);\n    }\n    return existing.includes(lowered);\n  }\n}\n","<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n  <!-- Name -->\n  <div class=\"cqa-flex cqa-flex-col\">\n    <cqa-custom-input\n      label=\"Environment name\"\n      placeholder=\"e.g. QA, SIT, UAT, Prod, Demo\"\n      type=\"text\"\n      [value]=\"name\"\n      [required]=\"true\"\n      [fullWidth]=\"true\"\n      [errors]=\"nameErrorsArray\"\n      (valueChange)=\"onNameChange($event)\">\n    </cqa-custom-input>\n    <span *ngIf=\"!nameError\"\n      class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n      Names are user-defined. Be concise — you'll see this on every test run.\n    </span>\n  </div>\n\n  <!-- Description -->\n  <div class=\"cqa-flex cqa-flex-col\">\n    <cqa-custom-input\n      label=\"Description\"\n      placeholder=\"What is this environment for?\"\n      type=\"text\"\n      [value]=\"description\"\n      [fullWidth]=\"true\"\n      (valueChange)=\"onDescriptionChange($event)\">\n    </cqa-custom-input>\n  </div>\n\n  <!-- Accent color -->\n  <div class=\"cqa-flex cqa-flex-col\">\n    <label class=\"cqa-text-sm cqa-mb-2 cqa-font-medium cqa-text-gray-700\">Accent color</label>\n    <div class=\"cqa-flex cqa-items-center cqa-gap-2\">\n      <button\n        *ngFor=\"let swatch of colors\"\n        type=\"button\"\n        class=\"cqa-w-7 cqa-h-7 cqa-rounded-lg cqa-cursor-pointer cqa-transition-all\"\n        [attr.aria-label]=\"'Select ' + swatch\"\n        [attr.aria-pressed]=\"color === swatch\"\n        [style.background]=\"swatch\"\n        [style.border]=\"color === swatch ? '2px solid #0F172A' : '2px solid transparent'\"\n        [style.boxShadow]=\"color === swatch ? '0 0 0 2px rgba(15,23,42,0.12)' : '0 0 0 1px rgba(0,0,0,0.05)'\"\n        (click)=\"onColorSelect(swatch)\">\n      </button>\n    </div>\n  </div>\n\n  <!-- Make default -->\n  <div\n    *ngIf=\"allowDefaultToggle\"\n    class=\"cqa-flex cqa-items-start cqa-gap-3 cqa-p-3 cqa-rounded-lg cqa-border cqa-border-indigo-100 cqa-bg-indigo-50\">\n    <cqa-custom-toggle\n      [checked]=\"isDefault\"\n      ariaLabel=\"Make default environment\"\n      (checkedChange)=\"onDefaultChange($event)\">\n    </cqa-custom-toggle>\n    <div class=\"cqa-flex cqa-flex-col cqa-flex-1\">\n      <span class=\"cqa-text-sm cqa-font-medium cqa-text-gray-900\">\n        Make this the default environment\n      </span>\n      <span class=\"cqa-text-xs cqa-text-gray-600 cqa-mt-0.5\">\n        {{ defaultHelperText }}\n      </span>\n    </div>\n  </div>\n\n</div>\n"]}
@@ -0,0 +1,9 @@
1
+ export const ENVIRONMENT_ACCENT_COLORS = [
2
+ '#3F43EE',
3
+ '#A855F7',
4
+ '#00A63E',
5
+ '#F59E0B',
6
+ '#0EA5E9',
7
+ '#EC4899',
8
+ ];
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmV3LWVudmlyb25tZW50LWRpYWxvZy5tb2RlbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL25ldy1lbnZpcm9ubWVudC1kaWFsb2cvbmV3LWVudmlyb25tZW50LWRpYWxvZy5tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBZUEsTUFBTSxDQUFDLE1BQU0seUJBQXlCLEdBQXNCO0lBQzFELFNBQVM7SUFDVCxTQUFTO0lBQ1QsU0FBUztJQUNULFNBQVM7SUFDVCxTQUFTO0lBQ1QsU0FBUztDQUNWLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgaW50ZXJmYWNlIEVudmlyb25tZW50RGlhbG9nVmFsdWUge1xuICBuYW1lOiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmcgfCBudWxsO1xuICBjb2xvcjogc3RyaW5nO1xuICBpc0RlZmF1bHQ6IGJvb2xlYW47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTmV3RW52aXJvbm1lbnREaWFsb2dJbnB1dHMge1xuICBtb2RlPzogJ2NyZWF0ZScgfCAnZWRpdCc7XG4gIGluaXRpYWxWYWx1ZT86IFBhcnRpYWw8RW52aXJvbm1lbnREaWFsb2dWYWx1ZT47XG4gIGV4aXN0aW5nTmFtZXM/OiBzdHJpbmdbXTtcbiAgYWxsb3dEZWZhdWx0VG9nZ2xlPzogYm9vbGVhbjtcbiAgZGVmYXVsdEFscmVhZHlTZXQ/OiBib29sZWFuO1xufVxuXG5leHBvcnQgY29uc3QgRU5WSVJPTk1FTlRfQUNDRU5UX0NPTE9SUzogcmVhZG9ubHkgc3RyaW5nW10gPSBbXG4gICcjM0Y0M0VFJyxcbiAgJyNBODU1RjcnLFxuICAnIzAwQTYzRScsXG4gICcjRjU5RTBCJyxcbiAgJyMwRUE1RTknLFxuICAnI0VDNDg5OScsXG5dO1xuIl19
@@ -0,0 +1,190 @@
1
+ import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "../custom-input/custom-input.component";
4
+ import * as i2 from "../dropdown-button/dropdown-button.component";
5
+ import * as i3 from "../permission-toggle/permission-toggle.component";
6
+ import * as i4 from "@angular/common";
7
+ export class NewEnvironmentVariableDialogComponent {
8
+ constructor(cdr) {
9
+ this.cdr = cdr;
10
+ this.mode = 'create';
11
+ this.existingNames = [];
12
+ this.envName = '';
13
+ this.typeDropdownOptions = [
14
+ { label: 'String', value: 'String' },
15
+ { label: 'Number', value: 'Number' },
16
+ { label: 'Boolean', value: 'Boolean' },
17
+ { label: 'Password', value: 'Password' },
18
+ ];
19
+ this.name = '';
20
+ this.type = 'String';
21
+ this.value = '';
22
+ this.readWriteMode = 'RW';
23
+ this.passwordTouched = false;
24
+ this.nameError = null;
25
+ this.valueError = null;
26
+ }
27
+ ngOnInit() {
28
+ if (this.initialValue) {
29
+ this.name = this.initialValue.name ?? '';
30
+ this.type = this.initialValue.type ?? 'String';
31
+ this.value = this.initialValue.value ?? '';
32
+ this.readWriteMode = this.initialValue.readWriteMode ?? 'RW';
33
+ }
34
+ if (this.type === 'Boolean' && this.value !== 'true' && this.value !== 'false') {
35
+ this.value = 'false';
36
+ }
37
+ if (this.mode === 'edit' && this.type === 'Password') {
38
+ this.value = '';
39
+ }
40
+ }
41
+ get title() {
42
+ return this.mode === 'edit' ? 'Edit variable' : 'New variable';
43
+ }
44
+ get subtitle() {
45
+ return this.envName ? `Scoped to the ${this.envName} environment.` : 'Scoped to this environment.';
46
+ }
47
+ get primaryButtonLabel() {
48
+ return this.mode === 'edit' ? 'Save changes' : 'Add variable';
49
+ }
50
+ get isBoolean() { return this.type === 'Boolean'; }
51
+ get isPassword() { return this.type === 'Password'; }
52
+ get isNumber() { return this.type === 'Number'; }
53
+ get valuePlaceholder() {
54
+ if (this.isPassword && this.mode === 'edit') {
55
+ return '•••••• (leave blank to keep existing)';
56
+ }
57
+ if (this.readWriteMode === 'RW') {
58
+ return 'Leave blank to populate during test execution';
59
+ }
60
+ return 'Enter a value';
61
+ }
62
+ get nameHelperText() {
63
+ if (this.nameError) {
64
+ return this.nameError;
65
+ }
66
+ return 'Variable names must be unique within the environment.';
67
+ }
68
+ get permissionHelperText() {
69
+ return this.readWriteMode === 'RO'
70
+ ? 'Read only — test steps can read but not overwrite.'
71
+ : 'Read/write — test cases can capture values here.';
72
+ }
73
+ get nameErrorsArray() { return this.nameError ? [this.nameError] : []; }
74
+ get valueErrorsArray() { return this.valueError ? [this.valueError] : []; }
75
+ get booleanDropdownOptions() {
76
+ return [
77
+ { label: 'true', value: 'true' },
78
+ { label: 'false', value: 'false' },
79
+ ];
80
+ }
81
+ onNameChange(next) {
82
+ this.name = next;
83
+ this.nameError = null;
84
+ this.cdr.markForCheck();
85
+ }
86
+ onTypeChange(next) {
87
+ if (!next) {
88
+ return;
89
+ }
90
+ this.type = next;
91
+ if (this.type === 'Boolean' && this.value !== 'true' && this.value !== 'false') {
92
+ this.value = 'false';
93
+ }
94
+ if (this.type === 'Password') {
95
+ this.passwordTouched = false;
96
+ if (this.mode === 'edit') {
97
+ this.value = '';
98
+ }
99
+ }
100
+ this.valueError = null;
101
+ this.cdr.markForCheck();
102
+ }
103
+ onPermissionChange(next) {
104
+ this.readWriteMode = next === 'RO' ? 'RO' : 'RW';
105
+ this.cdr.markForCheck();
106
+ }
107
+ onValueChange(next) {
108
+ this.value = next;
109
+ if (this.isPassword) {
110
+ this.passwordTouched = true;
111
+ }
112
+ this.valueError = null;
113
+ this.cdr.markForCheck();
114
+ }
115
+ onBooleanValueChange(next) {
116
+ this.value = next;
117
+ this.cdr.markForCheck();
118
+ }
119
+ /**
120
+ * Host reads via DialogRef.getComponentInstance(). Returns null on validation
121
+ * failure — host keeps the dialog open and surfaces the inline error already
122
+ * written onto nameError / valueError.
123
+ */
124
+ getValue() {
125
+ const trimmedName = this.name.trim();
126
+ if (!trimmedName) {
127
+ this.nameError = 'Name is required.';
128
+ }
129
+ else if (this.isDuplicateName(trimmedName)) {
130
+ this.nameError = 'A variable with this name already exists in this environment.';
131
+ }
132
+ else {
133
+ this.nameError = null;
134
+ }
135
+ if (this.isNumber && this.value && !this.isValidNumber(this.value)) {
136
+ this.valueError = 'Value must be a number.';
137
+ }
138
+ else {
139
+ this.valueError = null;
140
+ }
141
+ if (this.nameError || this.valueError) {
142
+ this.cdr.markForCheck();
143
+ return null;
144
+ }
145
+ const omitPasswordValue = this.isPassword && this.mode === 'edit' && !this.passwordTouched;
146
+ return {
147
+ name: trimmedName,
148
+ type: this.type,
149
+ value: omitPasswordValue ? null : this.serializeValue(),
150
+ readWriteMode: this.readWriteMode,
151
+ };
152
+ }
153
+ serializeValue() {
154
+ if (this.isBoolean) {
155
+ return this.value === 'true' ? 'true' : 'false';
156
+ }
157
+ return this.value ?? '';
158
+ }
159
+ isDuplicateName(candidate) {
160
+ const existing = (this.existingNames ?? []).map(n => (n ?? '').trim().toLowerCase());
161
+ const lowered = candidate.toLowerCase();
162
+ if (this.mode === 'edit') {
163
+ const original = (this.initialValue?.name ?? '').trim().toLowerCase();
164
+ return lowered !== original && existing.includes(lowered);
165
+ }
166
+ return existing.includes(lowered);
167
+ }
168
+ isValidNumber(raw) {
169
+ const trimmed = raw.trim();
170
+ if (!trimmed) {
171
+ return true;
172
+ }
173
+ return !Number.isNaN(Number(trimmed));
174
+ }
175
+ }
176
+ NewEnvironmentVariableDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NewEnvironmentVariableDialogComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
177
+ NewEnvironmentVariableDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: NewEnvironmentVariableDialogComponent, selector: "cqa-new-environment-variable-dialog", inputs: { mode: "mode", initialValue: "initialValue", existingNames: "existingNames", envName: "envName" }, host: { styleAttribute: "display:block;width:100%;", classAttribute: "cqa-ui-root" }, ngImport: i0, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n <!-- Name -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Name\"\n placeholder=\"e.g. order_id\"\n type=\"text\"\n [value]=\"name\"\n [required]=\"true\"\n [fullWidth]=\"true\"\n [errors]=\"nameErrorsArray\"\n inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onNameChange($event)\">\n </cqa-custom-input>\n <span *ngIf=\"!nameError\"\n class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n {{ nameHelperText }}\n </span>\n </div>\n\n <!-- Type + Permission -->\n <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n <div class=\"cqa-flex cqa-flex-col\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Type</label>\n <cqa-dropdown-button\n [options]=\"typeDropdownOptions\"\n [selectedValue]=\"type\"\n (selectionChange)=\"onTypeChange($event)\">\n </cqa-dropdown-button>\n </div>\n <div class=\"cqa-flex cqa-flex-col\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Permission</label>\n <div class=\"cqa-pt-0.5\">\n <cqa-permission-toggle\n [value]=\"readWriteMode\"\n (valueChange)=\"onPermissionChange($event)\">\n </cqa-permission-toggle>\n </div>\n <span class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">{{ permissionHelperText }}</span>\n </div>\n </div>\n\n <!-- Value -->\n <div class=\"cqa-flex cqa-flex-col\">\n <ng-container *ngIf=\"isBoolean; else nonBooleanValue\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Value</label>\n <cqa-dropdown-button\n [options]=\"booleanDropdownOptions\"\n [selectedValue]=\"value\"\n (selectionChange)=\"onBooleanValueChange($event)\">\n </cqa-dropdown-button>\n </ng-container>\n <ng-template #nonBooleanValue>\n <cqa-custom-input\n label=\"Value\"\n [type]=\"isPassword ? 'password' : 'text'\"\n [value]=\"value\"\n [placeholder]=\"valuePlaceholder\"\n [fullWidth]=\"true\"\n [errors]=\"valueErrorsArray\"\n inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onValueChange($event)\">\n </cqa-custom-input>\n </ng-template>\n </div>\n\n</div>\n", components: [{ type: i1.CustomInputComponent, selector: "cqa-custom-input", inputs: ["inputId", "label", "type", "placeholder", "value", "disabled", "errors", "required", "ariaLabel", "size", "fullWidth", "maxLength", "showCharCount", "inputInlineStyle", "labelInlineStyle"], outputs: ["valueChange", "blurred", "focused", "enterPressed"] }, { type: i2.DropdownButtonComponent, selector: "cqa-dropdown-button", inputs: ["label", "options", "selectedValue", "disabled", "placeholder"], outputs: ["selectionChange", "opened", "closed"] }, { type: i3.PermissionToggleComponent, selector: "cqa-permission-toggle", inputs: ["value", "disabled", "roTooltip", "rwTooltip"], outputs: ["valueChange"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
178
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NewEnvironmentVariableDialogComponent, decorators: [{
179
+ type: Component,
180
+ args: [{ selector: 'cqa-new-environment-variable-dialog', changeDetection: ChangeDetectionStrategy.OnPush, host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' }, template: "<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n <!-- Name -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Name\"\n placeholder=\"e.g. order_id\"\n type=\"text\"\n [value]=\"name\"\n [required]=\"true\"\n [fullWidth]=\"true\"\n [errors]=\"nameErrorsArray\"\n inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onNameChange($event)\">\n </cqa-custom-input>\n <span *ngIf=\"!nameError\"\n class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n {{ nameHelperText }}\n </span>\n </div>\n\n <!-- Type + Permission -->\n <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n <div class=\"cqa-flex cqa-flex-col\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Type</label>\n <cqa-dropdown-button\n [options]=\"typeDropdownOptions\"\n [selectedValue]=\"type\"\n (selectionChange)=\"onTypeChange($event)\">\n </cqa-dropdown-button>\n </div>\n <div class=\"cqa-flex cqa-flex-col\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Permission</label>\n <div class=\"cqa-pt-0.5\">\n <cqa-permission-toggle\n [value]=\"readWriteMode\"\n (valueChange)=\"onPermissionChange($event)\">\n </cqa-permission-toggle>\n </div>\n <span class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">{{ permissionHelperText }}</span>\n </div>\n </div>\n\n <!-- Value -->\n <div class=\"cqa-flex cqa-flex-col\">\n <ng-container *ngIf=\"isBoolean; else nonBooleanValue\">\n <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Value</label>\n <cqa-dropdown-button\n [options]=\"booleanDropdownOptions\"\n [selectedValue]=\"value\"\n (selectionChange)=\"onBooleanValueChange($event)\">\n </cqa-dropdown-button>\n </ng-container>\n <ng-template #nonBooleanValue>\n <cqa-custom-input\n label=\"Value\"\n [type]=\"isPassword ? 'password' : 'text'\"\n [value]=\"value\"\n [placeholder]=\"valuePlaceholder\"\n [fullWidth]=\"true\"\n [errors]=\"valueErrorsArray\"\n inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onValueChange($event)\">\n </cqa-custom-input>\n </ng-template>\n </div>\n\n</div>\n" }]
181
+ }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { mode: [{
182
+ type: Input
183
+ }], initialValue: [{
184
+ type: Input
185
+ }], existingNames: [{
186
+ type: Input
187
+ }], envName: [{
188
+ type: Input
189
+ }] } });
190
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"new-environment-variable-dialog.component.js","sourceRoot":"","sources":["../../../../../src/lib/new-environment-variable-dialog/new-environment-variable-dialog.component.ts","../../../../../src/lib/new-environment-variable-dialog/new-environment-variable-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Input, OnInit } from '@angular/core';\nimport {\n  EnvVariableDialogValue,\n  EnvVariableUIPermission,\n  EnvVariableUIType,\n} from './new-environment-variable-dialog.models';\n\n@Component({\n  selector: 'cqa-new-environment-variable-dialog',\n  templateUrl: './new-environment-variable-dialog.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' },\n})\nexport class NewEnvironmentVariableDialogComponent implements OnInit {\n  @Input() mode: 'create' | 'edit' = 'create';\n  @Input() initialValue?: Partial<EnvVariableDialogValue>;\n  @Input() existingNames: string[] = [];\n  @Input() envName: string = '';\n\n  readonly typeDropdownOptions: { label: string; value: EnvVariableUIType }[] = [\n    { label: 'String', value: 'String' },\n    { label: 'Number', value: 'Number' },\n    { label: 'Boolean', value: 'Boolean' },\n    { label: 'Password', value: 'Password' },\n  ];\n\n  name = '';\n  type: EnvVariableUIType = 'String';\n  value = '';\n  readWriteMode: EnvVariableUIPermission = 'RW';\n  passwordTouched = false;\n\n  nameError: string | null = null;\n  valueError: string | null = null;\n\n  constructor(private readonly cdr: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    if (this.initialValue) {\n      this.name = this.initialValue.name ?? '';\n      this.type = this.initialValue.type ?? 'String';\n      this.value = this.initialValue.value ?? '';\n      this.readWriteMode = this.initialValue.readWriteMode ?? 'RW';\n    }\n    if (this.type === 'Boolean' && this.value !== 'true' && this.value !== 'false') {\n      this.value = 'false';\n    }\n    if (this.mode === 'edit' && this.type === 'Password') {\n      this.value = '';\n    }\n  }\n\n  get title(): string {\n    return this.mode === 'edit' ? 'Edit variable' : 'New variable';\n  }\n\n  get subtitle(): string {\n    return this.envName ? `Scoped to the ${this.envName} environment.` : 'Scoped to this environment.';\n  }\n\n  get primaryButtonLabel(): string {\n    return this.mode === 'edit' ? 'Save changes' : 'Add variable';\n  }\n\n  get isBoolean(): boolean { return this.type === 'Boolean'; }\n  get isPassword(): boolean { return this.type === 'Password'; }\n  get isNumber(): boolean { return this.type === 'Number'; }\n\n  get valuePlaceholder(): string {\n    if (this.isPassword && this.mode === 'edit') {\n      return '•••••• (leave blank to keep existing)';\n    }\n    if (this.readWriteMode === 'RW') {\n      return 'Leave blank to populate during test execution';\n    }\n    return 'Enter a value';\n  }\n\n  get nameHelperText(): string {\n    if (this.nameError) { return this.nameError; }\n    return 'Variable names must be unique within the environment.';\n  }\n\n  get permissionHelperText(): string {\n    return this.readWriteMode === 'RO'\n      ? 'Read only — test steps can read but not overwrite.'\n      : 'Read/write — test cases can capture values here.';\n  }\n\n  get nameErrorsArray(): string[] { return this.nameError ? [this.nameError] : []; }\n  get valueErrorsArray(): string[] { return this.valueError ? [this.valueError] : []; }\n\n  get booleanDropdownOptions(): { label: string; value: string }[] {\n    return [\n      { label: 'true', value: 'true' },\n      { label: 'false', value: 'false' },\n    ];\n  }\n\n  onNameChange(next: string): void {\n    this.name = next;\n    this.nameError = null;\n    this.cdr.markForCheck();\n  }\n\n  onTypeChange(next: EnvVariableUIType): void {\n    if (!next) { return; }\n    this.type = next;\n    if (this.type === 'Boolean' && this.value !== 'true' && this.value !== 'false') {\n      this.value = 'false';\n    }\n    if (this.type === 'Password') {\n      this.passwordTouched = false;\n      if (this.mode === 'edit') { this.value = ''; }\n    }\n    this.valueError = null;\n    this.cdr.markForCheck();\n  }\n\n  onPermissionChange(next: string): void {\n    this.readWriteMode = next === 'RO' ? 'RO' : 'RW';\n    this.cdr.markForCheck();\n  }\n\n  onValueChange(next: string): void {\n    this.value = next;\n    if (this.isPassword) { this.passwordTouched = true; }\n    this.valueError = null;\n    this.cdr.markForCheck();\n  }\n\n  onBooleanValueChange(next: string): void {\n    this.value = next;\n    this.cdr.markForCheck();\n  }\n\n  /**\n   * Host reads via DialogRef.getComponentInstance(). Returns null on validation\n   * failure — host keeps the dialog open and surfaces the inline error already\n   * written onto nameError / valueError.\n   */\n  getValue(): EnvVariableDialogValue | null {\n    const trimmedName = this.name.trim();\n\n    if (!trimmedName) {\n      this.nameError = 'Name is required.';\n    } else if (this.isDuplicateName(trimmedName)) {\n      this.nameError = 'A variable with this name already exists in this environment.';\n    } else {\n      this.nameError = null;\n    }\n\n    if (this.isNumber && this.value && !this.isValidNumber(this.value)) {\n      this.valueError = 'Value must be a number.';\n    } else {\n      this.valueError = null;\n    }\n\n    if (this.nameError || this.valueError) {\n      this.cdr.markForCheck();\n      return null;\n    }\n\n    const omitPasswordValue = this.isPassword && this.mode === 'edit' && !this.passwordTouched;\n\n    return {\n      name: trimmedName,\n      type: this.type,\n      value: omitPasswordValue ? null : this.serializeValue(),\n      readWriteMode: this.readWriteMode,\n    };\n  }\n\n  private serializeValue(): string | null {\n    if (this.isBoolean) {\n      return this.value === 'true' ? 'true' : 'false';\n    }\n    return this.value ?? '';\n  }\n\n  private isDuplicateName(candidate: string): boolean {\n    const existing = (this.existingNames ?? []).map(n => (n ?? '').trim().toLowerCase());\n    const lowered = candidate.toLowerCase();\n    if (this.mode === 'edit') {\n      const original = (this.initialValue?.name ?? '').trim().toLowerCase();\n      return lowered !== original && existing.includes(lowered);\n    }\n    return existing.includes(lowered);\n  }\n\n  private isValidNumber(raw: string): boolean {\n    const trimmed = raw.trim();\n    if (!trimmed) { return true; }\n    return !Number.isNaN(Number(trimmed));\n  }\n}\n","<div class=\"cqa-flex cqa-flex-col cqa-gap-4 cqa-w-full\">\n\n  <!-- Name -->\n  <div class=\"cqa-flex cqa-flex-col\">\n    <cqa-custom-input\n      label=\"Name\"\n      placeholder=\"e.g. order_id\"\n      type=\"text\"\n      [value]=\"name\"\n      [required]=\"true\"\n      [fullWidth]=\"true\"\n      [errors]=\"nameErrorsArray\"\n      inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n      (valueChange)=\"onNameChange($event)\">\n    </cqa-custom-input>\n    <span *ngIf=\"!nameError\"\n      class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">\n      {{ nameHelperText }}\n    </span>\n  </div>\n\n  <!-- Type + Permission -->\n  <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n    <div class=\"cqa-flex cqa-flex-col\">\n      <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Type</label>\n      <cqa-dropdown-button\n        [options]=\"typeDropdownOptions\"\n        [selectedValue]=\"type\"\n        (selectionChange)=\"onTypeChange($event)\">\n      </cqa-dropdown-button>\n    </div>\n    <div class=\"cqa-flex cqa-flex-col\">\n      <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Permission</label>\n      <div class=\"cqa-pt-0.5\">\n        <cqa-permission-toggle\n          [value]=\"readWriteMode\"\n          (valueChange)=\"onPermissionChange($event)\">\n        </cqa-permission-toggle>\n      </div>\n      <span class=\"cqa-text-xs cqa-text-gray-500 cqa-mt-1\">{{ permissionHelperText }}</span>\n    </div>\n  </div>\n\n  <!-- Value -->\n  <div class=\"cqa-flex cqa-flex-col\">\n    <ng-container *ngIf=\"isBoolean; else nonBooleanValue\">\n      <label class=\"cqa-text-sm cqa-mb-1.5 cqa-font-medium cqa-text-gray-700\">Value</label>\n      <cqa-dropdown-button\n        [options]=\"booleanDropdownOptions\"\n        [selectedValue]=\"value\"\n        (selectionChange)=\"onBooleanValueChange($event)\">\n      </cqa-dropdown-button>\n    </ng-container>\n    <ng-template #nonBooleanValue>\n      <cqa-custom-input\n        label=\"Value\"\n        [type]=\"isPassword ? 'password' : 'text'\"\n        [value]=\"value\"\n        [placeholder]=\"valuePlaceholder\"\n        [fullWidth]=\"true\"\n        [errors]=\"valueErrorsArray\"\n        inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n        (valueChange)=\"onValueChange($event)\">\n      </cqa-custom-input>\n    </ng-template>\n  </div>\n\n</div>\n"]}