@cqa-lib/cqa-ui 1.1.541-gamma.8 → 1.1.542

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 (64) hide show
  1. package/.github/workflows/README.md +1 -8
  2. package/esm2020/lib/compare-runs/compare-runs.component.mjs +1 -1
  3. package/esm2020/lib/execution-screen/condition-debug-step/condition-branch-editor.component.mjs +3 -32
  4. package/esm2020/lib/execution-screen/db-query-execution-item/db-query-execution-item.component.mjs +1 -1
  5. package/esm2020/lib/execution-screen/db-verification-step/db-verification-step.component.mjs +1 -1
  6. package/esm2020/lib/iterations-loop/iterations-loop.component.mjs +1 -1
  7. package/esm2020/lib/mixed-variable-input/mixed-variable-input.component.mjs +10 -30
  8. package/esm2020/lib/new-global-variable-dialog/new-global-variable-dialog.component.mjs +5 -16
  9. package/esm2020/lib/new-global-variable-dialog/new-global-variable-dialog.models.mjs +1 -1
  10. package/esm2020/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.component.mjs +51 -50
  11. package/esm2020/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.models.mjs +1 -1
  12. package/esm2020/lib/segment-control/segment-control.component.mjs +4 -12
  13. package/esm2020/lib/step-builder/step-builder-action/step-builder-action.component.mjs +3 -21
  14. package/esm2020/lib/step-builder/step-builder-ai-agent/step-builder-ai-agent.component.mjs +23 -47
  15. package/esm2020/lib/step-builder/step-builder-condition/step-builder-condition.component.mjs +3 -21
  16. package/esm2020/lib/step-builder/step-builder-database/step-builder-database.component.mjs +47 -76
  17. package/esm2020/lib/step-builder/step-builder-loop/step-builder-loop.component.mjs +3 -16
  18. package/esm2020/lib/step-builder/template-variables-form/template-variables-form.component.mjs +170 -326
  19. package/esm2020/lib/templates/table-template.component.mjs +3 -5
  20. package/esm2020/lib/test-case-details/ai-agent-step/ai-agent-step.component.mjs +3 -3
  21. package/esm2020/lib/test-case-details/ai-verify-step/ai-verify-step.component.mjs +3 -3
  22. package/esm2020/lib/test-case-details/api-edit-step/api-edit-step.component.mjs +7 -20
  23. package/esm2020/lib/test-case-details/test-case-details-edit/test-case-details-edit.component.mjs +4 -43
  24. package/esm2020/lib/test-case-details/test-case-details.models.mjs +1 -3
  25. package/esm2020/lib/test-case-details/test-data-modal/test-data-modal-data.mjs +1 -1
  26. package/esm2020/lib/test-case-details/test-data-modal/test-data-modal.component.mjs +7 -137
  27. package/esm2020/lib/ui-kit.module.mjs +3 -23
  28. package/esm2020/public-api.mjs +1 -6
  29. package/fesm2015/cqa-lib-cqa-ui.mjs +404 -1245
  30. package/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -1
  31. package/fesm2020/cqa-lib-cqa-ui.mjs +330 -1156
  32. package/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -1
  33. package/lib/execution-screen/condition-debug-step/condition-branch-editor.component.d.ts +1 -26
  34. package/lib/new-global-variable-dialog/new-global-variable-dialog.component.d.ts +2 -5
  35. package/lib/new-global-variable-dialog/new-global-variable-dialog.models.d.ts +0 -8
  36. package/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.component.d.ts +10 -9
  37. package/lib/new-test-data-profile-dialog/new-test-data-profile-dialog.models.d.ts +1 -1
  38. package/lib/segment-control/segment-control.component.d.ts +1 -3
  39. package/lib/step-builder/step-builder-action/step-builder-action.component.d.ts +1 -19
  40. package/lib/step-builder/step-builder-ai-agent/step-builder-ai-agent.component.d.ts +2 -16
  41. package/lib/step-builder/step-builder-condition/step-builder-condition.component.d.ts +1 -19
  42. package/lib/step-builder/step-builder-database/step-builder-database.component.d.ts +11 -33
  43. package/lib/step-builder/step-builder-loop/step-builder-loop.component.d.ts +1 -15
  44. package/lib/step-builder/template-variables-form/template-variables-form.component.d.ts +20 -75
  45. package/lib/templates/table-template.component.d.ts +1 -8
  46. package/lib/test-case-details/api-edit-step/api-edit-step.component.d.ts +1 -7
  47. package/lib/test-case-details/test-case-details-edit/test-case-details-edit.component.d.ts +0 -7
  48. package/lib/test-case-details/test-case-details.models.d.ts +0 -2
  49. package/lib/test-case-details/test-data-modal/test-data-modal-data.d.ts +1 -23
  50. package/lib/test-case-details/test-data-modal/test-data-modal.component.d.ts +6 -40
  51. package/lib/ui-kit.module.d.ts +96 -100
  52. package/package.json +1 -1
  53. package/public-api.d.ts +0 -5
  54. package/styles.css +1 -1
  55. package/esm2020/lib/new-global-variable-dialog/global-variable-tags-input.component.mjs +0 -129
  56. package/esm2020/lib/test-case-details/data-library-panel/data-library-panel.component.mjs +0 -116
  57. package/esm2020/lib/test-case-details/data-library-panel/data-library-panel.models.mjs +0 -2
  58. package/esm2020/lib/test-case-details/data-library-panel/element-card.component.mjs +0 -36
  59. package/esm2020/lib/test-case-details/data-library-panel/test-data-profile-card.component.mjs +0 -38
  60. package/lib/new-global-variable-dialog/global-variable-tags-input.component.d.ts +0 -39
  61. package/lib/test-case-details/data-library-panel/data-library-panel.component.d.ts +0 -42
  62. package/lib/test-case-details/data-library-panel/data-library-panel.models.d.ts +0 -17
  63. package/lib/test-case-details/data-library-panel/element-card.component.d.ts +0 -13
  64. package/lib/test-case-details/data-library-panel/test-data-profile-card.component.d.ts +0 -13
@@ -3,36 +3,37 @@ import { FormControl, FormGroup } from '@angular/forms';
3
3
  import * as i0 from "@angular/core";
4
4
  import * as i1 from "../custom-input/custom-input.component";
5
5
  import * as i2 from "../permission-toggle/permission-toggle.component";
6
- import * as i3 from "../dynamic-select/dynamic-select-field.component";
7
- import * as i4 from "@angular/common";
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";
8
9
  export class NewTestDataProfileDialogComponent {
9
10
  constructor(cdr) {
10
11
  this.cdr = cdr;
11
12
  this.mode = 'create';
12
13
  this.existingNames = [];
13
14
  this.environments = [];
15
+ this.lockedColumns = [];
14
16
  this.name = '';
15
17
  this.description = '';
16
18
  this.readWriteMode = 'RW';
17
- this.rowsInput = '1';
18
- this.columnsInput = '1';
19
+ this.columns = ['col_1'];
19
20
  this.envForm = new FormGroup({
20
21
  environmentIds: new FormControl([]),
21
22
  });
22
23
  this.envConfig = this.buildEnvConfig([]);
23
24
  this.nameError = null;
24
- this.matrixError = null;
25
+ this.columnsError = null;
25
26
  this.envsError = null;
27
+ this.trackByIndex = (_, __) => _;
26
28
  }
27
29
  ngOnInit() {
28
30
  if (this.initialValue) {
29
31
  this.name = this.initialValue.name ?? '';
30
32
  this.description = this.initialValue.description ?? '';
31
33
  this.readWriteMode = this.initialValue.readWriteMode ?? 'RW';
32
- const colCount = this.initialValue.columns?.length ?? 1;
33
- const rowCount = this.initialValue.rows?.length ?? 1;
34
- this.rowsInput = String(rowCount);
35
- this.columnsInput = String(colCount);
34
+ this.columns = (this.initialValue.columns && this.initialValue.columns.length)
35
+ ? [...this.initialValue.columns]
36
+ : ['col_1'];
36
37
  const ids = this.initialValue.environmentIds ?? [];
37
38
  this.envForm.get('environmentIds').setValue(ids);
38
39
  }
@@ -49,12 +50,13 @@ export class NewTestDataProfileDialogComponent {
49
50
  return this.mode === 'edit' ? 'Edit test data profile' : 'New test data profile';
50
51
  }
51
52
  get subtitle() {
52
- return 'Define the matrix; data values can be added per environment after creation.';
53
+ return 'Define columns; data values can be added per environment after creation.';
53
54
  }
54
55
  get primaryButtonLabel() {
55
56
  return this.mode === 'edit' ? 'Save changes' : 'Create profile';
56
57
  }
57
58
  get nameErrorsArray() { return this.nameError ? [this.nameError] : []; }
59
+ get canAddColumn() { return true; }
58
60
  get selectedEnvIds() {
59
61
  const val = this.envForm.get('environmentIds').value ?? [];
60
62
  return Array.isArray(val) ? val : [];
@@ -72,16 +74,31 @@ export class NewTestDataProfileDialogComponent {
72
74
  this.readWriteMode = next === 'RO' ? 'RO' : 'RW';
73
75
  this.cdr.markForCheck();
74
76
  }
75
- onRowsChange(next) {
76
- this.rowsInput = next ?? '';
77
- this.matrixError = null;
77
+ onColumnChange(index, value) {
78
+ const next = [...this.columns];
79
+ next[index] = value;
80
+ this.columns = next;
81
+ this.columnsError = null;
78
82
  this.cdr.markForCheck();
79
83
  }
80
- onColumnsChange(next) {
81
- this.columnsInput = next ?? '';
82
- this.matrixError = null;
84
+ onAddColumn() {
85
+ this.columns = [...this.columns, `col_${this.columns.length + 1}`];
83
86
  this.cdr.markForCheck();
84
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
+ }
85
102
  getValue() {
86
103
  const trimmedName = this.name.trim();
87
104
  if (!trimmedName) {
@@ -93,7 +110,17 @@ export class NewTestDataProfileDialogComponent {
93
110
  else {
94
111
  this.nameError = null;
95
112
  }
96
- const parsed = this.parseMatrix();
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
+ }
97
124
  const envIds = this.selectedEnvIds;
98
125
  if (!envIds || envIds.length === 0) {
99
126
  this.envsError = 'Select at least one environment to assign this profile to.';
@@ -101,44 +128,18 @@ export class NewTestDataProfileDialogComponent {
101
128
  else {
102
129
  this.envsError = null;
103
130
  }
104
- if (this.nameError || this.matrixError || this.envsError || !parsed) {
131
+ if (this.nameError || this.columnsError || this.envsError) {
105
132
  this.cdr.markForCheck();
106
133
  return null;
107
134
  }
108
- const cols = Array.from({ length: parsed.cols }, (_, i) => `col_${i + 1}`);
109
- const rows = Array.from({ length: parsed.rows }, (_, i) => `row_${i + 1}`);
110
135
  return {
111
136
  name: trimmedName,
112
137
  description: this.description.trim() || null,
113
138
  readWriteMode: this.readWriteMode,
114
- columns: cols,
115
- rows,
139
+ columns: trimmedCols,
116
140
  environmentIds: this.selectedEnvIds,
117
141
  };
118
142
  }
119
- parseMatrix() {
120
- const rawRows = (this.rowsInput ?? '').trim();
121
- const rawCols = (this.columnsInput ?? '').trim();
122
- const rowsMatch = NewTestDataProfileDialogComponent.INTEGER_REGEX.exec(rawRows);
123
- const colsMatch = NewTestDataProfileDialogComponent.INTEGER_REGEX.exec(rawCols);
124
- if (!rowsMatch || !colsMatch) {
125
- this.matrixError = 'Rows and columns must be whole numbers.';
126
- return null;
127
- }
128
- const rows = parseInt(rowsMatch[1], 10);
129
- const cols = parseInt(colsMatch[1], 10);
130
- if (rows < 1 || cols < 1) {
131
- this.matrixError = 'Rows and columns must be at least 1.';
132
- return null;
133
- }
134
- if (rows > NewTestDataProfileDialogComponent.MAX_DIM
135
- || cols > NewTestDataProfileDialogComponent.MAX_DIM) {
136
- this.matrixError = `Rows and columns can be at most ${NewTestDataProfileDialogComponent.MAX_DIM}.`;
137
- return null;
138
- }
139
- this.matrixError = null;
140
- return { rows, cols };
141
- }
142
143
  buildEnvConfig(envs) {
143
144
  const options = (envs || []).map(e => ({
144
145
  id: e.id,
@@ -168,13 +169,11 @@ export class NewTestDataProfileDialogComponent {
168
169
  return existing.includes(lowered);
169
170
  }
170
171
  }
171
- NewTestDataProfileDialogComponent.INTEGER_REGEX = /^\s*(\d+)\s*$/;
172
- NewTestDataProfileDialogComponent.MAX_DIM = 100;
173
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 });
174
- 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" }, 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 <!-- Rows + Columns matrix -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1\">\n <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n <cqa-custom-input\n label=\"Rows\"\n placeholder=\"e.g. 2\"\n type=\"number\"\n [value]=\"rowsInput\"\n [fullWidth]=\"true\"\n (valueChange)=\"onRowsChange($event)\">\n </cqa-custom-input>\n <cqa-custom-input\n label=\"Columns\"\n placeholder=\"e.g. 3\"\n type=\"number\"\n [value]=\"columnsInput\"\n [fullWidth]=\"true\"\n (valueChange)=\"onColumnsChange($event)\">\n </cqa-custom-input>\n </div>\n <span *ngIf=\"matrixError\" class=\"cqa-text-xs cqa-text-red-600\">{{ matrixError }}</span>\n <span *ngIf=\"!matrixError\" class=\"cqa-text-xs cqa-text-gray-500\">\n Defaults to <code>row_1\u2026row_N</code> and <code>col_1\u2026col_M</code>.\n </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.DynamicSelectFieldComponent, selector: "cqa-dynamic-select", inputs: ["form", "config"], outputs: ["selectionChange", "selectClick", "searchChange", "loadMore", "addCustomValue"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
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 });
175
174
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: NewTestDataProfileDialogComponent, decorators: [{
176
175
  type: Component,
177
- 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 <!-- Rows + Columns matrix -->\n <div class=\"cqa-flex cqa-flex-col cqa-gap-1\">\n <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n <cqa-custom-input\n label=\"Rows\"\n placeholder=\"e.g. 2\"\n type=\"number\"\n [value]=\"rowsInput\"\n [fullWidth]=\"true\"\n (valueChange)=\"onRowsChange($event)\">\n </cqa-custom-input>\n <cqa-custom-input\n label=\"Columns\"\n placeholder=\"e.g. 3\"\n type=\"number\"\n [value]=\"columnsInput\"\n [fullWidth]=\"true\"\n (valueChange)=\"onColumnsChange($event)\">\n </cqa-custom-input>\n </div>\n <span *ngIf=\"matrixError\" class=\"cqa-text-xs cqa-text-red-600\">{{ matrixError }}</span>\n <span *ngIf=\"!matrixError\" class=\"cqa-text-xs cqa-text-gray-500\">\n Defaults to <code>row_1\u2026row_N</code> and <code>col_1\u2026col_M</code>.\n </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" }]
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" }]
178
177
  }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { mode: [{
179
178
  type: Input
180
179
  }], initialValue: [{
@@ -183,5 +182,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
183
182
  type: Input
184
183
  }], environments: [{
185
184
  type: Input
185
+ }], lockedColumns: [{
186
+ type: Input
186
187
  }] } });
187
- //# 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  private static readonly INTEGER_REGEX = /^\\s*(\\d+)\\s*$/;\n  private static readonly MAX_DIM = 100;\n\n  @Input() mode: 'create' | 'edit' = 'create';\n  @Input() initialValue?: Partial<TdpDialogValue>;\n  @Input() existingNames: string[] = [];\n  @Input() environments: TdpEnvironmentOption[] = [];\n\n  public name = '';\n  public description = '';\n  public readWriteMode: TdpDialogPermission = 'RW';\n  public rowsInput = '1';\n  public columnsInput = '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 matrixError: string | null = null;\n  public envsError: 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.readWriteMode = this.initialValue.readWriteMode ?? 'RW';\n      const colCount = this.initialValue.columns?.length ?? 1;\n      const rowCount = this.initialValue.rows?.length ?? 1;\n      this.rowsInput = String(rowCount);\n      this.columnsInput = String(colCount);\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 the matrix; 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 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 onRowsChange(next: string): void {\n    this.rowsInput = next ?? '';\n    this.matrixError = null;\n    this.cdr.markForCheck();\n  }\n\n  public onColumnsChange(next: string): void {\n    this.columnsInput = next ?? '';\n    this.matrixError = null;\n    this.cdr.markForCheck();\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 parsed = this.parseMatrix();\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.matrixError || this.envsError || !parsed) {\n      this.cdr.markForCheck();\n      return null;\n    }\n\n    const cols = Array.from({ length: parsed.cols }, (_, i) => `col_${i + 1}`);\n    const rows = Array.from({ length: parsed.rows }, (_, i) => `row_${i + 1}`);\n\n    return {\n      name: trimmedName,\n      description: this.description.trim() || null,\n      readWriteMode: this.readWriteMode,\n      columns: cols,\n      rows,\n      environmentIds: this.selectedEnvIds,\n    };\n  }\n\n  private parseMatrix(): { rows: number; cols: number } | null {\n    const rawRows = (this.rowsInput ?? '').trim();\n    const rawCols = (this.columnsInput ?? '').trim();\n    const rowsMatch = NewTestDataProfileDialogComponent.INTEGER_REGEX.exec(rawRows);\n    const colsMatch = NewTestDataProfileDialogComponent.INTEGER_REGEX.exec(rawCols);\n    if (!rowsMatch || !colsMatch) {\n      this.matrixError = 'Rows and columns must be whole numbers.';\n      return null;\n    }\n    const rows = parseInt(rowsMatch[1], 10);\n    const cols = parseInt(colsMatch[1], 10);\n    if (rows < 1 || cols < 1) {\n      this.matrixError = 'Rows and columns must be at least 1.';\n      return null;\n    }\n    if (rows > NewTestDataProfileDialogComponent.MAX_DIM\n      || cols > NewTestDataProfileDialogComponent.MAX_DIM) {\n      this.matrixError = `Rows and columns can be at most ${NewTestDataProfileDialogComponent.MAX_DIM}.`;\n      return null;\n    }\n    this.matrixError = null;\n    return { rows, cols };\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  <!-- Rows + Columns matrix -->\n  <div class=\"cqa-flex cqa-flex-col cqa-gap-1\">\n    <div class=\"cqa-grid cqa-grid-cols-2 cqa-gap-4\">\n      <cqa-custom-input\n        label=\"Rows\"\n        placeholder=\"e.g. 2\"\n        type=\"number\"\n        [value]=\"rowsInput\"\n        [fullWidth]=\"true\"\n        (valueChange)=\"onRowsChange($event)\">\n      </cqa-custom-input>\n      <cqa-custom-input\n        label=\"Columns\"\n        placeholder=\"e.g. 3\"\n        type=\"number\"\n        [value]=\"columnsInput\"\n        [fullWidth]=\"true\"\n        (valueChange)=\"onColumnsChange($event)\">\n      </cqa-custom-input>\n    </div>\n    <span *ngIf=\"matrixError\" class=\"cqa-text-xs cqa-text-red-600\">{{ matrixError }}</span>\n    <span *ngIf=\"!matrixError\" class=\"cqa-text-xs cqa-text-gray-500\">\n      Defaults to <code>row_1…row_N</code> and <code>col_1…col_M</code>.\n    </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"]}
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"]}
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmV3LXRlc3QtZGF0YS1wcm9maWxlLWRpYWxvZy5tb2RlbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL25ldy10ZXN0LWRhdGEtcHJvZmlsZS1kaWFsb2cvbmV3LXRlc3QtZGF0YS1wcm9maWxlLWRpYWxvZy5tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIFRkcERpYWxvZ1Blcm1pc3Npb24gPSAnUk8nIHwgJ1JXJztcblxuZXhwb3J0IGludGVyZmFjZSBUZHBEaWFsb2dWYWx1ZSB7XG4gIG5hbWU6IHN0cmluZztcbiAgZGVzY3JpcHRpb246IHN0cmluZyB8IG51bGw7XG4gIHJlYWRXcml0ZU1vZGU6IFRkcERpYWxvZ1Blcm1pc3Npb247XG4gIGNvbHVtbnM6IHN0cmluZ1tdO1xuICByb3dzOiBzdHJpbmdbXTtcbiAgZW52aXJvbm1lbnRJZHM6IG51bWJlcltdO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFRkcEVudmlyb25tZW50T3B0aW9uIHtcbiAgaWQ6IG51bWJlcjtcbiAgbmFtZTogc3RyaW5nO1xuICBjb2xvcj86IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBOZXdUZXN0RGF0YVByb2ZpbGVEaWFsb2dJbnB1dHMge1xuICBtb2RlPzogJ2NyZWF0ZScgfCAnZWRpdCc7XG4gIGluaXRpYWxWYWx1ZT86IFBhcnRpYWw8VGRwRGlhbG9nVmFsdWU+O1xuICBleGlzdGluZ05hbWVzPzogc3RyaW5nW107XG4gIGVudmlyb25tZW50cz86IFRkcEVudmlyb25tZW50T3B0aW9uW107XG59XG4iXX0=
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmV3LXRlc3QtZGF0YS1wcm9maWxlLWRpYWxvZy5tb2RlbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL25ldy10ZXN0LWRhdGEtcHJvZmlsZS1kaWFsb2cvbmV3LXRlc3QtZGF0YS1wcm9maWxlLWRpYWxvZy5tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIFRkcERpYWxvZ1Blcm1pc3Npb24gPSAnUk8nIHwgJ1JXJztcblxuZXhwb3J0IGludGVyZmFjZSBUZHBEaWFsb2dWYWx1ZSB7XG4gIG5hbWU6IHN0cmluZztcbiAgZGVzY3JpcHRpb246IHN0cmluZyB8IG51bGw7XG4gIHJlYWRXcml0ZU1vZGU6IFRkcERpYWxvZ1Blcm1pc3Npb247XG4gIGNvbHVtbnM6IHN0cmluZ1tdO1xuICBlbnZpcm9ubWVudElkczogbnVtYmVyW107XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgVGRwRW52aXJvbm1lbnRPcHRpb24ge1xuICBpZDogbnVtYmVyO1xuICBuYW1lOiBzdHJpbmc7XG4gIGNvbG9yPzogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE5ld1Rlc3REYXRhUHJvZmlsZURpYWxvZ0lucHV0cyB7XG4gIG1vZGU/OiAnY3JlYXRlJyB8ICdlZGl0JztcbiAgaW5pdGlhbFZhbHVlPzogUGFydGlhbDxUZHBEaWFsb2dWYWx1ZT47XG4gIGV4aXN0aW5nTmFtZXM/OiBzdHJpbmdbXTtcbiAgZW52aXJvbm1lbnRzPzogVGRwRW52aXJvbm1lbnRPcHRpb25bXTtcbiAgbG9ja2VkQ29sdW1ucz86IHN0cmluZ1tdO1xufVxuIl19
@@ -4,8 +4,7 @@ import * as i1 from "@angular/material/icon";
4
4
  import * as i2 from "@angular/common";
5
5
  import * as i3 from "@angular/material/tooltip";
6
6
  export class SegmentControlComponent {
7
- constructor(cdr) {
8
- this.cdr = cdr;
7
+ constructor() {
9
8
  this.segments = [
10
9
  { label: 'Tab Group', value: 'tab-group-1' },
11
10
  { label: 'Tab Group', value: 'tab-group-2' },
@@ -181,24 +180,18 @@ export class SegmentControlComponent {
181
180
  if (!container || buttons.length === 0) {
182
181
  this.indicatorVisible = false;
183
182
  this.indicatorStyle = {};
184
- // OnPush parents (e.g. template-variables-form) won't tick this child
185
- // automatically when its own state mutates inside a microtask — explicit
186
- // markForCheck is required for the indicator binding to update.
187
- this.cdr.markForCheck();
188
183
  return;
189
184
  }
190
185
  const index = preferredIndex ?? buttons.findIndex((button, idx) => this.segments[idx]?.value === this.value);
191
186
  if (index === -1) {
192
187
  this.indicatorVisible = false;
193
188
  this.indicatorStyle = {};
194
- this.cdr.markForCheck();
195
189
  return;
196
190
  }
197
191
  const buttonEl = buttons[index]?.nativeElement;
198
192
  if (!buttonEl) {
199
193
  this.indicatorVisible = false;
200
194
  this.indicatorStyle = {};
201
- this.cdr.markForCheck();
202
195
  return;
203
196
  }
204
197
  const containerRect = container.getBoundingClientRect();
@@ -216,19 +209,18 @@ export class SegmentControlComponent {
216
209
  backgroundColor: isDisabled ? '#9BA0F4' : '#3F43EE',
217
210
  };
218
211
  this.indicatorVisible = true;
219
- this.cdr.markForCheck();
220
212
  });
221
213
  }
222
214
  get isIndicatorVisible() {
223
215
  return this.indicatorVisible;
224
216
  }
225
217
  }
226
- SegmentControlComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SegmentControlComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
218
+ SegmentControlComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SegmentControlComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
227
219
  SegmentControlComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: SegmentControlComponent, selector: "cqa-segment-control", inputs: { segments: "segments", value: "value", disabled: "disabled", containerBgColor: "containerBgColor", fullWidth: "fullWidth", size: "size" }, outputs: { valueChange: "valueChange" }, host: { properties: { "style.display": "this.hostDisplay", "style.width": "this.hostWidth" }, classAttribute: "cqa-ui-root" }, viewQueries: [{ propertyName: "segmentContainer", first: true, predicate: ["segmentContainer"], descendants: true }, { propertyName: "segmentButtons", predicate: ["segmentButton"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"cqa-ui-root\" [style.display]=\"fullWidth ? 'block' : 'inline-block'\" [style.width]=\"fullWidth ? '100%' : null\">\n <div\n #segmentContainer\n class=\"cqa-relative cqa-flex-row cqa-items-start cqa-bg-surface-light cqa-rounded-[8px]\"\n [ngClass]=\"[\n fullWidth ? 'cqa-flex' : 'cqa-inline-flex',\n size === 'lg' ? 'cqa-h-[40px] cqa-p-[4px]' : 'cqa-h-[31.5px] cqa-p-[3.5px]'\n ]\"\n role=\"tablist\"\n [attr.aria-disabled]=\"disabled || null\"\n [ngStyle]=\"containerBgColor ? {'background-color': containerBgColor} : null\"\n >\n <div\n class=\"cqa-absolute cqa-rounded-[8px] cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-pointer-events-none\"\n [class.cqa-opacity-0]=\"!isIndicatorVisible\" [ngStyle]=\"indicatorStyle\" aria-hidden=\"true\"></div>\n\n <button *ngFor=\"let segment of segments; index as index; trackBy: trackByValue\" #segmentButton type=\"button\"\n role=\"tab\"\n class=\"cqa-relative cqa-z-0 cqa-flex cqa-flex-col cqa-justify-center cqa-items-center cqa-rounded-[8px] cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-whitespace-nowrap cqa-text-center focus:cqa-outline-none focus-visible:cqa-outline-none focus-visible:cqa-ring-0 focus-visible:cqa-ring-offset-0\"\n [ngClass]=\"{\n 'cqa-flex-1 cqa-min-w-0': fullWidth,\n 'cqa-flex-none': !fullWidth,\n 'cqa-text-white cqa-font-medium': isSelected(segment),\n 'cqa-text-muted': !isSelected(segment) && !(disabled || segment.disabled),\n 'cqa-cursor-not-allowed': disabled || segment.disabled,\n 'cqa-text-disabled': (disabled || segment.disabled) && !isSelected(segment),\n 'cqa-hover:cqa-text-black': !isSelected(segment) && !disabled && !segment.disabled,\n 'cqa-px-[16px] cqa-py-[6px] cqa-h-[32px]': size === 'lg',\n 'cqa-px-[14px] cqa-py-[3.5px] cqa-h-[25px]': size !== 'lg'\n }\" [disabled]=\"disabled || segment.disabled\" [attr.aria-selected]=\"isSelected(segment)\"\n [attr.tabindex]=\"!disabled && !segment.disabled ? (isSelected(segment) ? 0 : -1) : -1\"\n (click)=\"select(segment, index)\" (keydown)=\"onKeyDown($event, index)\">\n <span\n class=\"cqa-flex cqa-gap-1 cqa-items-center cqa-justify-center cqa-h-[18px] cqa-font-['Inter'] cqa-font-normal cqa-text-[12px] cqa-leading-[12px] cqa-text-center cqa-align-middle\">\n \n <mat-icon *ngIf=\"segment?.icon\" class=\"!cqa-w-[12px] !cqa-h-[12px] !cqa-text-[12px]\" >\n {{ segment?.icon }}\n </mat-icon>\n\n {{ segment.label }}\n\n <span *ngIf=\"segment?.count != null\"\n class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-ml-1 cqa-px-[6px] cqa-rounded-full cqa-text-[10px] cqa-leading-[14px] cqa-font-semibold cqa-min-w-[16px]\"\n [ngClass]=\"isSelected(segment) ? 'cqa-bg-white cqa-text-[#3F43EE]' : 'cqa-bg-[#E5E7EB] cqa-text-[#475569]'\">\n {{ segment.count }}\n </span>\n\n <span *ngIf=\"segment?.tooltip\"\n style=\"display: inline-flex; align-items: center; justify-content: center; margin-left: 4px; cursor: help;\"\n [matTooltip]=\"segment.tooltip\"\n matTooltipPosition=\"above\"\n matTooltipShowDelay=\"0\"\n (click)=\"$event.stopPropagation()\">\n <mat-icon style=\"width: 14px; height: 14px; font-size: 14px; opacity: 0.8;\">info</mat-icon>\n </span>\n </span>\n </button>\n</div>", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }], directives: [{ type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }] });
228
220
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SegmentControlComponent, decorators: [{
229
221
  type: Component,
230
222
  args: [{ selector: 'cqa-segment-control', host: { class: 'cqa-ui-root' }, template: "<div class=\"cqa-ui-root\" [style.display]=\"fullWidth ? 'block' : 'inline-block'\" [style.width]=\"fullWidth ? '100%' : null\">\n <div\n #segmentContainer\n class=\"cqa-relative cqa-flex-row cqa-items-start cqa-bg-surface-light cqa-rounded-[8px]\"\n [ngClass]=\"[\n fullWidth ? 'cqa-flex' : 'cqa-inline-flex',\n size === 'lg' ? 'cqa-h-[40px] cqa-p-[4px]' : 'cqa-h-[31.5px] cqa-p-[3.5px]'\n ]\"\n role=\"tablist\"\n [attr.aria-disabled]=\"disabled || null\"\n [ngStyle]=\"containerBgColor ? {'background-color': containerBgColor} : null\"\n >\n <div\n class=\"cqa-absolute cqa-rounded-[8px] cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-pointer-events-none\"\n [class.cqa-opacity-0]=\"!isIndicatorVisible\" [ngStyle]=\"indicatorStyle\" aria-hidden=\"true\"></div>\n\n <button *ngFor=\"let segment of segments; index as index; trackBy: trackByValue\" #segmentButton type=\"button\"\n role=\"tab\"\n class=\"cqa-relative cqa-z-0 cqa-flex cqa-flex-col cqa-justify-center cqa-items-center cqa-rounded-[8px] cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-whitespace-nowrap cqa-text-center focus:cqa-outline-none focus-visible:cqa-outline-none focus-visible:cqa-ring-0 focus-visible:cqa-ring-offset-0\"\n [ngClass]=\"{\n 'cqa-flex-1 cqa-min-w-0': fullWidth,\n 'cqa-flex-none': !fullWidth,\n 'cqa-text-white cqa-font-medium': isSelected(segment),\n 'cqa-text-muted': !isSelected(segment) && !(disabled || segment.disabled),\n 'cqa-cursor-not-allowed': disabled || segment.disabled,\n 'cqa-text-disabled': (disabled || segment.disabled) && !isSelected(segment),\n 'cqa-hover:cqa-text-black': !isSelected(segment) && !disabled && !segment.disabled,\n 'cqa-px-[16px] cqa-py-[6px] cqa-h-[32px]': size === 'lg',\n 'cqa-px-[14px] cqa-py-[3.5px] cqa-h-[25px]': size !== 'lg'\n }\" [disabled]=\"disabled || segment.disabled\" [attr.aria-selected]=\"isSelected(segment)\"\n [attr.tabindex]=\"!disabled && !segment.disabled ? (isSelected(segment) ? 0 : -1) : -1\"\n (click)=\"select(segment, index)\" (keydown)=\"onKeyDown($event, index)\">\n <span\n class=\"cqa-flex cqa-gap-1 cqa-items-center cqa-justify-center cqa-h-[18px] cqa-font-['Inter'] cqa-font-normal cqa-text-[12px] cqa-leading-[12px] cqa-text-center cqa-align-middle\">\n \n <mat-icon *ngIf=\"segment?.icon\" class=\"!cqa-w-[12px] !cqa-h-[12px] !cqa-text-[12px]\" >\n {{ segment?.icon }}\n </mat-icon>\n\n {{ segment.label }}\n\n <span *ngIf=\"segment?.count != null\"\n class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-ml-1 cqa-px-[6px] cqa-rounded-full cqa-text-[10px] cqa-leading-[14px] cqa-font-semibold cqa-min-w-[16px]\"\n [ngClass]=\"isSelected(segment) ? 'cqa-bg-white cqa-text-[#3F43EE]' : 'cqa-bg-[#E5E7EB] cqa-text-[#475569]'\">\n {{ segment.count }}\n </span>\n\n <span *ngIf=\"segment?.tooltip\"\n style=\"display: inline-flex; align-items: center; justify-content: center; margin-left: 4px; cursor: help;\"\n [matTooltip]=\"segment.tooltip\"\n matTooltipPosition=\"above\"\n matTooltipShowDelay=\"0\"\n (click)=\"$event.stopPropagation()\">\n <mat-icon style=\"width: 14px; height: 14px; font-size: 14px; opacity: 0.8;\">info</mat-icon>\n </span>\n </span>\n </button>\n</div>", styles: [] }]
231
- }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { segments: [{
223
+ }], propDecorators: { segments: [{
232
224
  type: Input
233
225
  }], value: [{
234
226
  type: Input
@@ -255,4 +247,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImpor
255
247
  type: ViewChild,
256
248
  args: ['segmentContainer']
257
249
  }] } });
258
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"segment-control.component.js","sourceRoot":"","sources":["../../../../../src/lib/segment-control/segment-control.component.ts","../../../../../src/lib/segment-control/segment-control.component.html"],"names":[],"mappings":"AAAA,OAAO,EAGL,SAAS,EAET,YAAY,EACZ,WAAW,EACX,KAAK,EAGL,MAAM,EAGN,SAAS,EACT,YAAY,GACb,MAAM,eAAe,CAAC;;;;;AAkBvB,MAAM,OAAO,uBAAuB;IAgClC,YAA6B,GAAsB;QAAtB,QAAG,GAAH,GAAG,CAAmB;QA9B1C,aAAQ,GAAoB;YACnC,EAAE,KAAK,EAAE,WAAW,EAAE,KAAK,EAAE,aAAa,EAAE;YAC5C,EAAE,KAAK,EAAE,WAAW,EAAE,KAAK,EAAE,aAAa,EAAE;SAC7C,CAAC;QAOO,aAAQ,GAAG,KAAK,CAAC;QACjB,qBAAgB,GAAG,EAAE,CAAC;QAE/B,yFAAyF;QAChF,cAAS,GAAG,KAAK,CAAC;QAElB,SAAI,GAAgB,IAAI,CAAC;QAKxB,gBAAW,GAAG,IAAI,YAAY,EAAU,CAAC;QAKnD,mBAAc,GAA2B,EAAE,CAAC;QAC5C,qBAAgB,GAAG,KAAK,CAAC;IAG6B,CAAC;IAZvD,IAAkC,WAAW,KAAK,OAAO,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;IAC3F,IAAgC,SAAS,KAAK,OAAO,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;IAatF,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,UAAU,CAAC,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE;YAC3C,IAAI,CAAC,mBAAmB,EAAE,CAAC;SAC5B;IACH,CAAC;IAED,eAAe;QACb,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC;QAC5F,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC3B,IAAI,CAAC,eAAe,EAAE,CAAC;IACzB,CAAC;IAED,WAAW;QACT,IAAI,CAAC,gBAAgB,EAAE,WAAW,EAAE,EAAE,CAAC;IACzC,CAAC;IAED,YAAY,CAAC,MAAc,EAAE,MAAqB;QAChD,OAAO,MAAM,CAAC,KAAK,CAAC;IACtB,CAAC;IAED,UAAU,CAAC,MAAqB;QAC9B,OAAO,MAAM,CAAC,KAAK,KAAK,IAAI,CAAC,KAAK,CAAC;IACrC,CAAC;IAED,MAAM,CAAC,MAAqB,EAAE,KAAa;QACzC,IAAI,IAAI,CAAC,QAAQ,IAAI,MAAM,CAAC,QAAQ,EAAE;YACpC,OAAO;SACR;QAED,MAAM,SAAS,GAAG,MAAM,CAAC,KAAK,CAAC;QAC/B,IAAI,SAAS,KAAK,IAAI,CAAC,KAAK,EAAE;YAC5B,IAAI,CAAC,KAAK,GAAG,SAAS,CAAC;YACvB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;SAClC;QAED,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;QACxB,IAAI,CAAC,eAAe,EAAE,CAAC;IACzB,CAAC;IAED,SAAS,CAAC,KAAoB,EAAE,YAAoB;QAClD,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,OAAO;SACR;QAED,QAAQ,KAAK,CAAC,GAAG,EAAE;YACjB,KAAK,YAAY,CAAC;YAClB,KAAK,WAAW;gBACd,KAAK,CAAC,cAAc,EAAE,CAAC;gBACvB,IAAI,CAAC,aAAa,CAAC,CAAC,EAAE,YAAY,CAAC,CAAC;gBACpC,MAAM;YACR,KAAK,WAAW,CAAC;YACjB,KAAK,SAAS;gBACZ,KAAK,CAAC,cAAc,EAAE,CAAC;gBACvB,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,EAAE,YAAY,CAAC,CAAC;gBACrC,MAAM;YACR,KAAK,MAAM;gBACT,KAAK,CAAC,cAAc,EAAE,CAAC;gBACvB,IAAI,CAAC,kBAAkB,EAAE,CAAC;gBAC1B,MAAM;YACR,KAAK,KAAK;gBACR,KAAK,CAAC,cAAc,EAAE,CAAC;gBACvB,IAAI,CAAC,iBAAiB,EAAE,CAAC;gBACzB,MAAM;YACR,KAAK,GAAG,CAAC;YACT,KAAK,OAAO;gBACV,KAAK,CAAC,cAAc,EAAE,CAAC;gBACvB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,EAAE,YAAY,CAAC,CAAC;gBACvD,MAAM;YACR;gBACE,MAAM;SACT;IACH,CAAC;IAEO,aAAa,CAAC,IAAY,EAAE,UAAkB;QACpD,MAAM,cAAc,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAChD,IAAI,cAAc,CAAC,MAAM,KAAK,CAAC,EAAE;YAC/B,OAAO;SACR;QAED,MAAM,mBAAmB,GAAG,cAAc,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;QAC/D,MAAM,aAAa,GAAG,IAAI,CAAC,gBAAgB,CAAC,cAAc,CAAC,CAAC;QAC5D,MAAM,SAAS,GAAG,mBAAmB,IAAI,CAAC,CAAC,CAAC,CAAC,mBAAmB,CAAC,CAAC,CAAC,aAAa,CAAC;QACjF,MAAM,YAAY,GAAG,CAAC,SAAS,GAAG,IAAI,GAAG,cAAc,CAAC,MAAM,CAAC,GAAG,cAAc,CAAC,MAAM,CAAC;QACxF,MAAM,WAAW,GAAG,cAAc,CAAC,YAAY,CAAC,CAAC;QAEjD,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE,WAAW,CAAC,CAAC;IACvD,CAAC;IAEO,kBAAkB;QACxB,MAAM,cAAc,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAChD,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE;YAC7B,MAAM,KAAK,GAAG,cAAc,CAAC,CAAC,CAAC,CAAC;YAChC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,KAAK,CAAC,CAAC;YACzC,IAAI,CAAC,eAAe,EAAE,CAAC;SACxB;IACH,CAAC;IAEO,iBAAiB;QACvB,MAAM,cAAc,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAChD,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE;YAC7B,MAAM,KAAK,GAAG,cAAc,CAAC,cAAc,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;YACxD,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,KAAK,CAAC,CAAC;YACzC,IAAI,CAAC,eAAe,EAAE,CAAC;SACxB;IACH,CAAC;IAEO,iBAAiB;QACvB,OAAO,IAAI,CAAC,QAAQ;aACjB,GAAG,CAAC,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE,CAAC,CAAC,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,CAAC;aAC3C,MAAM,CAAC,CAAC,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC;aACxC,GAAG,CAAC,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,KAAK,CAAC,CAAC;IAC/B,CAAC;IAEO,gBAAgB,CAAC,cAAwB;QAC/C,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,KAAK,KAAK,IAAI,CAAC,KAAK,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;QACrG,IAAI,OAAO,IAAI,CAAC,EAAE;YAChB,OAAO,cAAc,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;SACxC;QAED,OAAO,CAAC,CAAC;IACX,CAAC;IAED;;;OAGG;IACK,mBAAmB;QACzB,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;QACnE,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE;YACxB,IAAI,CAAC,KAAK,GAAG,SAAS,CAAC;YACvB,OAAO;SACR;QAED,2FAA2F;QAC3F,IAAI,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,KAAK,KAAK,IAAI,CAAC,KAAK,CAAC,EAAE;YAC/E,MAAM,iBAAiB,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;YAC3C,sCAAsC;YACtC,IAAI,IAAI,CAAC,KAAK,KAAK,iBAAiB,EAAE;gBACpC,IAAI,CAAC,KAAK,GAAG,iBAAiB,CAAC;gBAC/B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;aACnC;YACD,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,KAAK,KAAK,iBAAiB,CAAC,CAAC;YACtF,IAAI,KAAK,IAAI,CAAC,EAAE;gBACd,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;gBACxB,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;aAC7B;YACD,OAAO;SACR;QAED,qEAAqE;QACrE,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,KAAK,KAAK,IAAI,CAAC,KAAK,CAAC,CAAC;QAC7E,IAAI,QAAQ,EAAE,QAAQ,EAAE;YACtB,MAAM,iBAAiB,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;YAC3C,sCAAsC;YACtC,IAAI,IAAI,CAAC,KAAK,KAAK,iBAAiB,EAAE;gBACpC,IAAI,CAAC,KAAK,GAAG,iBAAiB,CAAC;gBAC/B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;aACnC;YACD,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,KAAK,KAAK,iBAAiB,CAAC,CAAC;YACtF,IAAI,KAAK,IAAI,CAAC,EAAE;gBACd,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;gBACxB,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;aAC7B;YACD,OAAO;SACR;QAED,IAAI,CAAC,eAAe,EAAE,CAAC;IACzB,CAAC;IAEO,WAAW,CAAC,KAAa;QAC/B,cAAc,CAAC,GAAG,EAAE;YAClB,MAAM,MAAM,GAAG,IAAI,CAAC,cAAc,EAAE,GAAG,CAAC,KAAK,CAAC,EAAE,aAAa,CAAC;YAC9D,MAAM,EAAE,KAAK,EAAE,CAAC;QAClB,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,eAAe,CAAC,cAAuB;QAC7C,cAAc,CAAC,GAAG,EAAE;YAClB,MAAM,SAAS,GAAG,IAAI,CAAC,gBAAgB,EAAE,aAAa,CAAC;YACvD,MAAM,OAAO,GAAG,IAAI,CAAC,cAAc,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC;YACrD,IAAI,CAAC,SAAS,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE;gBACtC,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;gBAC9B,IAAI,CAAC,cAAc,GAAG,EAAE,CAAC;gBACzB,sEAAsE;gBACtE,yEAAyE;gBACzE,gEAAgE;gBAChE,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;gBACxB,OAAO;aACR;YAED,MAAM,KAAK,GAAG,cAAc,IAAI,OAAO,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,KAAK,KAAK,IAAI,CAAC,KAAK,CAAC,CAAC;YAC7G,IAAI,KAAK,KAAK,CAAC,CAAC,EAAE;gBAChB,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;gBAC9B,IAAI,CAAC,cAAc,GAAG,EAAE,CAAC;gBACzB,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;gBACxB,OAAO;aACR;YAED,MAAM,QAAQ,GAAG,OAAO,CAAC,KAAK,CAAC,EAAE,aAAa,CAAC;YAC/C,IAAI,CAAC,QAAQ,EAAE;gBACb,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;gBAC9B,IAAI,CAAC,cAAc,GAAG,EAAE,CAAC;gBACzB,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;gBACxB,OAAO;aACR;YAED,MAAM,aAAa,GAAG,SAAS,CAAC,qBAAqB,EAAE,CAAC;YACxD,MAAM,UAAU,GAAG,QAAQ,CAAC,qBAAqB,EAAE,CAAC;YACpD,MAAM,UAAU,GAAG,QAAQ,CAAC,UAAU,CAAC;YACvC,MAAM,SAAS,GAAG,QAAQ,CAAC,SAAS,CAAC;YACrC,MAAM,KAAK,GAAG,QAAQ,CAAC,WAAW,CAAC;YACnC,MAAM,MAAM,GAAG,QAAQ,CAAC,YAAY,CAAC;YAErC,MAAM,UAAU,GAAG,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,QAAQ,CAAC;YACnE,IAAI,CAAC,cAAc,GAAG;gBACpB,KAAK,EAAE,GAAG,KAAK,IAAI;gBACnB,MAAM,EAAE,GAAG,MAAM,IAAI;gBACrB,IAAI,EAAE,GAAG,UAAU,IAAI;gBACvB,GAAG,EAAE,GAAG,SAAS,IAAI;gBACrB,eAAe,EAAE,UAAU,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS;aACpD,CAAC;YACF,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;YAC7B,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;QAC1B,CAAC,CAAC,CAAC;IACL,CAAC;IAED,IAAI,kBAAkB;QACpB,OAAO,IAAI,CAAC,gBAAgB,CAAC;IAC/B,CAAC;;oHAtQU,uBAAuB;wGAAvB,uBAAuB,ulBCjCpC,k3GAyDM;2FDxBO,uBAAuB;kBANnC,SAAS;+BACE,qBAAqB,QAGzB,EAAE,KAAK,EAAE,aAAa,EAAE;wGAIrB,QAAQ;sBAAhB,KAAK;gBASG,KAAK;sBAAb,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,gBAAgB;sBAAxB,KAAK;gBAGG,SAAS;sBAAjB,KAAK;gBAEG,IAAI;sBAAZ,KAAK;gBAE4B,WAAW;sBAA5C,WAAW;uBAAC,eAAe;gBACI,SAAS;sBAAxC,WAAW;uBAAC,aAAa;gBAEhB,WAAW;sBAApB,MAAM;gBAEwB,cAAc;sBAA5C,YAAY;uBAAC,eAAe;gBACE,gBAAgB;sBAA9C,SAAS;uBAAC,kBAAkB","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  HostBinding,\n  Input,\n  OnDestroy,\n  OnChanges,\n  Output,\n  QueryList,\n  SimpleChanges,\n  ViewChild,\n  ViewChildren,\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\n\nexport type SegmentOption = {\n  label: string;\n  value: string;\n  disabled?: boolean;\n  icon?: string;\n  tooltip?: string;\n  count?: number;\n};\n\n@Component({\n  selector: 'cqa-segment-control',\n  templateUrl: './segment-control.component.html',\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' }\n})\nexport class SegmentControlComponent implements OnChanges, AfterViewInit, OnDestroy {\n\n  @Input() segments: SegmentOption[] = [\n    { label: 'Tab Group', value: 'tab-group-1' },\n    { label: 'Tab Group', value: 'tab-group-2' },\n  ];\n\n  /**\n   * The currently selected segment value.\n   * If not provided, defaults to the first enabled segment.\n   */\n  @Input() value?: string;\n  @Input() disabled = false;\n  @Input() containerBgColor = '';\n\n  /** When true, the control stretches to fill its parent and buttons share equal width. */\n  @Input() fullWidth = false;\n\n  @Input() size: 'md' | 'lg' = 'md';\n\n  @HostBinding('style.display') get hostDisplay() { return this.fullWidth ? 'block' : null; }\n  @HostBinding('style.width') get hostWidth() { return this.fullWidth ? '100%' : null; }\n\n  @Output() valueChange = new EventEmitter<string>();\n\n  @ViewChildren('segmentButton') segmentButtons!: QueryList<ElementRef<HTMLButtonElement>>;\n  @ViewChild('segmentContainer') segmentContainer?: ElementRef<HTMLDivElement>;\n\n  indicatorStyle: Record<string, string> = {};\n  indicatorVisible = false;\n  private buttonChangesSub?: Subscription;\n\n  constructor(private readonly cdr: ChangeDetectorRef) {}\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['segments'] || changes['value']) {\n      this.ensureSelectedValue();\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.buttonChangesSub = this.segmentButtons.changes.subscribe(() => this.updateIndicator());\n    this.ensureSelectedValue();\n    this.updateIndicator();\n  }\n\n  ngOnDestroy(): void {\n    this.buttonChangesSub?.unsubscribe?.();\n  }\n\n  trackByValue(_index: number, option: SegmentOption): string {\n    return option.value;\n  }\n\n  isSelected(option: SegmentOption): boolean {\n    return option.value === this.value;\n  }\n\n  select(option: SegmentOption, index: number): void {\n    if (this.disabled || option.disabled) {\n      return;\n    }\n\n    const nextValue = option.value;\n    if (nextValue !== this.value) {\n      this.value = nextValue;\n      this.valueChange.emit(nextValue);\n    }\n\n    this.focusButton(index);\n    this.updateIndicator();\n  }\n\n  onKeyDown(event: KeyboardEvent, currentIndex: number): void {\n    if (this.disabled) {\n      return;\n    }\n\n    switch (event.key) {\n      case 'ArrowRight':\n      case 'ArrowDown':\n        event.preventDefault();\n        this.moveSelection(1, currentIndex);\n        break;\n      case 'ArrowLeft':\n      case 'ArrowUp':\n        event.preventDefault();\n        this.moveSelection(-1, currentIndex);\n        break;\n      case 'Home':\n        event.preventDefault();\n        this.selectFirstEnabled();\n        break;\n      case 'End':\n        event.preventDefault();\n        this.selectLastEnabled();\n        break;\n      case ' ':\n      case 'Enter':\n        event.preventDefault();\n        this.select(this.segments[currentIndex], currentIndex);\n        break;\n      default:\n        break;\n    }\n  }\n\n  private moveSelection(step: 1 | -1, startIndex: number): void {\n    const enabledIndexes = this.getEnabledIndexes();\n    if (enabledIndexes.length === 0) {\n      return;\n    }\n\n    const currentEnabledIndex = enabledIndexes.indexOf(startIndex);\n    const fallbackIndex = this.getSelectedIndex(enabledIndexes);\n    const baseIndex = currentEnabledIndex >= 0 ? currentEnabledIndex : fallbackIndex;\n    const nextPosition = (baseIndex + step + enabledIndexes.length) % enabledIndexes.length;\n    const targetIndex = enabledIndexes[nextPosition];\n\n    this.select(this.segments[targetIndex], targetIndex);\n  }\n\n  private selectFirstEnabled(): void {\n    const enabledIndexes = this.getEnabledIndexes();\n    if (enabledIndexes.length > 0) {\n      const index = enabledIndexes[0];\n      this.select(this.segments[index], index);\n      this.updateIndicator();\n    }\n  }\n\n  private selectLastEnabled(): void {\n    const enabledIndexes = this.getEnabledIndexes();\n    if (enabledIndexes.length > 0) {\n      const index = enabledIndexes[enabledIndexes.length - 1];\n      this.select(this.segments[index], index);\n      this.updateIndicator();\n    }\n  }\n\n  private getEnabledIndexes(): number[] {\n    return this.segments\n      .map((option, index) => ({ option, index }))\n      .filter(({ option }) => !option.disabled)\n      .map(({ index }) => index);\n  }\n\n  private getSelectedIndex(enabledIndexes: number[]): number {\n    const current = this.segments.findIndex((option) => option.value === this.value && !option.disabled);\n    if (current >= 0) {\n      return enabledIndexes.indexOf(current);\n    }\n\n    return 0;\n  }\n\n  /**\n   * Ensures a valid value is selected.\n   * If no value is provided or the value is invalid, defaults to the first enabled segment.\n   */\n  private ensureSelectedValue(): void {\n    const enabled = this.segments.filter((option) => !option.disabled);\n    if (enabled.length === 0) {\n      this.value = undefined;\n      return;\n    }\n\n    // If no value is set, or value doesn't exist in segments, default to first enabled segment\n    if (!this.value || !this.segments.some((option) => option.value === this.value)) {\n      const firstEnabledValue = enabled[0].value;\n      // Only emit if value actually changed\n      if (this.value !== firstEnabledValue) {\n        this.value = firstEnabledValue;\n        this.valueChange.emit(this.value);\n      }\n      const index = this.segments.findIndex((option) => option.value === firstEnabledValue);\n      if (index >= 0) {\n        this.focusButton(index);\n        this.updateIndicator(index);\n      }\n      return;\n    }\n\n    // If the selected value is disabled, switch to first enabled segment\n    const selected = this.segments.find((option) => option.value === this.value);\n    if (selected?.disabled) {\n      const firstEnabledValue = enabled[0].value;\n      // Only emit if value actually changed\n      if (this.value !== firstEnabledValue) {\n        this.value = firstEnabledValue;\n        this.valueChange.emit(this.value);\n      }\n      const index = this.segments.findIndex((option) => option.value === firstEnabledValue);\n      if (index >= 0) {\n        this.focusButton(index);\n        this.updateIndicator(index);\n      }\n      return;\n    }\n    \n    this.updateIndicator();\n  }\n\n  private focusButton(index: number): void {\n    queueMicrotask(() => {\n      const button = this.segmentButtons?.get(index)?.nativeElement;\n      button?.focus();\n    });\n  }\n\n  private updateIndicator(preferredIndex?: number): void {\n    queueMicrotask(() => {\n      const container = this.segmentContainer?.nativeElement;\n      const buttons = this.segmentButtons?.toArray() ?? [];\n      if (!container || buttons.length === 0) {\n        this.indicatorVisible = false;\n        this.indicatorStyle = {};\n        // OnPush parents (e.g. template-variables-form) won't tick this child\n        // automatically when its own state mutates inside a microtask — explicit\n        // markForCheck is required for the indicator binding to update.\n        this.cdr.markForCheck();\n        return;\n      }\n\n      const index = preferredIndex ?? buttons.findIndex((button, idx) => this.segments[idx]?.value === this.value);\n      if (index === -1) {\n        this.indicatorVisible = false;\n        this.indicatorStyle = {};\n        this.cdr.markForCheck();\n        return;\n      }\n\n      const buttonEl = buttons[index]?.nativeElement;\n      if (!buttonEl) {\n        this.indicatorVisible = false;\n        this.indicatorStyle = {};\n        this.cdr.markForCheck();\n        return;\n      }\n\n      const containerRect = container.getBoundingClientRect();\n      const buttonRect = buttonEl.getBoundingClientRect();\n      const offsetLeft = buttonEl.offsetLeft;\n      const offsetTop = buttonEl.offsetTop;\n      const width = buttonEl.offsetWidth;\n      const height = buttonEl.offsetHeight;\n\n      const isDisabled = this.disabled || this.segments[index]?.disabled;\n      this.indicatorStyle = {\n        width: `${width}px`,\n        height: `${height}px`,\n        left: `${offsetLeft}px`,\n        top: `${offsetTop}px`,\n        backgroundColor: isDisabled ? '#9BA0F4' : '#3F43EE',\n      };\n      this.indicatorVisible = true;\n      this.cdr.markForCheck();\n    });\n  }\n\n  get isIndicatorVisible(): boolean {\n    return this.indicatorVisible;\n  }\n}\n","<div class=\"cqa-ui-root\" [style.display]=\"fullWidth ? 'block' : 'inline-block'\" [style.width]=\"fullWidth ? '100%' : null\">\n  <div\n    #segmentContainer\n    class=\"cqa-relative cqa-flex-row cqa-items-start cqa-bg-surface-light cqa-rounded-[8px]\"\n    [ngClass]=\"[\n      fullWidth ? 'cqa-flex' : 'cqa-inline-flex',\n      size === 'lg' ? 'cqa-h-[40px] cqa-p-[4px]' : 'cqa-h-[31.5px] cqa-p-[3.5px]'\n    ]\"\n    role=\"tablist\"\n    [attr.aria-disabled]=\"disabled || null\"\n    [ngStyle]=\"containerBgColor ? {'background-color': containerBgColor} : null\"\n  >\n    <div\n      class=\"cqa-absolute cqa-rounded-[8px] cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-pointer-events-none\"\n      [class.cqa-opacity-0]=\"!isIndicatorVisible\" [ngStyle]=\"indicatorStyle\" aria-hidden=\"true\"></div>\n\n    <button *ngFor=\"let segment of segments; index as index; trackBy: trackByValue\" #segmentButton type=\"button\"\n      role=\"tab\"\n      class=\"cqa-relative cqa-z-0 cqa-flex cqa-flex-col cqa-justify-center cqa-items-center cqa-rounded-[8px] cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-whitespace-nowrap cqa-text-center focus:cqa-outline-none focus-visible:cqa-outline-none focus-visible:cqa-ring-0 focus-visible:cqa-ring-offset-0\"\n      [ngClass]=\"{\n        'cqa-flex-1 cqa-min-w-0': fullWidth,\n        'cqa-flex-none': !fullWidth,\n        'cqa-text-white cqa-font-medium': isSelected(segment),\n        'cqa-text-muted': !isSelected(segment) && !(disabled || segment.disabled),\n        'cqa-cursor-not-allowed': disabled || segment.disabled,\n        'cqa-text-disabled': (disabled || segment.disabled) && !isSelected(segment),\n        'cqa-hover:cqa-text-black': !isSelected(segment) && !disabled && !segment.disabled,\n        'cqa-px-[16px] cqa-py-[6px] cqa-h-[32px]': size === 'lg',\n        'cqa-px-[14px] cqa-py-[3.5px] cqa-h-[25px]': size !== 'lg'\n      }\" [disabled]=\"disabled || segment.disabled\" [attr.aria-selected]=\"isSelected(segment)\"\n      [attr.tabindex]=\"!disabled && !segment.disabled ? (isSelected(segment) ? 0 : -1) : -1\"\n      (click)=\"select(segment, index)\" (keydown)=\"onKeyDown($event, index)\">\n      <span\n        class=\"cqa-flex cqa-gap-1 cqa-items-center cqa-justify-center cqa-h-[18px] cqa-font-['Inter'] cqa-font-normal cqa-text-[12px] cqa-leading-[12px] cqa-text-center cqa-align-middle\">\n        \n      <mat-icon *ngIf=\"segment?.icon\" class=\"!cqa-w-[12px] !cqa-h-[12px] !cqa-text-[12px]\" >\n        {{ segment?.icon }}\n      </mat-icon>\n\n      {{ segment.label }}\n\n      <span *ngIf=\"segment?.count != null\"\n        class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-ml-1 cqa-px-[6px] cqa-rounded-full cqa-text-[10px] cqa-leading-[14px] cqa-font-semibold cqa-min-w-[16px]\"\n        [ngClass]=\"isSelected(segment) ? 'cqa-bg-white cqa-text-[#3F43EE]' : 'cqa-bg-[#E5E7EB] cqa-text-[#475569]'\">\n        {{ segment.count }}\n      </span>\n\n      <span *ngIf=\"segment?.tooltip\"\n        style=\"display: inline-flex; align-items: center; justify-content: center; margin-left: 4px; cursor: help;\"\n        [matTooltip]=\"segment.tooltip\"\n        matTooltipPosition=\"above\"\n        matTooltipShowDelay=\"0\"\n        (click)=\"$event.stopPropagation()\">\n        <mat-icon style=\"width: 14px; height: 14px; font-size: 14px; opacity: 0.8;\">info</mat-icon>\n      </span>\n    </span>\n  </button>\n</div>"]}
250
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"segment-control.component.js","sourceRoot":"","sources":["../../../../../src/lib/segment-control/segment-control.component.ts","../../../../../src/lib/segment-control/segment-control.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  Component,\n  ElementRef,\n  EventEmitter,\n  HostBinding,\n  Input,\n  OnDestroy,\n  OnChanges,\n  Output,\n  QueryList,\n  SimpleChanges,\n  ViewChild,\n  ViewChildren,\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\n\nexport type SegmentOption = {\n  label: string;\n  value: string;\n  disabled?: boolean;\n  icon?: string;\n  tooltip?: string;\n  count?: number;\n};\n\n@Component({\n  selector: 'cqa-segment-control',\n  templateUrl: './segment-control.component.html',\n  styleUrls: [],\n  host: { class: 'cqa-ui-root' }\n})\nexport class SegmentControlComponent implements OnChanges, AfterViewInit, OnDestroy {\n\n  @Input() segments: SegmentOption[] = [\n    { label: 'Tab Group', value: 'tab-group-1' },\n    { label: 'Tab Group', value: 'tab-group-2' },\n  ];\n\n  /**\n   * The currently selected segment value.\n   * If not provided, defaults to the first enabled segment.\n   */\n  @Input() value?: string;\n  @Input() disabled = false;\n  @Input() containerBgColor = '';\n\n  /** When true, the control stretches to fill its parent and buttons share equal width. */\n  @Input() fullWidth = false;\n\n  @Input() size: 'md' | 'lg' = 'md';\n\n  @HostBinding('style.display') get hostDisplay() { return this.fullWidth ? 'block' : null; }\n  @HostBinding('style.width') get hostWidth() { return this.fullWidth ? '100%' : null; }\n\n  @Output() valueChange = new EventEmitter<string>();\n\n  @ViewChildren('segmentButton') segmentButtons!: QueryList<ElementRef<HTMLButtonElement>>;\n  @ViewChild('segmentContainer') segmentContainer?: ElementRef<HTMLDivElement>;\n\n  indicatorStyle: Record<string, string> = {};\n  indicatorVisible = false;\n  private buttonChangesSub?: Subscription;\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['segments'] || changes['value']) {\n      this.ensureSelectedValue();\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.buttonChangesSub = this.segmentButtons.changes.subscribe(() => this.updateIndicator());\n    this.ensureSelectedValue();\n    this.updateIndicator();\n  }\n\n  ngOnDestroy(): void {\n    this.buttonChangesSub?.unsubscribe?.();\n  }\n\n  trackByValue(_index: number, option: SegmentOption): string {\n    return option.value;\n  }\n\n  isSelected(option: SegmentOption): boolean {\n    return option.value === this.value;\n  }\n\n  select(option: SegmentOption, index: number): void {\n    if (this.disabled || option.disabled) {\n      return;\n    }\n\n    const nextValue = option.value;\n    if (nextValue !== this.value) {\n      this.value = nextValue;\n      this.valueChange.emit(nextValue);\n    }\n\n    this.focusButton(index);\n    this.updateIndicator();\n  }\n\n  onKeyDown(event: KeyboardEvent, currentIndex: number): void {\n    if (this.disabled) {\n      return;\n    }\n\n    switch (event.key) {\n      case 'ArrowRight':\n      case 'ArrowDown':\n        event.preventDefault();\n        this.moveSelection(1, currentIndex);\n        break;\n      case 'ArrowLeft':\n      case 'ArrowUp':\n        event.preventDefault();\n        this.moveSelection(-1, currentIndex);\n        break;\n      case 'Home':\n        event.preventDefault();\n        this.selectFirstEnabled();\n        break;\n      case 'End':\n        event.preventDefault();\n        this.selectLastEnabled();\n        break;\n      case ' ':\n      case 'Enter':\n        event.preventDefault();\n        this.select(this.segments[currentIndex], currentIndex);\n        break;\n      default:\n        break;\n    }\n  }\n\n  private moveSelection(step: 1 | -1, startIndex: number): void {\n    const enabledIndexes = this.getEnabledIndexes();\n    if (enabledIndexes.length === 0) {\n      return;\n    }\n\n    const currentEnabledIndex = enabledIndexes.indexOf(startIndex);\n    const fallbackIndex = this.getSelectedIndex(enabledIndexes);\n    const baseIndex = currentEnabledIndex >= 0 ? currentEnabledIndex : fallbackIndex;\n    const nextPosition = (baseIndex + step + enabledIndexes.length) % enabledIndexes.length;\n    const targetIndex = enabledIndexes[nextPosition];\n\n    this.select(this.segments[targetIndex], targetIndex);\n  }\n\n  private selectFirstEnabled(): void {\n    const enabledIndexes = this.getEnabledIndexes();\n    if (enabledIndexes.length > 0) {\n      const index = enabledIndexes[0];\n      this.select(this.segments[index], index);\n      this.updateIndicator();\n    }\n  }\n\n  private selectLastEnabled(): void {\n    const enabledIndexes = this.getEnabledIndexes();\n    if (enabledIndexes.length > 0) {\n      const index = enabledIndexes[enabledIndexes.length - 1];\n      this.select(this.segments[index], index);\n      this.updateIndicator();\n    }\n  }\n\n  private getEnabledIndexes(): number[] {\n    return this.segments\n      .map((option, index) => ({ option, index }))\n      .filter(({ option }) => !option.disabled)\n      .map(({ index }) => index);\n  }\n\n  private getSelectedIndex(enabledIndexes: number[]): number {\n    const current = this.segments.findIndex((option) => option.value === this.value && !option.disabled);\n    if (current >= 0) {\n      return enabledIndexes.indexOf(current);\n    }\n\n    return 0;\n  }\n\n  /**\n   * Ensures a valid value is selected.\n   * If no value is provided or the value is invalid, defaults to the first enabled segment.\n   */\n  private ensureSelectedValue(): void {\n    const enabled = this.segments.filter((option) => !option.disabled);\n    if (enabled.length === 0) {\n      this.value = undefined;\n      return;\n    }\n\n    // If no value is set, or value doesn't exist in segments, default to first enabled segment\n    if (!this.value || !this.segments.some((option) => option.value === this.value)) {\n      const firstEnabledValue = enabled[0].value;\n      // Only emit if value actually changed\n      if (this.value !== firstEnabledValue) {\n        this.value = firstEnabledValue;\n        this.valueChange.emit(this.value);\n      }\n      const index = this.segments.findIndex((option) => option.value === firstEnabledValue);\n      if (index >= 0) {\n        this.focusButton(index);\n        this.updateIndicator(index);\n      }\n      return;\n    }\n\n    // If the selected value is disabled, switch to first enabled segment\n    const selected = this.segments.find((option) => option.value === this.value);\n    if (selected?.disabled) {\n      const firstEnabledValue = enabled[0].value;\n      // Only emit if value actually changed\n      if (this.value !== firstEnabledValue) {\n        this.value = firstEnabledValue;\n        this.valueChange.emit(this.value);\n      }\n      const index = this.segments.findIndex((option) => option.value === firstEnabledValue);\n      if (index >= 0) {\n        this.focusButton(index);\n        this.updateIndicator(index);\n      }\n      return;\n    }\n    \n    this.updateIndicator();\n  }\n\n  private focusButton(index: number): void {\n    queueMicrotask(() => {\n      const button = this.segmentButtons?.get(index)?.nativeElement;\n      button?.focus();\n    });\n  }\n\n  private updateIndicator(preferredIndex?: number): void {\n    queueMicrotask(() => {\n      const container = this.segmentContainer?.nativeElement;\n      const buttons = this.segmentButtons?.toArray() ?? [];\n      if (!container || buttons.length === 0) {\n        this.indicatorVisible = false;\n        this.indicatorStyle = {};\n        return;\n      }\n\n      const index = preferredIndex ?? buttons.findIndex((button, idx) => this.segments[idx]?.value === this.value);\n      if (index === -1) {\n        this.indicatorVisible = false;\n        this.indicatorStyle = {};\n        return;\n      }\n\n      const buttonEl = buttons[index]?.nativeElement;\n      if (!buttonEl) {\n        this.indicatorVisible = false;\n        this.indicatorStyle = {};\n        return;\n      }\n\n      const containerRect = container.getBoundingClientRect();\n      const buttonRect = buttonEl.getBoundingClientRect();\n      const offsetLeft = buttonEl.offsetLeft;\n      const offsetTop = buttonEl.offsetTop;\n      const width = buttonEl.offsetWidth;\n      const height = buttonEl.offsetHeight;\n\n      const isDisabled = this.disabled || this.segments[index]?.disabled;\n      this.indicatorStyle = {\n        width: `${width}px`,\n        height: `${height}px`,\n        left: `${offsetLeft}px`,\n        top: `${offsetTop}px`,\n        backgroundColor: isDisabled ? '#9BA0F4' : '#3F43EE',\n      };\n      this.indicatorVisible = true;\n    });\n  }\n\n  get isIndicatorVisible(): boolean {\n    return this.indicatorVisible;\n  }\n}\n","<div class=\"cqa-ui-root\" [style.display]=\"fullWidth ? 'block' : 'inline-block'\" [style.width]=\"fullWidth ? '100%' : null\">\n  <div\n    #segmentContainer\n    class=\"cqa-relative cqa-flex-row cqa-items-start cqa-bg-surface-light cqa-rounded-[8px]\"\n    [ngClass]=\"[\n      fullWidth ? 'cqa-flex' : 'cqa-inline-flex',\n      size === 'lg' ? 'cqa-h-[40px] cqa-p-[4px]' : 'cqa-h-[31.5px] cqa-p-[3.5px]'\n    ]\"\n    role=\"tablist\"\n    [attr.aria-disabled]=\"disabled || null\"\n    [ngStyle]=\"containerBgColor ? {'background-color': containerBgColor} : null\"\n  >\n    <div\n      class=\"cqa-absolute cqa-rounded-[8px] cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-pointer-events-none\"\n      [class.cqa-opacity-0]=\"!isIndicatorVisible\" [ngStyle]=\"indicatorStyle\" aria-hidden=\"true\"></div>\n\n    <button *ngFor=\"let segment of segments; index as index; trackBy: trackByValue\" #segmentButton type=\"button\"\n      role=\"tab\"\n      class=\"cqa-relative cqa-z-0 cqa-flex cqa-flex-col cqa-justify-center cqa-items-center cqa-rounded-[8px] cqa-transition-all cqa-duration-200 cqa-ease-in-out cqa-whitespace-nowrap cqa-text-center focus:cqa-outline-none focus-visible:cqa-outline-none focus-visible:cqa-ring-0 focus-visible:cqa-ring-offset-0\"\n      [ngClass]=\"{\n        'cqa-flex-1 cqa-min-w-0': fullWidth,\n        'cqa-flex-none': !fullWidth,\n        'cqa-text-white cqa-font-medium': isSelected(segment),\n        'cqa-text-muted': !isSelected(segment) && !(disabled || segment.disabled),\n        'cqa-cursor-not-allowed': disabled || segment.disabled,\n        'cqa-text-disabled': (disabled || segment.disabled) && !isSelected(segment),\n        'cqa-hover:cqa-text-black': !isSelected(segment) && !disabled && !segment.disabled,\n        'cqa-px-[16px] cqa-py-[6px] cqa-h-[32px]': size === 'lg',\n        'cqa-px-[14px] cqa-py-[3.5px] cqa-h-[25px]': size !== 'lg'\n      }\" [disabled]=\"disabled || segment.disabled\" [attr.aria-selected]=\"isSelected(segment)\"\n      [attr.tabindex]=\"!disabled && !segment.disabled ? (isSelected(segment) ? 0 : -1) : -1\"\n      (click)=\"select(segment, index)\" (keydown)=\"onKeyDown($event, index)\">\n      <span\n        class=\"cqa-flex cqa-gap-1 cqa-items-center cqa-justify-center cqa-h-[18px] cqa-font-['Inter'] cqa-font-normal cqa-text-[12px] cqa-leading-[12px] cqa-text-center cqa-align-middle\">\n        \n      <mat-icon *ngIf=\"segment?.icon\" class=\"!cqa-w-[12px] !cqa-h-[12px] !cqa-text-[12px]\" >\n        {{ segment?.icon }}\n      </mat-icon>\n\n      {{ segment.label }}\n\n      <span *ngIf=\"segment?.count != null\"\n        class=\"cqa-inline-flex cqa-items-center cqa-justify-center cqa-ml-1 cqa-px-[6px] cqa-rounded-full cqa-text-[10px] cqa-leading-[14px] cqa-font-semibold cqa-min-w-[16px]\"\n        [ngClass]=\"isSelected(segment) ? 'cqa-bg-white cqa-text-[#3F43EE]' : 'cqa-bg-[#E5E7EB] cqa-text-[#475569]'\">\n        {{ segment.count }}\n      </span>\n\n      <span *ngIf=\"segment?.tooltip\"\n        style=\"display: inline-flex; align-items: center; justify-content: center; margin-left: 4px; cursor: help;\"\n        [matTooltip]=\"segment.tooltip\"\n        matTooltipPosition=\"above\"\n        matTooltipShowDelay=\"0\"\n        (click)=\"$event.stopPropagation()\">\n        <mat-icon style=\"width: 14px; height: 14px; font-size: 14px; opacity: 0.8;\">info</mat-icon>\n      </span>\n    </span>\n  </button>\n</div>"]}