@cqa-lib/cqa-ui 0.0.1 → 0.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +90 -4
- package/dist/cqa-ui/README.md +226 -0
- package/dist/cqa-ui/esm2020/lib/button/button.component.mjs +257 -0
- package/dist/cqa-ui/esm2020/lib/dialog/dialog.component.mjs +127 -0
- package/dist/cqa-ui/esm2020/lib/search-bar/search-bar.component.mjs +95 -0
- package/dist/cqa-ui/esm2020/lib/segment-control/segment-control.component.mjs +211 -0
- package/{fesm2015 → dist/cqa-ui/fesm2015}/cqa-lib-cqa-ui.mjs +13 -53
- package/dist/cqa-ui/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -0
- package/{fesm2020 → dist/cqa-ui/fesm2020}/cqa-lib-cqa-ui.mjs +13 -53
- package/dist/cqa-ui/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -0
- package/{lib → dist/cqa-ui/lib}/button/button.component.d.ts +0 -2
- package/{lib → dist/cqa-ui/lib}/dialog/dialog.component.d.ts +0 -2
- package/{lib → dist/cqa-ui/lib}/search-bar/search-bar.component.d.ts +0 -3
- package/{lib → dist/cqa-ui/lib}/segment-control/segment-control.component.d.ts +0 -2
- package/dist/cqa-ui/package.json +56 -0
- package/dist/cqa-ui/styles.css +1 -0
- package/package.json +49 -23
- package/esm2020/lib/button/button.component.mjs +0 -265
- package/esm2020/lib/dialog/dialog.component.mjs +0 -135
- package/esm2020/lib/search-bar/search-bar.component.mjs +0 -111
- package/esm2020/lib/segment-control/segment-control.component.mjs +0 -219
- package/fesm2015/cqa-lib-cqa-ui.mjs.map +0 -1
- package/fesm2020/cqa-lib-cqa-ui.mjs.map +0 -1
- package/styles.css +0 -1
- /package/{cqa-lib-cqa-ui.d.ts → dist/cqa-ui/cqa-lib-cqa-ui.d.ts} +0 -0
- /package/{esm2020 → dist/cqa-ui/esm2020}/cqa-lib-cqa-ui.mjs +0 -0
- /package/{esm2020 → dist/cqa-ui/esm2020}/lib/dialog/dialog-ref.mjs +0 -0
- /package/{esm2020 → dist/cqa-ui/esm2020}/lib/dialog/dialog.models.mjs +0 -0
- /package/{esm2020 → dist/cqa-ui/esm2020}/lib/dialog/dialog.service.mjs +0 -0
- /package/{esm2020 → dist/cqa-ui/esm2020}/lib/dialog/dialog.tokens.mjs +0 -0
- /package/{esm2020 → dist/cqa-ui/esm2020}/lib/root-wrapper/root-wrapper.component.mjs +0 -0
- /package/{esm2020 → dist/cqa-ui/esm2020}/lib/ui-kit.module.mjs +0 -0
- /package/{esm2020 → dist/cqa-ui/esm2020}/public-api.mjs +0 -0
- /package/{lib → dist/cqa-ui/lib}/dialog/dialog-ref.d.ts +0 -0
- /package/{lib → dist/cqa-ui/lib}/dialog/dialog.models.d.ts +0 -0
- /package/{lib → dist/cqa-ui/lib}/dialog/dialog.service.d.ts +0 -0
- /package/{lib → dist/cqa-ui/lib}/dialog/dialog.tokens.d.ts +0 -0
- /package/{lib → dist/cqa-ui/lib}/root-wrapper/root-wrapper.component.d.ts +0 -0
- /package/{lib → dist/cqa-ui/lib}/ui-kit.module.d.ts +0 -0
- /package/{public-api.d.ts → dist/cqa-ui/public-api.d.ts} +0 -0
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, ViewChild, } from '@angular/core';
|
|
2
|
+
import { CdkPortalOutlet, TemplatePortal } from '@angular/cdk/portal';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "../button/button.component";
|
|
5
|
+
import * as i2 from "@angular/common";
|
|
6
|
+
import * as i3 from "@angular/cdk/portal";
|
|
7
|
+
export class DialogComponent {
|
|
8
|
+
constructor(viewContainerRef, cdr) {
|
|
9
|
+
this.viewContainerRef = viewContainerRef;
|
|
10
|
+
this.cdr = cdr;
|
|
11
|
+
this.contentAttached = false;
|
|
12
|
+
}
|
|
13
|
+
attachTemplate(template, context) {
|
|
14
|
+
if (!this.portalOutlet) {
|
|
15
|
+
return;
|
|
16
|
+
}
|
|
17
|
+
const templateContext = context ??
|
|
18
|
+
{
|
|
19
|
+
$implicit: this.config?.data,
|
|
20
|
+
data: this.config?.data,
|
|
21
|
+
};
|
|
22
|
+
const portal = new TemplatePortal(template, this.viewContainerRef, templateContext);
|
|
23
|
+
this.portalOutlet.attachTemplatePortal(portal);
|
|
24
|
+
this.markContentAttached();
|
|
25
|
+
}
|
|
26
|
+
attachComponent(component) {
|
|
27
|
+
if (!this.portalOutlet) {
|
|
28
|
+
return undefined;
|
|
29
|
+
}
|
|
30
|
+
const componentRef = this.portalOutlet.attachComponentPortal(component);
|
|
31
|
+
this.markContentAttached();
|
|
32
|
+
return componentRef;
|
|
33
|
+
}
|
|
34
|
+
async onButtonClick(button) {
|
|
35
|
+
const closeOnClick = button.closeOnClick ?? true;
|
|
36
|
+
let handlerResult = undefined;
|
|
37
|
+
if (button.handler) {
|
|
38
|
+
handlerResult = button.handler(this.dialogRef);
|
|
39
|
+
}
|
|
40
|
+
const resolved = handlerResult instanceof Promise ? await handlerResult : handlerResult;
|
|
41
|
+
if (!closeOnClick || resolved === false) {
|
|
42
|
+
return;
|
|
43
|
+
}
|
|
44
|
+
this.dialogRef.close(resolved);
|
|
45
|
+
}
|
|
46
|
+
get buttonAlignmentClass() {
|
|
47
|
+
const alignment = this.config?.buttonAlignment ?? 'right';
|
|
48
|
+
return this.mapAlignmentToClass(alignment);
|
|
49
|
+
}
|
|
50
|
+
get panelClassList() {
|
|
51
|
+
const baseClasses = [
|
|
52
|
+
'relative',
|
|
53
|
+
'w-full',
|
|
54
|
+
'bg-white',
|
|
55
|
+
'rounded-2xl',
|
|
56
|
+
'shadow-md',
|
|
57
|
+
'border',
|
|
58
|
+
'border-[#E5E7EB]',
|
|
59
|
+
'p-6',
|
|
60
|
+
'text-left',
|
|
61
|
+
];
|
|
62
|
+
const custom = this.config?.panelClass;
|
|
63
|
+
if (!custom) {
|
|
64
|
+
return baseClasses;
|
|
65
|
+
}
|
|
66
|
+
return Array.isArray(custom) ? [...baseClasses, ...custom] : [...baseClasses, custom];
|
|
67
|
+
}
|
|
68
|
+
get panelStyles() {
|
|
69
|
+
return {
|
|
70
|
+
width: this.config?.width,
|
|
71
|
+
maxWidth: this.config?.maxWidth ?? '480px',
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
buttonVariant(button) {
|
|
75
|
+
const role = this.normalizeRole(button.role);
|
|
76
|
+
switch (role) {
|
|
77
|
+
case 'secondary':
|
|
78
|
+
return 'outlined';
|
|
79
|
+
case 'text':
|
|
80
|
+
return 'text';
|
|
81
|
+
case 'tonal':
|
|
82
|
+
return 'tonal';
|
|
83
|
+
case 'elevated':
|
|
84
|
+
return 'elevated';
|
|
85
|
+
case 'filled':
|
|
86
|
+
case 'primary':
|
|
87
|
+
case 'warn':
|
|
88
|
+
default:
|
|
89
|
+
return 'filled';
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
buttonHostClasses(button) {
|
|
93
|
+
const role = this.normalizeRole(button.role);
|
|
94
|
+
if (role === 'warn') {
|
|
95
|
+
return ['cqa-dialog-btn-warn'];
|
|
96
|
+
}
|
|
97
|
+
return [];
|
|
98
|
+
}
|
|
99
|
+
mapAlignmentToClass(alignment) {
|
|
100
|
+
switch (alignment) {
|
|
101
|
+
case 'left':
|
|
102
|
+
return 'justify-start';
|
|
103
|
+
case 'center':
|
|
104
|
+
return 'justify-center';
|
|
105
|
+
case 'right':
|
|
106
|
+
default:
|
|
107
|
+
return 'justify-end';
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
markContentAttached() {
|
|
111
|
+
this.contentAttached = true;
|
|
112
|
+
this.cdr.markForCheck();
|
|
113
|
+
}
|
|
114
|
+
normalizeRole(role) {
|
|
115
|
+
return (role ?? 'secondary').trim().split(/\s+/)[0];
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
DialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DialogComponent, deps: [{ token: i0.ViewContainerRef }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
119
|
+
DialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: DialogComponent, selector: "cqa-dialog", viewQueries: [{ propertyName: "portalOutlet", first: true, predicate: CdkPortalOutlet, descendants: true, static: true }], ngImport: i0, template: "<div id=\"cqa-ui-root\" style=\"display: block;\">\n <div class=\"flex w-full justify-center px-4 sm:px-6\">\n <div [ngClass]=\"panelClassList\" [ngStyle]=\"panelStyles\">\n <div class=\"flex flex-col gap-5\">\n <div class=\"flex flex-col gap-3\">\n <h2 class=\"text-lg font-semibold text-[#111827]\">\n {{ config.title }}\n </h2>\n\n <p *ngIf=\"config.description\" class=\"text-sm leading-6 text-[#4B5563]\">\n {{ config.description }}\n </p>\n\n <div\n *ngIf=\"config.warning\"\n class=\"rounded-xl border border-red-200 bg-red-50 px-4 py-3 text-sm leading-5 text-red-700\"\n >\n {{ config.warning }}\n </div>\n </div>\n\n <div class=\"text-sm text-[#111827]\" [class.hidden]=\"!contentAttached\">\n <ng-template cdkPortalOutlet></ng-template>\n </div>\n\n <div class=\"mt-4 flex flex-wrap gap-3\" [ngClass]=\"buttonAlignmentClass\">\n <cqa-button\n *ngFor=\"let button of config.buttons\"\n type=\"button\"\n [variant]=\"buttonVariant(button)\"\n [ngClass]=\"buttonHostClasses(button)\"\n (clicked)=\"onButtonClick(button)\"\n >\n {{ button.label }}\n </cqa-button>\n </div>\n </div>\n </div>\n </div>\n</div>\n\n\n", components: [{ type: i1.ButtonComponent, selector: "cqa-button", inputs: ["variant", "disabled", "icon", "iconPosition", "type"], outputs: ["clicked"] }], directives: [{ type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.CdkPortalOutlet, selector: "[cdkPortalOutlet]", inputs: ["cdkPortalOutlet"], outputs: ["attached"], exportAs: ["cdkPortalOutlet"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
120
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: DialogComponent, decorators: [{
|
|
121
|
+
type: Component,
|
|
122
|
+
args: [{ selector: 'cqa-dialog', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div id=\"cqa-ui-root\" style=\"display: block;\">\n <div class=\"flex w-full justify-center px-4 sm:px-6\">\n <div [ngClass]=\"panelClassList\" [ngStyle]=\"panelStyles\">\n <div class=\"flex flex-col gap-5\">\n <div class=\"flex flex-col gap-3\">\n <h2 class=\"text-lg font-semibold text-[#111827]\">\n {{ config.title }}\n </h2>\n\n <p *ngIf=\"config.description\" class=\"text-sm leading-6 text-[#4B5563]\">\n {{ config.description }}\n </p>\n\n <div\n *ngIf=\"config.warning\"\n class=\"rounded-xl border border-red-200 bg-red-50 px-4 py-3 text-sm leading-5 text-red-700\"\n >\n {{ config.warning }}\n </div>\n </div>\n\n <div class=\"text-sm text-[#111827]\" [class.hidden]=\"!contentAttached\">\n <ng-template cdkPortalOutlet></ng-template>\n </div>\n\n <div class=\"mt-4 flex flex-wrap gap-3\" [ngClass]=\"buttonAlignmentClass\">\n <cqa-button\n *ngFor=\"let button of config.buttons\"\n type=\"button\"\n [variant]=\"buttonVariant(button)\"\n [ngClass]=\"buttonHostClasses(button)\"\n (clicked)=\"onButtonClick(button)\"\n >\n {{ button.label }}\n </cqa-button>\n </div>\n </div>\n </div>\n </div>\n</div>\n\n\n", styles: [] }]
|
|
123
|
+
}], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { portalOutlet: [{
|
|
124
|
+
type: ViewChild,
|
|
125
|
+
args: [CdkPortalOutlet, { static: true }]
|
|
126
|
+
}] } });
|
|
127
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dialog.component.js","sourceRoot":"","sources":["../../../../../src/lib/dialog/dialog.component.ts","../../../../../src/lib/dialog/dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ComponentRef,\n  TemplateRef,\n  ViewChild,\n  ViewContainerRef,\n} from '@angular/core';\nimport { CdkPortalOutlet, ComponentPortal, TemplatePortal } from '@angular/cdk/portal';\nimport { DialogButtonAlignment, DialogButtonConfig, DialogConfig, DialogContentConfig } from './dialog.models';\nimport { ButtonVariant } from '../button/button.component';\nimport { DialogRef } from './dialog-ref';\n\n@Component({\n  selector: 'cqa-dialog',\n  templateUrl: './dialog.component.html',\n  styleUrls: [],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class DialogComponent<TResult = unknown> {\n\n  @ViewChild(CdkPortalOutlet, { static: true }) portalOutlet?: CdkPortalOutlet;\n\n  config!: DialogConfig<DialogContentConfig, TResult>;\n  dialogRef!: DialogRef<TResult>;\n\n  contentAttached = false;\n\n  constructor(private readonly viewContainerRef: ViewContainerRef, private readonly cdr: ChangeDetectorRef) {}\n\n  attachTemplate(template: TemplateRef<unknown>, context?: Record<string, unknown>): void {\n    if (!this.portalOutlet) {\n      return;\n    }\n\n    const templateContext =\n      context ??\n      {\n        $implicit: this.config?.data,\n        data: this.config?.data,\n      };\n    const portal = new TemplatePortal(template, this.viewContainerRef, templateContext);\n    this.portalOutlet.attachTemplatePortal(portal);\n    this.markContentAttached();\n  }\n\n  attachComponent(component: ComponentPortal<unknown>): ComponentRef<unknown> | undefined {\n    if (!this.portalOutlet) {\n      return undefined;\n    }\n\n    const componentRef = this.portalOutlet.attachComponentPortal(component);\n    this.markContentAttached();\n    return componentRef;\n  }\n\n  async onButtonClick(button: DialogButtonConfig<TResult>): Promise<void> {\n    const closeOnClick = button.closeOnClick ?? true;\n    let handlerResult: TResult | false | void | Promise<TResult | false | void> = undefined;\n\n    if (button.handler) {\n      handlerResult = button.handler(this.dialogRef);\n    }\n\n    const resolved = handlerResult instanceof Promise ? await handlerResult : handlerResult;\n\n    if (!closeOnClick || resolved === false) {\n      return;\n    }\n\n    this.dialogRef.close(resolved as TResult | undefined);\n  }\n\n  get buttonAlignmentClass(): string {\n    const alignment = this.config?.buttonAlignment ?? 'right';\n    return this.mapAlignmentToClass(alignment);\n  }\n\n  get panelClassList(): string[] {\n    const baseClasses = [\n      'relative',\n      'w-full',\n      'bg-white',\n      'rounded-2xl',\n      'shadow-md',\n      'border',\n      'border-[#E5E7EB]',\n      'p-6',\n      'text-left',\n    ];\n\n    const custom = this.config?.panelClass;\n\n    if (!custom) {\n      return baseClasses;\n    }\n\n    return Array.isArray(custom) ? [...baseClasses, ...custom] : [...baseClasses, custom];\n  }\n\n  get panelStyles(): Record<string, string | undefined> {\n    return {\n      width: this.config?.width,\n      maxWidth: this.config?.maxWidth ?? '480px',\n    };\n  }\n\n  buttonVariant(button: DialogButtonConfig<TResult>): ButtonVariant {\n    const role = this.normalizeRole(button.role);\n\n    switch (role) {\n      case 'secondary':\n        return 'outlined';\n      case 'text':\n        return 'text';\n      case 'tonal':\n        return 'tonal';\n      case 'elevated':\n        return 'elevated';\n      case 'filled':\n      case 'primary':\n      case 'warn':\n      default:\n        return 'filled';\n    }\n  }\n\n  buttonHostClasses(button: DialogButtonConfig<TResult>): string[] {\n    const role = this.normalizeRole(button.role);\n\n    if (role === 'warn') {\n      return ['cqa-dialog-btn-warn'];\n    }\n\n    return [];\n  }\n\n  private mapAlignmentToClass(alignment: DialogButtonAlignment): string {\n    switch (alignment) {\n      case 'left':\n        return 'justify-start';\n      case 'center':\n        return 'justify-center';\n      case 'right':\n      default:\n        return 'justify-end';\n    }\n  }\n\n  private markContentAttached(): void {\n    this.contentAttached = true;\n    this.cdr.markForCheck();\n  }\n\n  private normalizeRole(role: string | undefined): string {\n    return (role ?? 'secondary').trim().split(/\\s+/)[0];\n  }\n}\n\n\n","<div id=\"cqa-ui-root\" style=\"display: block;\">\n  <div class=\"flex w-full justify-center px-4 sm:px-6\">\n    <div [ngClass]=\"panelClassList\" [ngStyle]=\"panelStyles\">\n      <div class=\"flex flex-col gap-5\">\n        <div class=\"flex flex-col gap-3\">\n          <h2 class=\"text-lg font-semibold text-[#111827]\">\n            {{ config.title }}\n          </h2>\n\n          <p *ngIf=\"config.description\" class=\"text-sm leading-6 text-[#4B5563]\">\n            {{ config.description }}\n          </p>\n\n          <div\n            *ngIf=\"config.warning\"\n            class=\"rounded-xl border border-red-200 bg-red-50 px-4 py-3 text-sm leading-5 text-red-700\"\n          >\n            {{ config.warning }}\n          </div>\n        </div>\n\n        <div class=\"text-sm text-[#111827]\" [class.hidden]=\"!contentAttached\">\n          <ng-template cdkPortalOutlet></ng-template>\n        </div>\n\n        <div class=\"mt-4 flex flex-wrap gap-3\" [ngClass]=\"buttonAlignmentClass\">\n          <cqa-button\n            *ngFor=\"let button of config.buttons\"\n            type=\"button\"\n            [variant]=\"buttonVariant(button)\"\n            [ngClass]=\"buttonHostClasses(button)\"\n            (clicked)=\"onButtonClick(button)\"\n          >\n            {{ button.label }}\n          </cqa-button>\n        </div>\n      </div>\n    </div>\n  </div>\n</div>\n\n\n"]}
|
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
import { Component, EventEmitter, Input, Output } 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
|
+
/** Placeholder text for the input */
|
|
9
|
+
this.placeholder = 'Search';
|
|
10
|
+
/** Initial value or externally controlled value */
|
|
11
|
+
this.value = '';
|
|
12
|
+
/** Disable interactions */
|
|
13
|
+
this.disabled = false;
|
|
14
|
+
/** Whether the clear button should be visible when there is text */
|
|
15
|
+
this.showClear = true;
|
|
16
|
+
/** Accessible label for the input */
|
|
17
|
+
this.ariaLabel = 'Search';
|
|
18
|
+
/** Automatically focus the input when rendered */
|
|
19
|
+
this.autoFocus = false;
|
|
20
|
+
/** Search bar size */
|
|
21
|
+
this.size = 'md';
|
|
22
|
+
/** Stretch to fill container width */
|
|
23
|
+
this.fullWidth = false;
|
|
24
|
+
/** Emit on value changes (e.g. for two-way binding) */
|
|
25
|
+
this.valueChange = new EventEmitter();
|
|
26
|
+
/** Emit when user submits search (Enter key or form submit) */
|
|
27
|
+
this.search = new EventEmitter();
|
|
28
|
+
/** Emit when the value is cleared via the clear button */
|
|
29
|
+
this.cleared = new EventEmitter();
|
|
30
|
+
this.inputValue = '';
|
|
31
|
+
this.widthClasses = {
|
|
32
|
+
sm: 'w-[295px]',
|
|
33
|
+
md: 'w-[395px]',
|
|
34
|
+
lg: 'w-[495px]',
|
|
35
|
+
};
|
|
36
|
+
}
|
|
37
|
+
ngOnChanges(changes) {
|
|
38
|
+
if (changes['value'] && changes['value'].currentValue !== undefined) {
|
|
39
|
+
const newValue = changes['value'].currentValue ?? '';
|
|
40
|
+
if (newValue !== this.inputValue) {
|
|
41
|
+
this.inputValue = newValue;
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
onInput(event) {
|
|
46
|
+
const target = event.target;
|
|
47
|
+
const nextValue = target?.value ?? '';
|
|
48
|
+
this.inputValue = nextValue;
|
|
49
|
+
this.valueChange.emit(this.inputValue);
|
|
50
|
+
}
|
|
51
|
+
onSubmit(event) {
|
|
52
|
+
event.preventDefault();
|
|
53
|
+
if (this.disabled) {
|
|
54
|
+
return;
|
|
55
|
+
}
|
|
56
|
+
this.search.emit(this.inputValue.trim());
|
|
57
|
+
}
|
|
58
|
+
clear() {
|
|
59
|
+
if (this.disabled || this.inputValue === '') {
|
|
60
|
+
return;
|
|
61
|
+
}
|
|
62
|
+
this.inputValue = '';
|
|
63
|
+
this.valueChange.emit('');
|
|
64
|
+
this.cleared.emit();
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
SearchBarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SearchBarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
68
|
+
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" }, usesOnChanges: true, ngImport: i0, template: "<div id=\"cqa-ui-root\" [style.display]=\"fullWidth ? 'block' : 'inline-block'\" [style.width]=\"fullWidth ? '100%' : 'auto'\">\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</div>\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"] }] });
|
|
69
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SearchBarComponent, decorators: [{
|
|
70
|
+
type: Component,
|
|
71
|
+
args: [{ selector: 'cqa-search-bar', template: "<div id=\"cqa-ui-root\" [style.display]=\"fullWidth ? 'block' : 'inline-block'\" [style.width]=\"fullWidth ? '100%' : 'auto'\">\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</div>\n", styles: [] }]
|
|
72
|
+
}], propDecorators: { placeholder: [{
|
|
73
|
+
type: Input
|
|
74
|
+
}], value: [{
|
|
75
|
+
type: Input
|
|
76
|
+
}], disabled: [{
|
|
77
|
+
type: Input
|
|
78
|
+
}], showClear: [{
|
|
79
|
+
type: Input
|
|
80
|
+
}], ariaLabel: [{
|
|
81
|
+
type: Input
|
|
82
|
+
}], autoFocus: [{
|
|
83
|
+
type: Input
|
|
84
|
+
}], size: [{
|
|
85
|
+
type: Input
|
|
86
|
+
}], fullWidth: [{
|
|
87
|
+
type: Input
|
|
88
|
+
}], valueChange: [{
|
|
89
|
+
type: Output
|
|
90
|
+
}], search: [{
|
|
91
|
+
type: Output
|
|
92
|
+
}], cleared: [{
|
|
93
|
+
type: Output
|
|
94
|
+
}] } });
|
|
95
|
+
//# 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":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAa,MAAM,EAAiB,MAAM,eAAe,CAAC;;;;;AASjG,MAAM,OAAO,kBAAkB;IAL/B;QAOE,qCAAqC;QAC5B,gBAAW,GAAG,QAAQ,CAAC;QAEhC,mDAAmD;QAC1C,UAAK,GAAG,EAAE,CAAC;QAEpB,2BAA2B;QAClB,aAAQ,GAAG,KAAK,CAAC;QAE1B,oEAAoE;QAC3D,cAAS,GAAG,IAAI,CAAC;QAE1B,qCAAqC;QAC5B,cAAS,GAAG,QAAQ,CAAC;QAE9B,kDAAkD;QACzC,cAAS,GAAG,KAAK,CAAC;QAE3B,sBAAsB;QACb,SAAI,GAAkB,IAAI,CAAC;QAEpC,sCAAsC;QAC7B,cAAS,GAAG,KAAK,CAAC;QAE3B,uDAAuD;QAC7C,gBAAW,GAAG,IAAI,YAAY,EAAU,CAAC;QAEnD,+DAA+D;QACrD,WAAM,GAAG,IAAI,YAAY,EAAU,CAAC;QAE9C,0DAA0D;QAChD,YAAO,GAAG,IAAI,YAAY,EAAQ,CAAC;QAE7C,eAAU,GAAG,EAAE,CAAC;QAWP,iBAAY,GAAkC;YACrD,EAAE,EAAE,WAAW;YACf,EAAE,EAAE,WAAW;YACf,EAAE,EAAE,WAAW;SAChB,CAAC;KA2BH;IAxCC,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,OAAO,CAAC,IAAI,OAAO,CAAC,OAAO,CAAC,CAAC,YAAY,KAAK,SAAS,EAAE;YACnE,MAAM,QAAQ,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC,YAAY,IAAI,EAAE,CAAC;YACrD,IAAI,QAAQ,KAAK,IAAI,CAAC,UAAU,EAAE;gBAChC,IAAI,CAAC,UAAU,GAAG,QAAQ,CAAC;aAC5B;SACF;IACH,CAAC;IAQD,OAAO,CAAC,KAAY;QAClB,MAAM,MAAM,GAAG,KAAK,CAAC,MAAiC,CAAC;QACvD,MAAM,SAAS,GAAG,MAAM,EAAE,KAAK,IAAI,EAAE,CAAC;QACtC,IAAI,CAAC,UAAU,GAAG,SAAS,CAAC;QAC5B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IACzC,CAAC;IAED,QAAQ,CAAC,KAAY;QACnB,KAAK,CAAC,cAAc,EAAE,CAAC;QACvB,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,OAAO;SACR;QAED,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC,CAAC;IAC3C,CAAC;IAED,KAAK;QACH,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,UAAU,KAAK,EAAE,EAAE;YAC3C,OAAO;SACR;QAED,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC;QACrB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAC1B,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;IACtB,CAAC;;+GA5EU,kBAAkB;mGAAlB,kBAAkB,sVCT/B,wpEAuDA;2FD9Ca,kBAAkB;kBAL9B,SAAS;+BACE,gBAAgB;8BAOjB,WAAW;sBAAnB,KAAK;gBAGG,KAAK;sBAAb,KAAK;gBAGG,QAAQ;sBAAhB,KAAK;gBAGG,SAAS;sBAAjB,KAAK;gBAGG,SAAS;sBAAjB,KAAK;gBAGG,SAAS;sBAAjB,KAAK;gBAGG,IAAI;sBAAZ,KAAK;gBAGG,SAAS;sBAAjB,KAAK;gBAGI,WAAW;sBAApB,MAAM;gBAGG,MAAM;sBAAf,MAAM;gBAGG,OAAO;sBAAhB,MAAM","sourcesContent":["import { Component, EventEmitter, Input, OnChanges, Output, SimpleChanges } 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\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","<div id=\"cqa-ui-root\" [style.display]=\"fullWidth ? 'block' : 'inline-block'\" [style.width]=\"fullWidth ? '100%' : 'auto'\">\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</div>\n"]}
|
|
@@ -0,0 +1,211 @@
|
|
|
1
|
+
import { Component, EventEmitter, Input, Output, ViewChild, ViewChildren, } 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.segments = [
|
|
7
|
+
{ label: 'Tab Group', value: 'tab-group-1' },
|
|
8
|
+
{ label: 'Tab Group', value: 'tab-group-2' },
|
|
9
|
+
];
|
|
10
|
+
this.disabled = false;
|
|
11
|
+
this.valueChange = new EventEmitter();
|
|
12
|
+
this.indicatorStyle = {};
|
|
13
|
+
this.indicatorVisible = false;
|
|
14
|
+
}
|
|
15
|
+
ngOnChanges(changes) {
|
|
16
|
+
if (changes['segments'] || changes['value']) {
|
|
17
|
+
this.ensureSelectedValue();
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
ngAfterViewInit() {
|
|
21
|
+
this.buttonChangesSub = this.segmentButtons.changes.subscribe(() => this.updateIndicator());
|
|
22
|
+
this.ensureSelectedValue();
|
|
23
|
+
this.updateIndicator();
|
|
24
|
+
}
|
|
25
|
+
ngOnDestroy() {
|
|
26
|
+
this.buttonChangesSub?.unsubscribe?.();
|
|
27
|
+
}
|
|
28
|
+
trackByValue(_index, option) {
|
|
29
|
+
return option.value;
|
|
30
|
+
}
|
|
31
|
+
isSelected(option) {
|
|
32
|
+
return option.value === this.value;
|
|
33
|
+
}
|
|
34
|
+
select(option, index) {
|
|
35
|
+
if (this.disabled || option.disabled) {
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
const nextValue = option.value;
|
|
39
|
+
if (nextValue !== this.value) {
|
|
40
|
+
this.value = nextValue;
|
|
41
|
+
this.valueChange.emit(nextValue);
|
|
42
|
+
}
|
|
43
|
+
this.focusButton(index);
|
|
44
|
+
this.updateIndicator();
|
|
45
|
+
}
|
|
46
|
+
onKeyDown(event, currentIndex) {
|
|
47
|
+
if (this.disabled) {
|
|
48
|
+
return;
|
|
49
|
+
}
|
|
50
|
+
switch (event.key) {
|
|
51
|
+
case 'ArrowRight':
|
|
52
|
+
case 'ArrowDown':
|
|
53
|
+
event.preventDefault();
|
|
54
|
+
this.moveSelection(1, currentIndex);
|
|
55
|
+
break;
|
|
56
|
+
case 'ArrowLeft':
|
|
57
|
+
case 'ArrowUp':
|
|
58
|
+
event.preventDefault();
|
|
59
|
+
this.moveSelection(-1, currentIndex);
|
|
60
|
+
break;
|
|
61
|
+
case 'Home':
|
|
62
|
+
event.preventDefault();
|
|
63
|
+
this.selectFirstEnabled();
|
|
64
|
+
break;
|
|
65
|
+
case 'End':
|
|
66
|
+
event.preventDefault();
|
|
67
|
+
this.selectLastEnabled();
|
|
68
|
+
break;
|
|
69
|
+
case ' ':
|
|
70
|
+
case 'Enter':
|
|
71
|
+
event.preventDefault();
|
|
72
|
+
this.select(this.segments[currentIndex], currentIndex);
|
|
73
|
+
break;
|
|
74
|
+
default:
|
|
75
|
+
break;
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
moveSelection(step, startIndex) {
|
|
79
|
+
const enabledIndexes = this.getEnabledIndexes();
|
|
80
|
+
if (enabledIndexes.length === 0) {
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
const currentEnabledIndex = enabledIndexes.indexOf(startIndex);
|
|
84
|
+
const fallbackIndex = this.getSelectedIndex(enabledIndexes);
|
|
85
|
+
const baseIndex = currentEnabledIndex >= 0 ? currentEnabledIndex : fallbackIndex;
|
|
86
|
+
const nextPosition = (baseIndex + step + enabledIndexes.length) % enabledIndexes.length;
|
|
87
|
+
const targetIndex = enabledIndexes[nextPosition];
|
|
88
|
+
this.select(this.segments[targetIndex], targetIndex);
|
|
89
|
+
}
|
|
90
|
+
selectFirstEnabled() {
|
|
91
|
+
const enabledIndexes = this.getEnabledIndexes();
|
|
92
|
+
if (enabledIndexes.length > 0) {
|
|
93
|
+
const index = enabledIndexes[0];
|
|
94
|
+
this.select(this.segments[index], index);
|
|
95
|
+
this.updateIndicator();
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
selectLastEnabled() {
|
|
99
|
+
const enabledIndexes = this.getEnabledIndexes();
|
|
100
|
+
if (enabledIndexes.length > 0) {
|
|
101
|
+
const index = enabledIndexes[enabledIndexes.length - 1];
|
|
102
|
+
this.select(this.segments[index], index);
|
|
103
|
+
this.updateIndicator();
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
getEnabledIndexes() {
|
|
107
|
+
return this.segments
|
|
108
|
+
.map((option, index) => ({ option, index }))
|
|
109
|
+
.filter(({ option }) => !option.disabled)
|
|
110
|
+
.map(({ index }) => index);
|
|
111
|
+
}
|
|
112
|
+
getSelectedIndex(enabledIndexes) {
|
|
113
|
+
const current = this.segments.findIndex((option) => option.value === this.value && !option.disabled);
|
|
114
|
+
if (current >= 0) {
|
|
115
|
+
return enabledIndexes.indexOf(current);
|
|
116
|
+
}
|
|
117
|
+
return 0;
|
|
118
|
+
}
|
|
119
|
+
ensureSelectedValue() {
|
|
120
|
+
const enabled = this.segments.filter((option) => !option.disabled);
|
|
121
|
+
if (enabled.length === 0) {
|
|
122
|
+
this.value = undefined;
|
|
123
|
+
return;
|
|
124
|
+
}
|
|
125
|
+
if (!this.value || !this.segments.some((option) => option.value === this.value)) {
|
|
126
|
+
this.value = enabled[0].value;
|
|
127
|
+
this.valueChange.emit(this.value);
|
|
128
|
+
const index = this.segments.indexOf(enabled[0]);
|
|
129
|
+
this.focusButton(index);
|
|
130
|
+
this.updateIndicator(index);
|
|
131
|
+
return;
|
|
132
|
+
}
|
|
133
|
+
const selected = this.segments.find((option) => option.value === this.value);
|
|
134
|
+
if (selected?.disabled) {
|
|
135
|
+
this.value = enabled[0].value;
|
|
136
|
+
this.valueChange.emit(this.value);
|
|
137
|
+
const index = this.segments.indexOf(enabled[0]);
|
|
138
|
+
this.focusButton(index);
|
|
139
|
+
this.updateIndicator(index);
|
|
140
|
+
}
|
|
141
|
+
this.updateIndicator();
|
|
142
|
+
}
|
|
143
|
+
focusButton(index) {
|
|
144
|
+
queueMicrotask(() => {
|
|
145
|
+
const button = this.segmentButtons?.get(index)?.nativeElement;
|
|
146
|
+
button?.focus();
|
|
147
|
+
});
|
|
148
|
+
}
|
|
149
|
+
updateIndicator(preferredIndex) {
|
|
150
|
+
queueMicrotask(() => {
|
|
151
|
+
const container = this.segmentContainer?.nativeElement;
|
|
152
|
+
const buttons = this.segmentButtons?.toArray() ?? [];
|
|
153
|
+
if (!container || buttons.length === 0) {
|
|
154
|
+
this.indicatorVisible = false;
|
|
155
|
+
this.indicatorStyle = {};
|
|
156
|
+
return;
|
|
157
|
+
}
|
|
158
|
+
const index = preferredIndex ?? buttons.findIndex((button, idx) => this.segments[idx]?.value === this.value);
|
|
159
|
+
if (index === -1) {
|
|
160
|
+
this.indicatorVisible = false;
|
|
161
|
+
this.indicatorStyle = {};
|
|
162
|
+
return;
|
|
163
|
+
}
|
|
164
|
+
const buttonEl = buttons[index]?.nativeElement;
|
|
165
|
+
if (!buttonEl) {
|
|
166
|
+
this.indicatorVisible = false;
|
|
167
|
+
this.indicatorStyle = {};
|
|
168
|
+
return;
|
|
169
|
+
}
|
|
170
|
+
const containerRect = container.getBoundingClientRect();
|
|
171
|
+
const buttonRect = buttonEl.getBoundingClientRect();
|
|
172
|
+
const offsetLeft = buttonEl.offsetLeft;
|
|
173
|
+
const offsetTop = buttonEl.offsetTop;
|
|
174
|
+
const width = buttonEl.offsetWidth;
|
|
175
|
+
const height = buttonEl.offsetHeight;
|
|
176
|
+
const isDisabled = this.disabled || this.segments[index]?.disabled;
|
|
177
|
+
this.indicatorStyle = {
|
|
178
|
+
width: `${width}px`,
|
|
179
|
+
height: `${height}px`,
|
|
180
|
+
left: `${offsetLeft}px`,
|
|
181
|
+
top: `${offsetTop}px`,
|
|
182
|
+
backgroundColor: isDisabled ? '#9BA0F4' : '#3F43EE',
|
|
183
|
+
};
|
|
184
|
+
this.indicatorVisible = true;
|
|
185
|
+
});
|
|
186
|
+
}
|
|
187
|
+
get isIndicatorVisible() {
|
|
188
|
+
return this.indicatorVisible;
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
SegmentControlComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SegmentControlComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
192
|
+
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" }, viewQueries: [{ propertyName: "segmentContainer", first: true, predicate: ["segmentContainer"], descendants: true }, { propertyName: "segmentButtons", predicate: ["segmentButton"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div id=\"cqa-ui-root\" style=\"display: inline-block;\">\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</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"] }] });
|
|
193
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: SegmentControlComponent, decorators: [{
|
|
194
|
+
type: Component,
|
|
195
|
+
args: [{ selector: 'cqa-segment-control', template: "<div id=\"cqa-ui-root\" style=\"display: inline-block;\">\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</div>\n", styles: [] }]
|
|
196
|
+
}], propDecorators: { segments: [{
|
|
197
|
+
type: Input
|
|
198
|
+
}], value: [{
|
|
199
|
+
type: Input
|
|
200
|
+
}], disabled: [{
|
|
201
|
+
type: Input
|
|
202
|
+
}], valueChange: [{
|
|
203
|
+
type: Output
|
|
204
|
+
}], segmentButtons: [{
|
|
205
|
+
type: ViewChildren,
|
|
206
|
+
args: ['segmentButton']
|
|
207
|
+
}], segmentContainer: [{
|
|
208
|
+
type: ViewChild,
|
|
209
|
+
args: ['segmentContainer']
|
|
210
|
+
}] } });
|
|
211
|
+
//# 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  Input,\n  OnDestroy,\n  OnChanges,\n  Output,\n  QueryList,\n  SimpleChanges,\n  ViewChild,\n  ViewChildren,\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\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 id=\"cqa-ui-root\" style=\"display: inline-block;\">\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</div>\n"]}
|