@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,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmV3LWVudmlyb25tZW50LXZhcmlhYmxlLWRpYWxvZy5tb2RlbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL25ldy1lbnZpcm9ubWVudC12YXJpYWJsZS1kaWFsb2cvbmV3LWVudmlyb25tZW50LXZhcmlhYmxlLWRpYWxvZy5tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIEVudlZhcmlhYmxlVUlUeXBlID0gJ1N0cmluZycgfCAnTnVtYmVyJyB8ICdCb29sZWFuJyB8ICdQYXNzd29yZCc7XG5leHBvcnQgdHlwZSBFbnZWYXJpYWJsZVVJUGVybWlzc2lvbiA9ICdSTycgfCAnUlcnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEVudlZhcmlhYmxlRGlhbG9nVmFsdWUge1xuICBuYW1lOiBzdHJpbmc7XG4gIHR5cGU6IEVudlZhcmlhYmxlVUlUeXBlO1xuICB2YWx1ZTogc3RyaW5nIHwgbnVsbDtcbiAgcmVhZFdyaXRlTW9kZTogRW52VmFyaWFibGVVSVBlcm1pc3Npb247XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTmV3RW52aXJvbm1lbnRWYXJpYWJsZURpYWxvZ0lucHV0cyB7XG4gIG1vZGU/OiAnY3JlYXRlJyB8ICdlZGl0JztcbiAgaW5pdGlhbFZhbHVlPzogUGFydGlhbDxFbnZWYXJpYWJsZURpYWxvZ1ZhbHVlPjtcbiAgZXhpc3RpbmdOYW1lcz86IHN0cmluZ1tdO1xuICBlbnZOYW1lPzogc3RyaW5nO1xufVxuIl19
@@ -0,0 +1,188 @@
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 "../custom-input/custom-input.component";
5
+ import * as i2 from "../permission-toggle/permission-toggle.component";
6
+ import * as i3 from "../button/button.component";
7
+ import * as i4 from "../dynamic-select/dynamic-select-field.component";
8
+ import * as i5 from "@angular/common";
9
+ export class NewTestDataProfileDialogComponent {
10
+ constructor(cdr) {
11
+ this.cdr = cdr;
12
+ this.mode = 'create';
13
+ this.existingNames = [];
14
+ this.environments = [];
15
+ this.lockedColumns = [];
16
+ this.name = '';
17
+ this.description = '';
18
+ this.readWriteMode = 'RW';
19
+ this.columns = ['col_1'];
20
+ this.envForm = new FormGroup({
21
+ environmentIds: new FormControl([]),
22
+ });
23
+ this.envConfig = this.buildEnvConfig([]);
24
+ this.nameError = null;
25
+ this.columnsError = null;
26
+ this.envsError = null;
27
+ this.trackByIndex = (_, __) => _;
28
+ }
29
+ ngOnInit() {
30
+ if (this.initialValue) {
31
+ this.name = this.initialValue.name ?? '';
32
+ this.description = this.initialValue.description ?? '';
33
+ this.readWriteMode = this.initialValue.readWriteMode ?? 'RW';
34
+ this.columns = (this.initialValue.columns && this.initialValue.columns.length)
35
+ ? [...this.initialValue.columns]
36
+ : ['col_1'];
37
+ const ids = this.initialValue.environmentIds ?? [];
38
+ this.envForm.get('environmentIds').setValue(ids);
39
+ }
40
+ this.envConfig = this.buildEnvConfig(this.environments ?? []);
41
+ // Clear envs error as soon as the user picks something.
42
+ this.envForm.get('environmentIds').valueChanges.subscribe(() => {
43
+ if (this.envsError) {
44
+ this.envsError = null;
45
+ this.cdr.markForCheck();
46
+ }
47
+ });
48
+ }
49
+ get title() {
50
+ return this.mode === 'edit' ? 'Edit test data profile' : 'New test data profile';
51
+ }
52
+ get subtitle() {
53
+ return 'Define columns; data values can be added per environment after creation.';
54
+ }
55
+ get primaryButtonLabel() {
56
+ return this.mode === 'edit' ? 'Save changes' : 'Create profile';
57
+ }
58
+ get nameErrorsArray() { return this.nameError ? [this.nameError] : []; }
59
+ get canAddColumn() { return true; }
60
+ get selectedEnvIds() {
61
+ const val = this.envForm.get('environmentIds').value ?? [];
62
+ return Array.isArray(val) ? val : [];
63
+ }
64
+ onNameChange(next) {
65
+ this.name = next;
66
+ this.nameError = null;
67
+ this.cdr.markForCheck();
68
+ }
69
+ onDescriptionChange(next) {
70
+ this.description = next;
71
+ this.cdr.markForCheck();
72
+ }
73
+ onPermissionChange(next) {
74
+ this.readWriteMode = next === 'RO' ? 'RO' : 'RW';
75
+ this.cdr.markForCheck();
76
+ }
77
+ onColumnChange(index, value) {
78
+ const next = [...this.columns];
79
+ next[index] = value;
80
+ this.columns = next;
81
+ this.columnsError = null;
82
+ this.cdr.markForCheck();
83
+ }
84
+ onAddColumn() {
85
+ this.columns = [...this.columns, `col_${this.columns.length + 1}`];
86
+ this.cdr.markForCheck();
87
+ }
88
+ onRemoveColumn(index) {
89
+ if (this.columns.length <= 1) {
90
+ return;
91
+ }
92
+ const target = this.columns[index];
93
+ if (this.lockedColumns && this.lockedColumns.indexOf(target) !== -1) {
94
+ return;
95
+ }
96
+ this.columns = this.columns.filter((_, i) => i !== index);
97
+ this.cdr.markForCheck();
98
+ }
99
+ isColumnLocked(col) {
100
+ return !!(this.lockedColumns && this.lockedColumns.indexOf(col) !== -1);
101
+ }
102
+ getValue() {
103
+ const trimmedName = this.name.trim();
104
+ if (!trimmedName) {
105
+ this.nameError = 'Name is required.';
106
+ }
107
+ else if (this.isDuplicateName(trimmedName)) {
108
+ this.nameError = 'A profile with this name already exists.';
109
+ }
110
+ else {
111
+ this.nameError = null;
112
+ }
113
+ const trimmedCols = this.columns.map(c => (c ?? '').trim()).filter(c => c.length > 0);
114
+ const unique = new Set(trimmedCols);
115
+ if (trimmedCols.length === 0) {
116
+ this.columnsError = 'At least one column is required.';
117
+ }
118
+ else if (unique.size !== trimmedCols.length) {
119
+ this.columnsError = 'Column names must be unique.';
120
+ }
121
+ else {
122
+ this.columnsError = null;
123
+ }
124
+ const envIds = this.selectedEnvIds;
125
+ if (!envIds || envIds.length === 0) {
126
+ this.envsError = 'Select at least one environment to assign this profile to.';
127
+ }
128
+ else {
129
+ this.envsError = null;
130
+ }
131
+ if (this.nameError || this.columnsError || this.envsError) {
132
+ this.cdr.markForCheck();
133
+ return null;
134
+ }
135
+ return {
136
+ name: trimmedName,
137
+ description: this.description.trim() || null,
138
+ readWriteMode: this.readWriteMode,
139
+ columns: trimmedCols,
140
+ environmentIds: this.selectedEnvIds,
141
+ };
142
+ }
143
+ buildEnvConfig(envs) {
144
+ const options = (envs || []).map(e => ({
145
+ id: e.id,
146
+ value: e.id,
147
+ name: e.name,
148
+ label: e.name,
149
+ statusColor: e.color,
150
+ }));
151
+ return {
152
+ key: 'environmentIds',
153
+ label: '',
154
+ placeholder: 'Select environments…',
155
+ multiple: true,
156
+ searchable: true,
157
+ optionStyle: 'checkbox',
158
+ showSelectAll: false,
159
+ options,
160
+ };
161
+ }
162
+ isDuplicateName(candidate) {
163
+ const existing = (this.existingNames ?? []).map(n => (n ?? '').trim().toLowerCase());
164
+ const lowered = candidate.toLowerCase();
165
+ if (this.mode === 'edit') {
166
+ const original = (this.initialValue?.name ?? '').trim().toLowerCase();
167
+ return lowered !== original && existing.includes(lowered);
168
+ }
169
+ return existing.includes(lowered);
170
+ }
171
+ }
172
+ NewTestDataProfileDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NewTestDataProfileDialogComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
173
+ NewTestDataProfileDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: NewTestDataProfileDialogComponent, selector: "cqa-new-test-data-profile-dialog", inputs: { mode: "mode", initialValue: "initialValue", existingNames: "existingNames", environments: "environments", lockedColumns: "lockedColumns" }, 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 + Permission -->\n <div class=\"cqa-grid cqa-grid-cols-[1fr_auto] cqa-gap-4 cqa-items-start\">\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Profile name\"\n placeholder=\"e.g. Guest Personas\"\n type=\"text\"\n [value]=\"name\"\n [required]=\"true\"\n [fullWidth]=\"true\"\n [errors]=\"nameErrorsArray\"\n (valueChange)=\"onNameChange($event)\">\n </cqa-custom-input>\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 <cqa-permission-toggle\n [value]=\"readWriteMode\"\n (valueChange)=\"onPermissionChange($event)\">\n </cqa-permission-toggle>\n </div>\n </div>\n\n <!-- Description -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Description\"\n placeholder=\"What data is this profile for?\"\n type=\"text\"\n [value]=\"description\"\n [fullWidth]=\"true\"\n (valueChange)=\"onDescriptionChange($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- Columns -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">Columns</label>\n <div *ngFor=\"let col of columns; let i = index; trackBy: trackByIndex\"\n class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <cqa-custom-input\n [value]=\"col\"\n [placeholder]=\"'column_' + (i + 1)\"\n [fullWidth]=\"true\"\n [disabled]=\"isColumnLocked(col)\"\n inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onColumnChange(i, $event)\">\n </cqa-custom-input>\n <cqa-button\n variant=\"text\"\n btnSize=\"sm\"\n icon=\"delete\"\n tooltip=\"Remove column\"\n [disabled]=\"columns.length <= 1 || isColumnLocked(col)\"\n (clicked)=\"onRemoveColumn(i)\">\n </cqa-button>\n </div>\n <div>\n <cqa-button\n variant=\"text\"\n btnSize=\"sm\"\n icon=\"add\"\n text=\"Add column\"\n (clicked)=\"onAddColumn()\">\n </cqa-button>\n </div>\n <span *ngIf=\"columnsError\" class=\"cqa-text-xs cqa-text-red-600\">{{ columnsError }}</span>\n </div>\n\n <!-- Assign to environments -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">\n Assign to environments <span class=\"cqa-text-red-600\">*</span>\n </label>\n <cqa-dynamic-select\n [form]=\"envForm\"\n [config]=\"envConfig\">\n </cqa-dynamic-select>\n <span *ngIf=\"envsError\" class=\"cqa-text-xs cqa-text-red-600\">{{ envsError }}</span>\n <span *ngIf=\"!envsError\" class=\"cqa-text-xs cqa-text-gray-500\">\n Each environment gets an independent copy of the data (same columns, separate rows).\n </span>\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.PermissionToggleComponent, selector: "cqa-permission-toggle", inputs: ["value", "disabled", "roTooltip", "rwTooltip"], outputs: ["valueChange"] }, { type: i3.ButtonComponent, selector: "cqa-button", inputs: ["variant", "btnSize", "disabled", "loading", "icon", "iconPosition", "fullWidth", "iconColor", "type", "text", "customClass", "inlineStyles", "tooltip", "tooltipPosition"], outputs: ["clicked"] }, { type: i4.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }], directives: [{ type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
174
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NewTestDataProfileDialogComponent, decorators: [{
175
+ type: Component,
176
+ args: [{ selector: 'cqa-new-test-data-profile-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 + Permission -->\n <div class=\"cqa-grid cqa-grid-cols-[1fr_auto] cqa-gap-4 cqa-items-start\">\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Profile name\"\n placeholder=\"e.g. Guest Personas\"\n type=\"text\"\n [value]=\"name\"\n [required]=\"true\"\n [fullWidth]=\"true\"\n [errors]=\"nameErrorsArray\"\n (valueChange)=\"onNameChange($event)\">\n </cqa-custom-input>\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 <cqa-permission-toggle\n [value]=\"readWriteMode\"\n (valueChange)=\"onPermissionChange($event)\">\n </cqa-permission-toggle>\n </div>\n </div>\n\n <!-- Description -->\n <div class=\"cqa-flex cqa-flex-col\">\n <cqa-custom-input\n label=\"Description\"\n placeholder=\"What data is this profile for?\"\n type=\"text\"\n [value]=\"description\"\n [fullWidth]=\"true\"\n (valueChange)=\"onDescriptionChange($event)\">\n </cqa-custom-input>\n </div>\n\n <!-- Columns -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">Columns</label>\n <div *ngFor=\"let col of columns; let i = index; trackBy: trackByIndex\"\n class=\"cqa-flex cqa-items-center cqa-gap-2\">\n <cqa-custom-input\n [value]=\"col\"\n [placeholder]=\"'column_' + (i + 1)\"\n [fullWidth]=\"true\"\n [disabled]=\"isColumnLocked(col)\"\n inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n (valueChange)=\"onColumnChange(i, $event)\">\n </cqa-custom-input>\n <cqa-button\n variant=\"text\"\n btnSize=\"sm\"\n icon=\"delete\"\n tooltip=\"Remove column\"\n [disabled]=\"columns.length <= 1 || isColumnLocked(col)\"\n (clicked)=\"onRemoveColumn(i)\">\n </cqa-button>\n </div>\n <div>\n <cqa-button\n variant=\"text\"\n btnSize=\"sm\"\n icon=\"add\"\n text=\"Add column\"\n (clicked)=\"onAddColumn()\">\n </cqa-button>\n </div>\n <span *ngIf=\"columnsError\" class=\"cqa-text-xs cqa-text-red-600\">{{ columnsError }}</span>\n </div>\n\n <!-- Assign to environments -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">\n Assign to environments <span class=\"cqa-text-red-600\">*</span>\n </label>\n <cqa-dynamic-select\n [form]=\"envForm\"\n [config]=\"envConfig\">\n </cqa-dynamic-select>\n <span *ngIf=\"envsError\" class=\"cqa-text-xs cqa-text-red-600\">{{ envsError }}</span>\n <span *ngIf=\"!envsError\" class=\"cqa-text-xs cqa-text-gray-500\">\n Each environment gets an independent copy of the data (same columns, separate rows).\n </span>\n </div>\n\n</div>\n" }]
177
+ }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { mode: [{
178
+ type: Input
179
+ }], initialValue: [{
180
+ type: Input
181
+ }], existingNames: [{
182
+ type: Input
183
+ }], environments: [{
184
+ type: Input
185
+ }], lockedColumns: [{
186
+ type: Input
187
+ }] } });
188
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"new-test-data-profile-dialog.component.js","sourceRoot":"","sources":["../../../../../src/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.component.ts","../../../../../src/lib/new-test-data-profile-dialog/new-test-data-profile-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  TdpDialogPermission,\n  TdpDialogValue,\n  TdpEnvironmentOption,\n} from './new-test-data-profile-dialog.models';\n\n@Component({\n  selector: 'cqa-new-test-data-profile-dialog',\n  templateUrl: './new-test-data-profile-dialog.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: { class: 'cqa-ui-root', style: 'display:block;width:100%;' },\n})\nexport class NewTestDataProfileDialogComponent implements OnInit {\n  @Input() mode: 'create' | 'edit' = 'create';\n  @Input() initialValue?: Partial<TdpDialogValue>;\n  @Input() existingNames: string[] = [];\n  @Input() environments: TdpEnvironmentOption[] = [];\n  @Input() lockedColumns: string[] = [];\n\n  public name = '';\n  public description = '';\n  public readWriteMode: TdpDialogPermission = 'RW';\n  public columns: string[] = ['col_1'];\n\n  public readonly envForm = new FormGroup({\n    environmentIds: new FormControl([]),\n  });\n  public envConfig: DynamicSelectFieldConfig = this.buildEnvConfig([]);\n\n  public nameError: string | null = null;\n  public columnsError: string | null = null;\n  public envsError: string | null = null;\n\n  public trackByIndex = (_: number, __: string): number => _;\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.readWriteMode = this.initialValue.readWriteMode ?? 'RW';\n      this.columns = (this.initialValue.columns && this.initialValue.columns.length)\n        ? [...this.initialValue.columns]\n        : ['col_1'];\n      const ids = this.initialValue.environmentIds ?? [];\n      this.envForm.get('environmentIds')!.setValue(ids);\n    }\n    this.envConfig = this.buildEnvConfig(this.environments ?? []);\n\n    // Clear envs error as soon as the user picks something.\n    this.envForm.get('environmentIds')!.valueChanges.subscribe(() => {\n      if (this.envsError) {\n        this.envsError = null;\n        this.cdr.markForCheck();\n      }\n    });\n  }\n\n  public get title(): string {\n    return this.mode === 'edit' ? 'Edit test data profile' : 'New test data profile';\n  }\n\n  public get subtitle(): string {\n    return 'Define columns; data values can be added per environment after creation.';\n  }\n\n  public get primaryButtonLabel(): string {\n    return this.mode === 'edit' ? 'Save changes' : 'Create profile';\n  }\n\n  public get nameErrorsArray(): string[] { return this.nameError ? [this.nameError] : []; }\n\n  public get canAddColumn(): boolean { return true; }\n\n  public get selectedEnvIds(): number[] {\n    const val = this.envForm.get('environmentIds')!.value ?? [];\n    return Array.isArray(val) ? val : [];\n  }\n\n  public onNameChange(next: string): void {\n    this.name = next;\n    this.nameError = null;\n    this.cdr.markForCheck();\n  }\n\n  public onDescriptionChange(next: string): void {\n    this.description = next;\n    this.cdr.markForCheck();\n  }\n\n  public onPermissionChange(next: string): void {\n    this.readWriteMode = next === 'RO' ? 'RO' : 'RW';\n    this.cdr.markForCheck();\n  }\n\n  public onColumnChange(index: number, value: string): void {\n    const next = [...this.columns];\n    next[index] = value;\n    this.columns = next;\n    this.columnsError = null;\n    this.cdr.markForCheck();\n  }\n\n  public onAddColumn(): void {\n    this.columns = [...this.columns, `col_${this.columns.length + 1}`];\n    this.cdr.markForCheck();\n  }\n\n  public onRemoveColumn(index: number): void {\n    if (this.columns.length <= 1) { return; }\n    const target = this.columns[index];\n    if (this.lockedColumns && this.lockedColumns.indexOf(target) !== -1) { return; }\n    this.columns = this.columns.filter((_, i) => i !== index);\n    this.cdr.markForCheck();\n  }\n\n  public isColumnLocked(col: string): boolean {\n    return !!(this.lockedColumns && this.lockedColumns.indexOf(col) !== -1);\n  }\n\n  public getValue(): TdpDialogValue | null {\n    const trimmedName = this.name.trim();\n    if (!trimmedName) {\n      this.nameError = 'Name is required.';\n    } else if (this.isDuplicateName(trimmedName)) {\n      this.nameError = 'A profile with this name already exists.';\n    } else {\n      this.nameError = null;\n    }\n\n    const trimmedCols = this.columns.map(c => (c ?? '').trim()).filter(c => c.length > 0);\n    const unique = new Set(trimmedCols);\n    if (trimmedCols.length === 0) {\n      this.columnsError = 'At least one column is required.';\n    } else if (unique.size !== trimmedCols.length) {\n      this.columnsError = 'Column names must be unique.';\n    } else {\n      this.columnsError = null;\n    }\n\n    const envIds = this.selectedEnvIds;\n    if (!envIds || envIds.length === 0) {\n      this.envsError = 'Select at least one environment to assign this profile to.';\n    } else {\n      this.envsError = null;\n    }\n\n    if (this.nameError || this.columnsError || this.envsError) {\n      this.cdr.markForCheck();\n      return null;\n    }\n\n    return {\n      name: trimmedName,\n      description: this.description.trim() || null,\n      readWriteMode: this.readWriteMode,\n      columns: trimmedCols,\n      environmentIds: this.selectedEnvIds,\n    };\n  }\n\n  private buildEnvConfig(envs: TdpEnvironmentOption[]): 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: 'environmentIds',\n      label: '',\n      placeholder: 'Select environments…',\n      multiple: true,\n      searchable: true,\n      optionStyle: 'checkbox',\n      showSelectAll: false,\n      options,\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 + Permission -->\n  <div class=\"cqa-grid cqa-grid-cols-[1fr_auto] cqa-gap-4 cqa-items-start\">\n    <div class=\"cqa-flex cqa-flex-col\">\n      <cqa-custom-input\n        label=\"Profile name\"\n        placeholder=\"e.g. Guest Personas\"\n        type=\"text\"\n        [value]=\"name\"\n        [required]=\"true\"\n        [fullWidth]=\"true\"\n        [errors]=\"nameErrorsArray\"\n        (valueChange)=\"onNameChange($event)\">\n      </cqa-custom-input>\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      <cqa-permission-toggle\n        [value]=\"readWriteMode\"\n        (valueChange)=\"onPermissionChange($event)\">\n      </cqa-permission-toggle>\n    </div>\n  </div>\n\n  <!-- Description -->\n  <div class=\"cqa-flex cqa-flex-col\">\n    <cqa-custom-input\n      label=\"Description\"\n      placeholder=\"What data is this profile for?\"\n      type=\"text\"\n      [value]=\"description\"\n      [fullWidth]=\"true\"\n      (valueChange)=\"onDescriptionChange($event)\">\n    </cqa-custom-input>\n  </div>\n\n  <!-- Columns -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n    <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">Columns</label>\n    <div *ngFor=\"let col of columns; let i = index; trackBy: trackByIndex\"\n         class=\"cqa-flex cqa-items-center cqa-gap-2\">\n      <cqa-custom-input\n        [value]=\"col\"\n        [placeholder]=\"'column_' + (i + 1)\"\n        [fullWidth]=\"true\"\n        [disabled]=\"isColumnLocked(col)\"\n        inputInlineStyle=\"font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', monospace;\"\n        (valueChange)=\"onColumnChange(i, $event)\">\n      </cqa-custom-input>\n      <cqa-button\n        variant=\"text\"\n        btnSize=\"sm\"\n        icon=\"delete\"\n        tooltip=\"Remove column\"\n        [disabled]=\"columns.length <= 1 || isColumnLocked(col)\"\n        (clicked)=\"onRemoveColumn(i)\">\n      </cqa-button>\n    </div>\n    <div>\n      <cqa-button\n        variant=\"text\"\n        btnSize=\"sm\"\n        icon=\"add\"\n        text=\"Add column\"\n        (clicked)=\"onAddColumn()\">\n      </cqa-button>\n    </div>\n    <span *ngIf=\"columnsError\" class=\"cqa-text-xs cqa-text-red-600\">{{ columnsError }}</span>\n  </div>\n\n  <!-- Assign to environments -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-2\">\n    <label class=\"cqa-text-sm cqa-font-medium cqa-text-gray-700\">\n      Assign to environments <span class=\"cqa-text-red-600\">*</span>\n    </label>\n    <cqa-dynamic-select\n      [form]=\"envForm\"\n      [config]=\"envConfig\">\n    </cqa-dynamic-select>\n    <span *ngIf=\"envsError\" class=\"cqa-text-xs cqa-text-red-600\">{{ envsError }}</span>\n    <span *ngIf=\"!envsError\" class=\"cqa-text-xs cqa-text-gray-500\">\n      Each environment gets an independent copy of the data (same columns, separate rows).\n    </span>\n  </div>\n\n</div>\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmV3LXRlc3QtZGF0YS1wcm9maWxlLWRpYWxvZy5tb2RlbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL25ldy10ZXN0LWRhdGEtcHJvZmlsZS1kaWFsb2cvbmV3LXRlc3QtZGF0YS1wcm9maWxlLWRpYWxvZy5tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIFRkcERpYWxvZ1Blcm1pc3Npb24gPSAnUk8nIHwgJ1JXJztcblxuZXhwb3J0IGludGVyZmFjZSBUZHBEaWFsb2dWYWx1ZSB7XG4gIG5hbWU6IHN0cmluZztcbiAgZGVzY3JpcHRpb246IHN0cmluZyB8IG51bGw7XG4gIHJlYWRXcml0ZU1vZGU6IFRkcERpYWxvZ1Blcm1pc3Npb247XG4gIGNvbHVtbnM6IHN0cmluZ1tdO1xuICBlbnZpcm9ubWVudElkczogbnVtYmVyW107XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgVGRwRW52aXJvbm1lbnRPcHRpb24ge1xuICBpZDogbnVtYmVyO1xuICBuYW1lOiBzdHJpbmc7XG4gIGNvbG9yPzogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE5ld1Rlc3REYXRhUHJvZmlsZURpYWxvZ0lucHV0cyB7XG4gIG1vZGU/OiAnY3JlYXRlJyB8ICdlZGl0JztcbiAgaW5pdGlhbFZhbHVlPzogUGFydGlhbDxUZHBEaWFsb2dWYWx1ZT47XG4gIGV4aXN0aW5nTmFtZXM/OiBzdHJpbmdbXTtcbiAgZW52aXJvbm1lbnRzPzogVGRwRW52aXJvbm1lbnRPcHRpb25bXTtcbiAgbG9ja2VkQ29sdW1ucz86IHN0cmluZ1tdO1xufVxuIl19