@cqa-lib/cqa-ui 0.0.1

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.
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGlhbG9nLm1vZGVscy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvZGlhbG9nL2RpYWxvZy5tb2RlbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFRlbXBsYXRlUmVmIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb21wb25lbnRUeXBlIH0gZnJvbSAnQGFuZ3VsYXIvY2RrL3BvcnRhbCc7XG5pbXBvcnQgeyBEaWFsb2dSZWYgfSBmcm9tICcuL2RpYWxvZy1yZWYnO1xuXG5leHBvcnQgdHlwZSBEaWFsb2dCdXR0b25Sb2xlID0gJ3ByaW1hcnknIHwgJ3NlY29uZGFyeScgfCAnd2FybicgfCAoc3RyaW5nICYge30pO1xuXG5leHBvcnQgaW50ZXJmYWNlIERpYWxvZ0J1dHRvbkNvbmZpZzxUUmVzdWx0ID0gdW5rbm93bj4ge1xuICBsYWJlbDogc3RyaW5nO1xuICByb2xlPzogRGlhbG9nQnV0dG9uUm9sZTtcbiAgaGFuZGxlcj86IChkaWFsb2dSZWY6IERpYWxvZ1JlZjxUUmVzdWx0PikgPT4gVFJlc3VsdCB8IHZvaWQgfCBQcm9taXNlPFRSZXN1bHQgfCB2b2lkPjtcbiAgY2xvc2VPbkNsaWNrPzogYm9vbGVhbjtcbn1cblxuZXhwb3J0IHR5cGUgRGlhbG9nQnV0dG9uQWxpZ25tZW50ID0gJ2xlZnQnIHwgJ3JpZ2h0JyB8ICdjZW50ZXInO1xuXG5leHBvcnQgaW50ZXJmYWNlIFRlbXBsYXRlRGlhbG9nQ29udGVudDxUQ29udGV4dCA9IGFueT4ge1xuICB0eXBlOiAndGVtcGxhdGUnO1xuICB0ZW1wbGF0ZTogVGVtcGxhdGVSZWY8VENvbnRleHQ+O1xuICBjb250ZXh0PzogVENvbnRleHQ7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29tcG9uZW50RGlhbG9nQ29udGVudDxUSW5wdXRzID0gUmVjb3JkPHN0cmluZywgYW55Pj4ge1xuICB0eXBlOiAnY29tcG9uZW50JztcbiAgY29tcG9uZW50OiBDb21wb25lbnRUeXBlPHVua25vd24+O1xuICBpbnB1dHM/OiBUSW5wdXRzO1xufVxuXG5leHBvcnQgdHlwZSBEaWFsb2dDb250ZW50Q29uZmlnID1cbiAgfCBUZW1wbGF0ZURpYWxvZ0NvbnRlbnRcbiAgfCBDb21wb25lbnREaWFsb2dDb250ZW50O1xuXG5leHBvcnQgaW50ZXJmYWNlIERpYWxvZ0NvbmZpZzxUQ29udGVudCBleHRlbmRzIERpYWxvZ0NvbnRlbnRDb25maWcgPSBEaWFsb2dDb250ZW50Q29uZmlnLCBUUmVzdWx0ID0gdW5rbm93bj4ge1xuICB0aXRsZTogc3RyaW5nO1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgd2FybmluZz86IHN0cmluZztcbiAgY29udGVudD86IFRDb250ZW50O1xuICBkYXRhPzogdW5rbm93bjtcbiAgYnV0dG9uczogRGlhbG9nQnV0dG9uQ29uZmlnPFRSZXN1bHQ+W107XG4gIGJ1dHRvbkFsaWdubWVudD86IERpYWxvZ0J1dHRvbkFsaWdubWVudDtcbiAgcGFuZWxDbGFzcz86IHN0cmluZyB8IHN0cmluZ1tdO1xuICB3aWR0aD86IHN0cmluZztcbiAgbWF4V2lkdGg/OiBzdHJpbmc7XG4gIGRpc2FibGVDbG9zZT86IGJvb2xlYW47XG59XG5cblxuIl19
@@ -0,0 +1,101 @@
1
+ import { Injectable, Injector } from '@angular/core';
2
+ import { OverlayConfig } from '@angular/cdk/overlay';
3
+ import { ComponentPortal } from '@angular/cdk/portal';
4
+ import { filter } from 'rxjs/operators';
5
+ import { DialogRef } from './dialog-ref';
6
+ import { DialogComponent } from './dialog.component';
7
+ import { DIALOG_DATA, DIALOG_REF } from './dialog.tokens';
8
+ import * as i0 from "@angular/core";
9
+ import * as i1 from "@angular/cdk/overlay";
10
+ export class DialogService {
11
+ constructor(overlay, injector) {
12
+ this.overlay = overlay;
13
+ this.injector = injector;
14
+ }
15
+ open(config) {
16
+ this.assertValidConfig(config);
17
+ const overlayRef = this.overlay.create(this.buildOverlayConfig(config));
18
+ const dialogRef = new DialogRef(overlayRef);
19
+ const injector = Injector.create({
20
+ parent: this.injector,
21
+ providers: [
22
+ { provide: DIALOG_REF, useValue: dialogRef },
23
+ { provide: DIALOG_DATA, useValue: config.data },
24
+ ],
25
+ });
26
+ const containerPortal = new ComponentPortal(DialogComponent, undefined, injector);
27
+ const containerRef = overlayRef.attach(containerPortal);
28
+ const containerInstance = containerRef.instance;
29
+ containerInstance.config = config;
30
+ containerInstance.dialogRef = dialogRef;
31
+ if (config.content?.type === 'template') {
32
+ containerInstance.attachTemplate(config.content.template, config.content.context ?? {
33
+ $implicit: config.data,
34
+ data: config.data,
35
+ });
36
+ }
37
+ if (config.content?.type === 'component') {
38
+ const componentPortal = new ComponentPortal(config.content.component, undefined, Injector.create({
39
+ parent: injector,
40
+ providers: [
41
+ { provide: DIALOG_REF, useValue: dialogRef },
42
+ { provide: DIALOG_DATA, useValue: config.data },
43
+ ],
44
+ }));
45
+ const componentRef = containerInstance.attachComponent(componentPortal);
46
+ if (componentRef && config.content.inputs) {
47
+ Object.entries(config.content.inputs).forEach(([key, value]) => {
48
+ componentRef.instance[key] = value;
49
+ });
50
+ componentRef.changeDetectorRef.markForCheck();
51
+ }
52
+ if (componentRef) {
53
+ dialogRef.setComponentInstance(componentRef.instance);
54
+ }
55
+ }
56
+ containerRef.changeDetectorRef.markForCheck();
57
+ if (!config.disableClose) {
58
+ overlayRef.backdropClick().subscribe(() => dialogRef.close());
59
+ overlayRef
60
+ .keydownEvents()
61
+ .pipe(filter((event) => {
62
+ return event.key === 'Escape' || event.key === 'Esc';
63
+ }))
64
+ .subscribe(() => dialogRef.close());
65
+ }
66
+ return dialogRef;
67
+ }
68
+ assertValidConfig(config) {
69
+ if (!config.title) {
70
+ throw new Error('Dialog title is required.');
71
+ }
72
+ if (!config.buttons || config.buttons.length < 2) {
73
+ throw new Error('Dialog requires at least two buttons to be provided.');
74
+ }
75
+ }
76
+ buildOverlayConfig(config) {
77
+ const panelClass = Array.isArray(config.panelClass)
78
+ ? config.panelClass
79
+ : config.panelClass
80
+ ? [config.panelClass]
81
+ : [];
82
+ return new OverlayConfig({
83
+ hasBackdrop: true,
84
+ backdropClass: ['cdk-overlay-dark-backdrop', 'cqa-dialog-backdrop'],
85
+ scrollStrategy: this.overlay.scrollStrategies.block(),
86
+ positionStrategy: this.overlay.position().global().centerHorizontally().centerVertically(),
87
+ width: config.width,
88
+ maxWidth: config.maxWidth ?? '90vw',
89
+ panelClass: ['cqa-dialog-panel', ...panelClass],
90
+ });
91
+ }
92
+ }
93
+ DialogService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DialogService, deps: [{ token: i1.Overlay }, { token: i0.Injector }], target: i0.ɵɵFactoryTarget.Injectable });
94
+ DialogService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DialogService, providedIn: 'root' });
95
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DialogService, decorators: [{
96
+ type: Injectable,
97
+ args: [{
98
+ providedIn: 'root',
99
+ }]
100
+ }], ctorParameters: function () { return [{ type: i1.Overlay }, { type: i0.Injector }]; } });
101
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dialog.service.js","sourceRoot":"","sources":["../../../../../src/lib/dialog/dialog.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Injector } from '@angular/core';\nimport { Overlay, OverlayConfig } from '@angular/cdk/overlay';\nimport { ComponentPortal } from '@angular/cdk/portal';\nimport { filter } from 'rxjs/operators';\nimport { DialogConfig, DialogContentConfig } from './dialog.models';\nimport { DialogRef } from './dialog-ref';\nimport { DialogComponent } from './dialog.component';\nimport { DIALOG_DATA, DIALOG_REF } from './dialog.tokens';\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class DialogService {\n  constructor(private readonly overlay: Overlay, private readonly injector: Injector) {}\n\n  open<TContent extends DialogContentConfig = DialogContentConfig, TResult = unknown>(\n    config: DialogConfig<TContent, TResult>\n  ): DialogRef<TResult> {\n    this.assertValidConfig(config);\n\n    const overlayRef = this.overlay.create(this.buildOverlayConfig(config));\n    const dialogRef = new DialogRef<TResult>(overlayRef);\n\n    const injector = Injector.create({\n      parent: this.injector,\n      providers: [\n        { provide: DIALOG_REF, useValue: dialogRef },\n        { provide: DIALOG_DATA, useValue: config.data },\n      ],\n    });\n\n    const containerPortal = new ComponentPortal(DialogComponent, undefined, injector);\n    const containerRef = overlayRef.attach(containerPortal);\n    const containerInstance = containerRef.instance as DialogComponent<TResult>;\n\n    containerInstance.config = config;\n    containerInstance.dialogRef = dialogRef;\n\n    if (config.content?.type === 'template') {\n      containerInstance.attachTemplate(\n        config.content.template,\n        config.content.context ?? {\n          $implicit: config.data,\n          data: config.data,\n        }\n      );\n    }\n\n    if (config.content?.type === 'component') {\n      const componentPortal = new ComponentPortal(\n        config.content.component,\n        undefined,\n        Injector.create({\n          parent: injector,\n          providers: [\n            { provide: DIALOG_REF, useValue: dialogRef },\n            { provide: DIALOG_DATA, useValue: config.data },\n          ],\n        })\n      );\n\n      const componentRef = containerInstance.attachComponent(componentPortal);\n\n      if (componentRef && config.content.inputs) {\n        Object.entries(config.content.inputs).forEach(([key, value]) => {\n          (componentRef.instance as Record<string, unknown>)[key] = value;\n        });\n        componentRef.changeDetectorRef.markForCheck();\n      }\n\n      if (componentRef) {\n        dialogRef.setComponentInstance(componentRef.instance);\n      }\n    }\n\n    containerRef.changeDetectorRef.markForCheck();\n\n    if (!config.disableClose) {\n      overlayRef.backdropClick().subscribe(() => dialogRef.close());\n      overlayRef\n        .keydownEvents()\n        .pipe(\n          filter((event): event is KeyboardEvent => {\n            return (event as KeyboardEvent).key === 'Escape' || (event as KeyboardEvent).key === 'Esc';\n          })\n        )\n        .subscribe(() => dialogRef.close());\n    }\n\n    return dialogRef;\n  }\n\n  private assertValidConfig<TContent extends DialogContentConfig, TResult>(\n    config: DialogConfig<TContent, TResult>\n  ): void {\n    if (!config.title) {\n      throw new Error('Dialog title is required.');\n    }\n\n    if (!config.buttons || config.buttons.length < 2) {\n      throw new Error('Dialog requires at least two buttons to be provided.');\n    }\n  }\n\n  private buildOverlayConfig<TContent extends DialogContentConfig, TResult>(\n    config: DialogConfig<TContent, TResult>\n  ): OverlayConfig {\n    const panelClass = Array.isArray(config.panelClass)\n      ? config.panelClass\n      : config.panelClass\n      ? [config.panelClass]\n      : [];\n\n    return new OverlayConfig({\n      hasBackdrop: true,\n      backdropClass: ['cdk-overlay-dark-backdrop', 'cqa-dialog-backdrop'],\n      scrollStrategy: this.overlay.scrollStrategies.block(),\n      positionStrategy: this.overlay.position().global().centerHorizontally().centerVertically(),\n      width: config.width,\n      maxWidth: config.maxWidth ?? '90vw',\n      panelClass: ['cqa-dialog-panel', ...panelClass],\n    });\n  }\n}\n\n\n"]}
@@ -0,0 +1,4 @@
1
+ import { InjectionToken } from '@angular/core';
2
+ export const DIALOG_REF = new InjectionToken('CQA_DIALOG_REF');
3
+ export const DIALOG_DATA = new InjectionToken('CQA_DIALOG_DATA');
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGlhbG9nLnRva2Vucy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvZGlhbG9nL2RpYWxvZy50b2tlbnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUcvQyxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQUcsSUFBSSxjQUFjLENBQWlCLGdCQUFnQixDQUFDLENBQUM7QUFDL0UsTUFBTSxDQUFDLE1BQU0sV0FBVyxHQUFHLElBQUksY0FBYyxDQUFNLGlCQUFpQixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3Rpb25Ub2tlbiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgRGlhbG9nUmVmIH0gZnJvbSAnLi9kaWFsb2ctcmVmJztcblxuZXhwb3J0IGNvbnN0IERJQUxPR19SRUYgPSBuZXcgSW5qZWN0aW9uVG9rZW48RGlhbG9nUmVmPGFueT4+KCdDUUFfRElBTE9HX1JFRicpO1xuZXhwb3J0IGNvbnN0IERJQUxPR19EQVRBID0gbmV3IEluamVjdGlvblRva2VuPGFueT4oJ0NRQV9ESUFMT0dfREFUQScpO1xuXG5cbiJdfQ==
@@ -0,0 +1,34 @@
1
+ import { Component, Input } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/common";
4
+ export class RootWrapperComponent {
5
+ constructor() {
6
+ this.display = 'inline-block';
7
+ this.fullWidth = false;
8
+ }
9
+ get rootStyles() {
10
+ const styles = {
11
+ display: this.fullWidth ? 'block' : this.display
12
+ };
13
+ if (this.fullWidth) {
14
+ styles['width'] = '100%';
15
+ }
16
+ else if (this.width) {
17
+ styles['width'] = this.width;
18
+ }
19
+ return styles;
20
+ }
21
+ }
22
+ RootWrapperComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: RootWrapperComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
23
+ RootWrapperComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: RootWrapperComponent, selector: "cqa-ui-root", inputs: { display: "display", width: "width", fullWidth: "fullWidth" }, ngImport: i0, template: "<div id=\"cqa-ui-root\" [ngStyle]=\"rootStyles\">\n <ng-content></ng-content>\n</div>\n\n", directives: [{ type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] });
24
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: RootWrapperComponent, decorators: [{
25
+ type: Component,
26
+ args: [{ selector: 'cqa-ui-root', styles: [], template: "<div id=\"cqa-ui-root\" [ngStyle]=\"rootStyles\">\n <ng-content></ng-content>\n</div>\n\n" }]
27
+ }], propDecorators: { display: [{
28
+ type: Input
29
+ }], width: [{
30
+ type: Input
31
+ }], fullWidth: [{
32
+ type: Input
33
+ }] } });
34
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicm9vdC13cmFwcGVyLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvcm9vdC13cmFwcGVyL3Jvb3Qtd3JhcHBlci5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL3Jvb3Qtd3JhcHBlci9yb290LXdyYXBwZXIuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxlQUFlLENBQUM7OztBQU9qRCxNQUFNLE9BQU8sb0JBQW9CO0lBTGpDO1FBTVcsWUFBTyxHQUFpRSxjQUFjLENBQUM7UUFFdkYsY0FBUyxHQUFHLEtBQUssQ0FBQztLQWU1QjtJQWJDLElBQUksVUFBVTtRQUNaLE1BQU0sTUFBTSxHQUEyQjtZQUNyQyxPQUFPLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsT0FBTztTQUNqRCxDQUFDO1FBRUYsSUFBSSxJQUFJLENBQUMsU0FBUyxFQUFFO1lBQ2xCLE1BQU0sQ0FBQyxPQUFPLENBQUMsR0FBRyxNQUFNLENBQUM7U0FDMUI7YUFBTSxJQUFJLElBQUksQ0FBQyxLQUFLLEVBQUU7WUFDckIsTUFBTSxDQUFDLE9BQU8sQ0FBQyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUM7U0FDOUI7UUFFRCxPQUFPLE1BQU0sQ0FBQztJQUNoQixDQUFDOztpSEFqQlUsb0JBQW9CO3FHQUFwQixvQkFBb0IsMkhDUGpDLDRGQUlBOzJGREdhLG9CQUFvQjtrQkFMaEMsU0FBUzsrQkFDRSxhQUFhLFVBRWYsRUFBRTs4QkFHRCxPQUFPO3NCQUFmLEtBQUs7Z0JBQ0csS0FBSztzQkFBYixLQUFLO2dCQUNHLFNBQVM7c0JBQWpCLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2NxYS11aS1yb290JyxcbiAgdGVtcGxhdGVVcmw6ICcuL3Jvb3Qtd3JhcHBlci5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlczogW11cbn0pXG5leHBvcnQgY2xhc3MgUm9vdFdyYXBwZXJDb21wb25lbnQge1xuICBASW5wdXQoKSBkaXNwbGF5OiAnYmxvY2snIHwgJ2lubGluZS1ibG9jaycgfCAnaW5saW5lJyB8ICdpbmxpbmUtZmxleCcgfCAnZmxleCcgPSAnaW5saW5lLWJsb2NrJztcbiAgQElucHV0KCkgd2lkdGg/OiBzdHJpbmc7XG4gIEBJbnB1dCgpIGZ1bGxXaWR0aCA9IGZhbHNlO1xuXG4gIGdldCByb290U3R5bGVzKCk6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4ge1xuICAgIGNvbnN0IHN0eWxlczogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHtcbiAgICAgIGRpc3BsYXk6IHRoaXMuZnVsbFdpZHRoID8gJ2Jsb2NrJyA6IHRoaXMuZGlzcGxheVxuICAgIH07XG4gICAgXG4gICAgaWYgKHRoaXMuZnVsbFdpZHRoKSB7XG4gICAgICBzdHlsZXNbJ3dpZHRoJ10gPSAnMTAwJSc7XG4gICAgfSBlbHNlIGlmICh0aGlzLndpZHRoKSB7XG4gICAgICBzdHlsZXNbJ3dpZHRoJ10gPSB0aGlzLndpZHRoO1xuICAgIH1cbiAgICBcbiAgICByZXR1cm4gc3R5bGVzO1xuICB9XG59XG5cbiIsIjxkaXYgaWQ9XCJjcWEtdWktcm9vdFwiIFtuZ1N0eWxlXT1cInJvb3RTdHlsZXNcIj5cbiAgPG5nLWNvbnRlbnQ+PC9uZy1jb250ZW50PlxuPC9kaXY+XG5cbiJdfQ==
@@ -0,0 +1,111 @@
1
+ import { Component, EventEmitter, Input, Output, HostBinding } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/material/icon";
4
+ import * as i2 from "@angular/forms";
5
+ import * as i3 from "@angular/common";
6
+ export class SearchBarComponent {
7
+ constructor() {
8
+ this.id = 'cqa-ui-root';
9
+ /** Placeholder text for the input */
10
+ this.placeholder = 'Search';
11
+ /** Initial value or externally controlled value */
12
+ this.value = '';
13
+ /** Disable interactions */
14
+ this.disabled = false;
15
+ /** Whether the clear button should be visible when there is text */
16
+ this.showClear = true;
17
+ /** Accessible label for the input */
18
+ this.ariaLabel = 'Search';
19
+ /** Automatically focus the input when rendered */
20
+ this.autoFocus = false;
21
+ /** Search bar size */
22
+ this.size = 'md';
23
+ /** Stretch to fill container width */
24
+ this.fullWidth = false;
25
+ /** Emit on value changes (e.g. for two-way binding) */
26
+ this.valueChange = new EventEmitter();
27
+ /** Emit when user submits search (Enter key or form submit) */
28
+ this.search = new EventEmitter();
29
+ /** Emit when the value is cleared via the clear button */
30
+ this.cleared = new EventEmitter();
31
+ this.inputValue = '';
32
+ this.widthClasses = {
33
+ sm: 'w-[295px]',
34
+ md: 'w-[395px]',
35
+ lg: 'w-[495px]',
36
+ };
37
+ }
38
+ get displayStyle() {
39
+ return this.fullWidth ? 'block' : 'inline-block';
40
+ }
41
+ get widthStyle() {
42
+ return this.fullWidth ? '100%' : 'auto';
43
+ }
44
+ ngOnChanges(changes) {
45
+ if (changes['value'] && changes['value'].currentValue !== undefined) {
46
+ const newValue = changes['value'].currentValue ?? '';
47
+ if (newValue !== this.inputValue) {
48
+ this.inputValue = newValue;
49
+ }
50
+ }
51
+ }
52
+ onInput(event) {
53
+ const target = event.target;
54
+ const nextValue = target?.value ?? '';
55
+ this.inputValue = nextValue;
56
+ this.valueChange.emit(this.inputValue);
57
+ }
58
+ onSubmit(event) {
59
+ event.preventDefault();
60
+ if (this.disabled) {
61
+ return;
62
+ }
63
+ this.search.emit(this.inputValue.trim());
64
+ }
65
+ clear() {
66
+ if (this.disabled || this.inputValue === '') {
67
+ return;
68
+ }
69
+ this.inputValue = '';
70
+ this.valueChange.emit('');
71
+ this.cleared.emit();
72
+ }
73
+ }
74
+ SearchBarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SearchBarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
75
+ SearchBarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: SearchBarComponent, selector: "cqa-search-bar", inputs: { placeholder: "placeholder", value: "value", disabled: "disabled", showClear: "showClear", ariaLabel: "ariaLabel", autoFocus: "autoFocus", size: "size", fullWidth: "fullWidth" }, outputs: { valueChange: "valueChange", search: "search", cleared: "cleared" }, host: { properties: { "attr.id": "this.id", "style.display": "this.displayStyle", "style.width": "this.widthStyle" } }, usesOnChanges: true, ngImport: i0, template: "<form\n class=\"inline-flex items-center gap-2 px-6 py-3 text-[14px] border border-gray-200 rounded-md bg-white shadow-sm transition-colors\"\n [ngClass]=\"fullWidth ? 'w-full' : widthClasses[size]\"\n (submit)=\"onSubmit($event)\"\n>\n <span\n class=\"flex-none flex items-center justify-center text-gray-400 w-4 h-4\"\n [ngClass]=\"{ 'opacity-[0.38]': disabled }\"\n >\n <mat-icon\n class=\"flex items-center justify-center leading-none p-0\"\n [style.width.px]=\"16\"\n [style.height.px]=\"16\"\n [style.fontSize.px]=\"16\"\n >\n search\n </mat-icon>\n </span>\n\n <input\n type=\"text\"\n class=\"flex-1 min-w-[180px] border-none outline-none bg-transparent placeholder:text-gray-400 disabled:text-gray-400 disabled:cursor-not-allowed font-['SF_Pro_Text'] font-normal text-[12.3px] leading-none tracking-normal align-middle text-[#99999E]\"\n style=\"font-family: 'SF Pro Text', -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif; letter-spacing: 0;\"\n [placeholder]=\"placeholder\"\n [value]=\"inputValue\"\n (input)=\"onInput($event)\"\n [disabled]=\"disabled\"\n [attr.aria-label]=\"ariaLabel\"\n autocomplete=\"off\"\n autocapitalize=\"none\"\n spellcheck=\"false\"\n [attr.autofocus]=\"autoFocus ? '' : null\"\n />\n\n <button\n *ngIf=\"showClear && inputValue\"\n type=\"button\"\n class=\"flex items-center justify-center p-0 w-4 h-4 border-0 bg-transparent cursor-pointer text-gray-500 hover:text-gray-700 disabled:text-gray-300 transition-colors leading-none\"\n (click)=\"clear()\"\n [disabled]=\"disabled\"\n aria-label=\"Clear search\"\n >\n <mat-icon\n class=\"flex items-center justify-center leading-none p-0\"\n [style.width.px]=\"16\"\n [style.height.px]=\"16\"\n [style.fontSize.px]=\"16\"\n [ngClass]=\"{ 'opacity-[0.38]': disabled }\"\n >\n close\n </mat-icon>\n </button>\n</form>\n", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }], directives: [{ type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { type: i2.NgForm, selector: "form:not([ngNoForm]):not([formGroup]),ng-form,[ngForm]", inputs: ["ngFormOptions"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
76
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SearchBarComponent, decorators: [{
77
+ type: Component,
78
+ args: [{ selector: 'cqa-search-bar', template: "<form\n class=\"inline-flex items-center gap-2 px-6 py-3 text-[14px] border border-gray-200 rounded-md bg-white shadow-sm transition-colors\"\n [ngClass]=\"fullWidth ? 'w-full' : widthClasses[size]\"\n (submit)=\"onSubmit($event)\"\n>\n <span\n class=\"flex-none flex items-center justify-center text-gray-400 w-4 h-4\"\n [ngClass]=\"{ 'opacity-[0.38]': disabled }\"\n >\n <mat-icon\n class=\"flex items-center justify-center leading-none p-0\"\n [style.width.px]=\"16\"\n [style.height.px]=\"16\"\n [style.fontSize.px]=\"16\"\n >\n search\n </mat-icon>\n </span>\n\n <input\n type=\"text\"\n class=\"flex-1 min-w-[180px] border-none outline-none bg-transparent placeholder:text-gray-400 disabled:text-gray-400 disabled:cursor-not-allowed font-['SF_Pro_Text'] font-normal text-[12.3px] leading-none tracking-normal align-middle text-[#99999E]\"\n style=\"font-family: 'SF Pro Text', -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif; letter-spacing: 0;\"\n [placeholder]=\"placeholder\"\n [value]=\"inputValue\"\n (input)=\"onInput($event)\"\n [disabled]=\"disabled\"\n [attr.aria-label]=\"ariaLabel\"\n autocomplete=\"off\"\n autocapitalize=\"none\"\n spellcheck=\"false\"\n [attr.autofocus]=\"autoFocus ? '' : null\"\n />\n\n <button\n *ngIf=\"showClear && inputValue\"\n type=\"button\"\n class=\"flex items-center justify-center p-0 w-4 h-4 border-0 bg-transparent cursor-pointer text-gray-500 hover:text-gray-700 disabled:text-gray-300 transition-colors leading-none\"\n (click)=\"clear()\"\n [disabled]=\"disabled\"\n aria-label=\"Clear search\"\n >\n <mat-icon\n class=\"flex items-center justify-center leading-none p-0\"\n [style.width.px]=\"16\"\n [style.height.px]=\"16\"\n [style.fontSize.px]=\"16\"\n [ngClass]=\"{ 'opacity-[0.38]': disabled }\"\n >\n close\n </mat-icon>\n </button>\n</form>\n", styles: [] }]
79
+ }], propDecorators: { id: [{
80
+ type: HostBinding,
81
+ args: ['attr.id']
82
+ }], displayStyle: [{
83
+ type: HostBinding,
84
+ args: ['style.display']
85
+ }], widthStyle: [{
86
+ type: HostBinding,
87
+ args: ['style.width']
88
+ }], placeholder: [{
89
+ type: Input
90
+ }], value: [{
91
+ type: Input
92
+ }], disabled: [{
93
+ type: Input
94
+ }], showClear: [{
95
+ type: Input
96
+ }], ariaLabel: [{
97
+ type: Input
98
+ }], autoFocus: [{
99
+ type: Input
100
+ }], size: [{
101
+ type: Input
102
+ }], fullWidth: [{
103
+ type: Input
104
+ }], valueChange: [{
105
+ type: Output
106
+ }], search: [{
107
+ type: Output
108
+ }], cleared: [{
109
+ type: Output
110
+ }] } });
111
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"search-bar.component.js","sourceRoot":"","sources":["../../../../../src/lib/search-bar/search-bar.component.ts","../../../../../src/lib/search-bar/search-bar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnChanges, Output, SimpleChanges, HostBinding } from '@angular/core';\n\ntype SearchBarSize = 'sm' | 'md' | 'lg';\n\n@Component({\n  selector: 'cqa-search-bar',\n  templateUrl: './search-bar.component.html',\n  styleUrls: []\n})\nexport class SearchBarComponent implements OnChanges {\n  @HostBinding('attr.id') id = 'cqa-ui-root';\n  @HostBinding('style.display') get displayStyle() {\n    return this.fullWidth ? 'block' : 'inline-block';\n  }\n  @HostBinding('style.width') get widthStyle() {\n    return this.fullWidth ? '100%' : 'auto';\n  }\n\n  /** Placeholder text for the input */\n  @Input() placeholder = 'Search';\n\n  /** Initial value or externally controlled value */\n  @Input() value = '';\n\n  /** Disable interactions */\n  @Input() disabled = false;\n\n  /** Whether the clear button should be visible when there is text */\n  @Input() showClear = true;\n\n  /** Accessible label for the input */\n  @Input() ariaLabel = 'Search';\n\n  /** Automatically focus the input when rendered */\n  @Input() autoFocus = false;\n\n  /** Search bar size */\n  @Input() size: SearchBarSize = 'md';\n\n  /** Stretch to fill container width */\n  @Input() fullWidth = false;\n\n  /** Emit on value changes (e.g. for two-way binding) */\n  @Output() valueChange = new EventEmitter<string>();\n\n  /** Emit when user submits search (Enter key or form submit) */\n  @Output() search = new EventEmitter<string>();\n\n  /** Emit when the value is cleared via the clear button */\n  @Output() cleared = new EventEmitter<void>();\n\n  inputValue = '';\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['value'] && changes['value'].currentValue !== undefined) {\n      const newValue = changes['value'].currentValue ?? '';\n      if (newValue !== this.inputValue) {\n        this.inputValue = newValue;\n      }\n    }\n  }\n\n  readonly widthClasses: Record<SearchBarSize, string> = {\n    sm: 'w-[295px]',\n    md: 'w-[395px]',\n    lg: 'w-[495px]',\n  };\n\n  onInput(event: Event): void {\n    const target = event.target as HTMLInputElement | null;\n    const nextValue = target?.value ?? '';\n    this.inputValue = nextValue;\n    this.valueChange.emit(this.inputValue);\n  }\n\n  onSubmit(event: Event): void {\n    event.preventDefault();\n    if (this.disabled) {\n      return;\n    }\n\n    this.search.emit(this.inputValue.trim());\n  }\n\n  clear(): void {\n    if (this.disabled || this.inputValue === '') {\n      return;\n    }\n\n    this.inputValue = '';\n    this.valueChange.emit('');\n    this.cleared.emit();\n  }\n}\n","<form\n  class=\"inline-flex items-center gap-2 px-6 py-3 text-[14px] border border-gray-200 rounded-md bg-white shadow-sm transition-colors\"\n  [ngClass]=\"fullWidth ? 'w-full' : widthClasses[size]\"\n  (submit)=\"onSubmit($event)\"\n>\n    <span\n      class=\"flex-none flex items-center justify-center text-gray-400 w-4 h-4\"\n      [ngClass]=\"{ 'opacity-[0.38]': disabled }\"\n    >\n      <mat-icon\n        class=\"flex items-center justify-center leading-none p-0\"\n        [style.width.px]=\"16\"\n        [style.height.px]=\"16\"\n        [style.fontSize.px]=\"16\"\n      >\n        search\n      </mat-icon>\n    </span>\n\n    <input\n      type=\"text\"\n      class=\"flex-1 min-w-[180px] border-none outline-none bg-transparent placeholder:text-gray-400 disabled:text-gray-400 disabled:cursor-not-allowed font-['SF_Pro_Text'] font-normal text-[12.3px] leading-none tracking-normal align-middle text-[#99999E]\"\n      style=\"font-family: 'SF Pro Text', -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif; letter-spacing: 0;\"\n      [placeholder]=\"placeholder\"\n      [value]=\"inputValue\"\n      (input)=\"onInput($event)\"\n      [disabled]=\"disabled\"\n      [attr.aria-label]=\"ariaLabel\"\n      autocomplete=\"off\"\n      autocapitalize=\"none\"\n      spellcheck=\"false\"\n      [attr.autofocus]=\"autoFocus ? '' : null\"\n    />\n\n    <button\n      *ngIf=\"showClear && inputValue\"\n      type=\"button\"\n      class=\"flex items-center justify-center p-0 w-4 h-4 border-0 bg-transparent cursor-pointer text-gray-500 hover:text-gray-700 disabled:text-gray-300 transition-colors leading-none\"\n      (click)=\"clear()\"\n      [disabled]=\"disabled\"\n      aria-label=\"Clear search\"\n    >\n      <mat-icon\n        class=\"flex items-center justify-center leading-none p-0\"\n        [style.width.px]=\"16\"\n        [style.height.px]=\"16\"\n        [style.fontSize.px]=\"16\"\n        [ngClass]=\"{ 'opacity-[0.38]': disabled }\"\n      >\n        close\n      </mat-icon>\n    </button>\n</form>\n"]}
@@ -0,0 +1,219 @@
1
+ import { Component, EventEmitter, Input, Output, ViewChild, ViewChildren, HostBinding, } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/common";
4
+ export class SegmentControlComponent {
5
+ constructor() {
6
+ this.id = 'cqa-ui-root';
7
+ this.display = 'inline-block';
8
+ this.segments = [
9
+ { label: 'Tab Group', value: 'tab-group-1' },
10
+ { label: 'Tab Group', value: 'tab-group-2' },
11
+ ];
12
+ this.disabled = false;
13
+ this.valueChange = new EventEmitter();
14
+ this.indicatorStyle = {};
15
+ this.indicatorVisible = false;
16
+ }
17
+ ngOnChanges(changes) {
18
+ if (changes['segments'] || changes['value']) {
19
+ this.ensureSelectedValue();
20
+ }
21
+ }
22
+ ngAfterViewInit() {
23
+ this.buttonChangesSub = this.segmentButtons.changes.subscribe(() => this.updateIndicator());
24
+ this.ensureSelectedValue();
25
+ this.updateIndicator();
26
+ }
27
+ ngOnDestroy() {
28
+ this.buttonChangesSub?.unsubscribe?.();
29
+ }
30
+ trackByValue(_index, option) {
31
+ return option.value;
32
+ }
33
+ isSelected(option) {
34
+ return option.value === this.value;
35
+ }
36
+ select(option, index) {
37
+ if (this.disabled || option.disabled) {
38
+ return;
39
+ }
40
+ const nextValue = option.value;
41
+ if (nextValue !== this.value) {
42
+ this.value = nextValue;
43
+ this.valueChange.emit(nextValue);
44
+ }
45
+ this.focusButton(index);
46
+ this.updateIndicator();
47
+ }
48
+ onKeyDown(event, currentIndex) {
49
+ if (this.disabled) {
50
+ return;
51
+ }
52
+ switch (event.key) {
53
+ case 'ArrowRight':
54
+ case 'ArrowDown':
55
+ event.preventDefault();
56
+ this.moveSelection(1, currentIndex);
57
+ break;
58
+ case 'ArrowLeft':
59
+ case 'ArrowUp':
60
+ event.preventDefault();
61
+ this.moveSelection(-1, currentIndex);
62
+ break;
63
+ case 'Home':
64
+ event.preventDefault();
65
+ this.selectFirstEnabled();
66
+ break;
67
+ case 'End':
68
+ event.preventDefault();
69
+ this.selectLastEnabled();
70
+ break;
71
+ case ' ':
72
+ case 'Enter':
73
+ event.preventDefault();
74
+ this.select(this.segments[currentIndex], currentIndex);
75
+ break;
76
+ default:
77
+ break;
78
+ }
79
+ }
80
+ moveSelection(step, startIndex) {
81
+ const enabledIndexes = this.getEnabledIndexes();
82
+ if (enabledIndexes.length === 0) {
83
+ return;
84
+ }
85
+ const currentEnabledIndex = enabledIndexes.indexOf(startIndex);
86
+ const fallbackIndex = this.getSelectedIndex(enabledIndexes);
87
+ const baseIndex = currentEnabledIndex >= 0 ? currentEnabledIndex : fallbackIndex;
88
+ const nextPosition = (baseIndex + step + enabledIndexes.length) % enabledIndexes.length;
89
+ const targetIndex = enabledIndexes[nextPosition];
90
+ this.select(this.segments[targetIndex], targetIndex);
91
+ }
92
+ selectFirstEnabled() {
93
+ const enabledIndexes = this.getEnabledIndexes();
94
+ if (enabledIndexes.length > 0) {
95
+ const index = enabledIndexes[0];
96
+ this.select(this.segments[index], index);
97
+ this.updateIndicator();
98
+ }
99
+ }
100
+ selectLastEnabled() {
101
+ const enabledIndexes = this.getEnabledIndexes();
102
+ if (enabledIndexes.length > 0) {
103
+ const index = enabledIndexes[enabledIndexes.length - 1];
104
+ this.select(this.segments[index], index);
105
+ this.updateIndicator();
106
+ }
107
+ }
108
+ getEnabledIndexes() {
109
+ return this.segments
110
+ .map((option, index) => ({ option, index }))
111
+ .filter(({ option }) => !option.disabled)
112
+ .map(({ index }) => index);
113
+ }
114
+ getSelectedIndex(enabledIndexes) {
115
+ const current = this.segments.findIndex((option) => option.value === this.value && !option.disabled);
116
+ if (current >= 0) {
117
+ return enabledIndexes.indexOf(current);
118
+ }
119
+ return 0;
120
+ }
121
+ ensureSelectedValue() {
122
+ const enabled = this.segments.filter((option) => !option.disabled);
123
+ if (enabled.length === 0) {
124
+ this.value = undefined;
125
+ return;
126
+ }
127
+ if (!this.value || !this.segments.some((option) => option.value === this.value)) {
128
+ this.value = enabled[0].value;
129
+ this.valueChange.emit(this.value);
130
+ const index = this.segments.indexOf(enabled[0]);
131
+ this.focusButton(index);
132
+ this.updateIndicator(index);
133
+ return;
134
+ }
135
+ const selected = this.segments.find((option) => option.value === this.value);
136
+ if (selected?.disabled) {
137
+ this.value = enabled[0].value;
138
+ this.valueChange.emit(this.value);
139
+ const index = this.segments.indexOf(enabled[0]);
140
+ this.focusButton(index);
141
+ this.updateIndicator(index);
142
+ }
143
+ this.updateIndicator();
144
+ }
145
+ focusButton(index) {
146
+ queueMicrotask(() => {
147
+ const button = this.segmentButtons?.get(index)?.nativeElement;
148
+ button?.focus();
149
+ });
150
+ }
151
+ updateIndicator(preferredIndex) {
152
+ queueMicrotask(() => {
153
+ const container = this.segmentContainer?.nativeElement;
154
+ const buttons = this.segmentButtons?.toArray() ?? [];
155
+ if (!container || buttons.length === 0) {
156
+ this.indicatorVisible = false;
157
+ this.indicatorStyle = {};
158
+ return;
159
+ }
160
+ const index = preferredIndex ?? buttons.findIndex((button, idx) => this.segments[idx]?.value === this.value);
161
+ if (index === -1) {
162
+ this.indicatorVisible = false;
163
+ this.indicatorStyle = {};
164
+ return;
165
+ }
166
+ const buttonEl = buttons[index]?.nativeElement;
167
+ if (!buttonEl) {
168
+ this.indicatorVisible = false;
169
+ this.indicatorStyle = {};
170
+ return;
171
+ }
172
+ const containerRect = container.getBoundingClientRect();
173
+ const buttonRect = buttonEl.getBoundingClientRect();
174
+ const offsetLeft = buttonEl.offsetLeft;
175
+ const offsetTop = buttonEl.offsetTop;
176
+ const width = buttonEl.offsetWidth;
177
+ const height = buttonEl.offsetHeight;
178
+ const isDisabled = this.disabled || this.segments[index]?.disabled;
179
+ this.indicatorStyle = {
180
+ width: `${width}px`,
181
+ height: `${height}px`,
182
+ left: `${offsetLeft}px`,
183
+ top: `${offsetTop}px`,
184
+ backgroundColor: isDisabled ? '#9BA0F4' : '#3F43EE',
185
+ };
186
+ this.indicatorVisible = true;
187
+ });
188
+ }
189
+ get isIndicatorVisible() {
190
+ return this.indicatorVisible;
191
+ }
192
+ }
193
+ SegmentControlComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SegmentControlComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
194
+ 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" }, outputs: { valueChange: "valueChange" }, host: { properties: { "attr.id": "this.id", "style.display": "this.display" } }, viewQueries: [{ propertyName: "segmentContainer", first: true, predicate: ["segmentContainer"], descendants: true }, { propertyName: "segmentButtons", predicate: ["segmentButton"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n #segmentContainer\n class=\"relative inline-flex flex-row items-start p-[3.5px] h-[31.5px] bg-[#F5F5F5] rounded-[8px]\"\n role=\"tablist\"\n [attr.aria-disabled]=\"disabled || null\"\n>\n <div\n class=\"absolute rounded-[8px] transition-all duration-200 ease-in-out pointer-events-none\"\n [class.opacity-0]=\"!isIndicatorVisible\"\n [ngStyle]=\"indicatorStyle\"\n aria-hidden=\"true\"\n ></div>\n\n <button\n *ngFor=\"let segment of segments; index as index; trackBy: trackByValue\"\n #segmentButton\n type=\"button\"\n role=\"tab\"\n class=\"relative z-10 flex flex-col justify-center items-center px-[14px] py-[3.5px] h-[25px] rounded-[8px] transition-all duration-200 ease-in-out whitespace-nowrap text-center focus:outline-none focus-visible:outline-none focus-visible:ring-0 focus-visible:ring-offset-0 flex-none\"\n [ngClass]=\"{\n 'text-white font-medium': isSelected(segment),\n 'text-[#99999E]': !isSelected(segment) && !(disabled || segment.disabled),\n 'cursor-not-allowed': disabled || segment.disabled,\n 'text-[#C7C7C7]': (disabled || segment.disabled) && !isSelected(segment),\n 'hover:text-black': !isSelected(segment) && !disabled && !segment.disabled\n }\"\n [disabled]=\"disabled || segment.disabled\"\n [attr.aria-selected]=\"isSelected(segment)\"\n [attr.tabindex]=\"!disabled && !segment.disabled ? (isSelected(segment) ? 0 : -1) : -1\"\n (click)=\"select(segment, index)\"\n (keydown)=\"onKeyDown($event, index)\"\n >\n <span class=\"flex items-center justify-center h-[18px] font-['Inter'] font-normal text-[12px] leading-[12px] text-center align-middle\">\n {{ segment.label }}\n </span>\n </button>\n</div>\n", directives: [{ type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }] });
195
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SegmentControlComponent, decorators: [{
196
+ type: Component,
197
+ args: [{ selector: 'cqa-segment-control', template: "<div\n #segmentContainer\n class=\"relative inline-flex flex-row items-start p-[3.5px] h-[31.5px] bg-[#F5F5F5] rounded-[8px]\"\n role=\"tablist\"\n [attr.aria-disabled]=\"disabled || null\"\n>\n <div\n class=\"absolute rounded-[8px] transition-all duration-200 ease-in-out pointer-events-none\"\n [class.opacity-0]=\"!isIndicatorVisible\"\n [ngStyle]=\"indicatorStyle\"\n aria-hidden=\"true\"\n ></div>\n\n <button\n *ngFor=\"let segment of segments; index as index; trackBy: trackByValue\"\n #segmentButton\n type=\"button\"\n role=\"tab\"\n class=\"relative z-10 flex flex-col justify-center items-center px-[14px] py-[3.5px] h-[25px] rounded-[8px] transition-all duration-200 ease-in-out whitespace-nowrap text-center focus:outline-none focus-visible:outline-none focus-visible:ring-0 focus-visible:ring-offset-0 flex-none\"\n [ngClass]=\"{\n 'text-white font-medium': isSelected(segment),\n 'text-[#99999E]': !isSelected(segment) && !(disabled || segment.disabled),\n 'cursor-not-allowed': disabled || segment.disabled,\n 'text-[#C7C7C7]': (disabled || segment.disabled) && !isSelected(segment),\n 'hover:text-black': !isSelected(segment) && !disabled && !segment.disabled\n }\"\n [disabled]=\"disabled || segment.disabled\"\n [attr.aria-selected]=\"isSelected(segment)\"\n [attr.tabindex]=\"!disabled && !segment.disabled ? (isSelected(segment) ? 0 : -1) : -1\"\n (click)=\"select(segment, index)\"\n (keydown)=\"onKeyDown($event, index)\"\n >\n <span class=\"flex items-center justify-center h-[18px] font-['Inter'] font-normal text-[12px] leading-[12px] text-center align-middle\">\n {{ segment.label }}\n </span>\n </button>\n</div>\n", styles: [] }]
198
+ }], propDecorators: { id: [{
199
+ type: HostBinding,
200
+ args: ['attr.id']
201
+ }], display: [{
202
+ type: HostBinding,
203
+ args: ['style.display']
204
+ }], segments: [{
205
+ type: Input
206
+ }], value: [{
207
+ type: Input
208
+ }], disabled: [{
209
+ type: Input
210
+ }], valueChange: [{
211
+ type: Output
212
+ }], segmentButtons: [{
213
+ type: ViewChildren,
214
+ args: ['segmentButton']
215
+ }], segmentContainer: [{
216
+ type: ViewChild,
217
+ args: ['segmentContainer']
218
+ }] } });
219
+ //# 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,EAEL,SAAS,EAET,YAAY,EACZ,KAAK,EAGL,MAAM,EAGN,SAAS,EACT,YAAY,EACZ,WAAW,GACZ,MAAM,eAAe,CAAC;;;AAcvB,MAAM,OAAO,uBAAuB;IALpC;QAM0B,OAAE,GAAG,aAAa,CAAC;QACb,YAAO,GAAG,cAAc,CAAC;QAE9C,aAAQ,GAAoB;YACnC,EAAE,KAAK,EAAE,WAAW,EAAE,KAAK,EAAE,aAAa,EAAE;YAC5C,EAAE,KAAK,EAAE,WAAW,EAAE,KAAK,EAAE,aAAa,EAAE;SAC7C,CAAC;QAGO,aAAQ,GAAG,KAAK,CAAC;QAEhB,gBAAW,GAAG,IAAI,YAAY,EAAU,CAAC;QAKnD,mBAAc,GAA2B,EAAE,CAAC;QAC5C,qBAAgB,GAAG,KAAK,CAAC;KA6M1B;IA1MC,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;IAEO,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,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,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;YAC9B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAClC,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;YAChD,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;YACxB,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;YAC5B,OAAO;SACR;QAED,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,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;YAC9B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAClC,MAAM,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;YAChD,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;YACxB,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;SAC7B;QACD,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,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,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,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;QAC/B,CAAC,CAAC,CAAC;IACL,CAAC;IAED,IAAI,kBAAkB;QACpB,OAAO,IAAI,CAAC,gBAAgB,CAAC;IAC/B,CAAC;;oHA9NU,uBAAuB;wGAAvB,uBAAuB,6dC5BpC,+vDAqCA;2FDTa,uBAAuB;kBALnC,SAAS;+BACE,qBAAqB;8BAKP,EAAE;sBAAzB,WAAW;uBAAC,SAAS;gBACQ,OAAO;sBAApC,WAAW;uBAAC,eAAe;gBAEnB,QAAQ;sBAAhB,KAAK;gBAKG,KAAK;sBAAb,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBAEI,WAAW;sBAApB,MAAM;gBAEwB,cAAc;sBAA5C,YAAY;uBAAC,eAAe;gBACE,gBAAgB;sBAA9C,SAAS;uBAAC,kBAAkB","sourcesContent":["import {\n  AfterViewInit,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnDestroy,\n  OnChanges,\n  Output,\n  QueryList,\n  SimpleChanges,\n  ViewChild,\n  ViewChildren,\n  HostBinding,\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\n\ntype SegmentOption = {\n  label: string;\n  value: string;\n  disabled?: boolean;\n};\n\n@Component({\n  selector: 'cqa-segment-control',\n  templateUrl: './segment-control.component.html',\n  styleUrls: [],\n})\nexport class SegmentControlComponent implements OnChanges, AfterViewInit, OnDestroy {\n  @HostBinding('attr.id') id = 'cqa-ui-root';\n  @HostBinding('style.display') display = 'inline-block';\n\n  @Input() segments: SegmentOption[] = [\n    { label: 'Tab Group', value: 'tab-group-1' },\n    { label: 'Tab Group', value: 'tab-group-2' },\n  ];\n\n  @Input() value?: string;\n  @Input() disabled = false;\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  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 (!this.value || !this.segments.some((option) => option.value === this.value)) {\n      this.value = enabled[0].value;\n      this.valueChange.emit(this.value);\n      const index = this.segments.indexOf(enabled[0]);\n      this.focusButton(index);\n      this.updateIndicator(index);\n      return;\n    }\n\n    const selected = this.segments.find((option) => option.value === this.value);\n    if (selected?.disabled) {\n      this.value = enabled[0].value;\n      this.valueChange.emit(this.value);\n      const index = this.segments.indexOf(enabled[0]);\n      this.focusButton(index);\n      this.updateIndicator(index);\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\n  #segmentContainer\n  class=\"relative inline-flex flex-row items-start p-[3.5px] h-[31.5px] bg-[#F5F5F5] rounded-[8px]\"\n  role=\"tablist\"\n  [attr.aria-disabled]=\"disabled || null\"\n>\n    <div\n      class=\"absolute rounded-[8px] transition-all duration-200 ease-in-out pointer-events-none\"\n      [class.opacity-0]=\"!isIndicatorVisible\"\n      [ngStyle]=\"indicatorStyle\"\n      aria-hidden=\"true\"\n    ></div>\n\n    <button\n      *ngFor=\"let segment of segments; index as index; trackBy: trackByValue\"\n      #segmentButton\n      type=\"button\"\n      role=\"tab\"\n      class=\"relative z-10 flex flex-col justify-center items-center px-[14px] py-[3.5px] h-[25px] rounded-[8px] transition-all duration-200 ease-in-out whitespace-nowrap text-center focus:outline-none focus-visible:outline-none focus-visible:ring-0 focus-visible:ring-offset-0 flex-none\"\n      [ngClass]=\"{\n        'text-white font-medium': isSelected(segment),\n        'text-[#99999E]': !isSelected(segment) && !(disabled || segment.disabled),\n        'cursor-not-allowed': disabled || segment.disabled,\n        'text-[#C7C7C7]': (disabled || segment.disabled) && !isSelected(segment),\n        'hover:text-black': !isSelected(segment) && !disabled && !segment.disabled\n      }\"\n      [disabled]=\"disabled || segment.disabled\"\n      [attr.aria-selected]=\"isSelected(segment)\"\n      [attr.tabindex]=\"!disabled && !segment.disabled ? (isSelected(segment) ? 0 : -1) : -1\"\n      (click)=\"select(segment, index)\"\n      (keydown)=\"onKeyDown($event, index)\"\n    >\n      <span class=\"flex items-center justify-center h-[18px] font-['Inter'] font-normal text-[12px] leading-[12px] text-center align-middle\">\n      {{ segment.label }}\n    </span>\n    </button>\n</div>\n"]}