@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.
Files changed (40) hide show
  1. package/README.md +90 -4
  2. package/dist/cqa-ui/README.md +226 -0
  3. package/dist/cqa-ui/esm2020/lib/button/button.component.mjs +257 -0
  4. package/dist/cqa-ui/esm2020/lib/dialog/dialog.component.mjs +127 -0
  5. package/dist/cqa-ui/esm2020/lib/search-bar/search-bar.component.mjs +95 -0
  6. package/dist/cqa-ui/esm2020/lib/segment-control/segment-control.component.mjs +211 -0
  7. package/{fesm2015 → dist/cqa-ui/fesm2015}/cqa-lib-cqa-ui.mjs +13 -53
  8. package/dist/cqa-ui/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -0
  9. package/{fesm2020 → dist/cqa-ui/fesm2020}/cqa-lib-cqa-ui.mjs +13 -53
  10. package/dist/cqa-ui/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -0
  11. package/{lib → dist/cqa-ui/lib}/button/button.component.d.ts +0 -2
  12. package/{lib → dist/cqa-ui/lib}/dialog/dialog.component.d.ts +0 -2
  13. package/{lib → dist/cqa-ui/lib}/search-bar/search-bar.component.d.ts +0 -3
  14. package/{lib → dist/cqa-ui/lib}/segment-control/segment-control.component.d.ts +0 -2
  15. package/dist/cqa-ui/package.json +56 -0
  16. package/dist/cqa-ui/styles.css +1 -0
  17. package/package.json +49 -23
  18. package/esm2020/lib/button/button.component.mjs +0 -265
  19. package/esm2020/lib/dialog/dialog.component.mjs +0 -135
  20. package/esm2020/lib/search-bar/search-bar.component.mjs +0 -111
  21. package/esm2020/lib/segment-control/segment-control.component.mjs +0 -219
  22. package/fesm2015/cqa-lib-cqa-ui.mjs.map +0 -1
  23. package/fesm2020/cqa-lib-cqa-ui.mjs.map +0 -1
  24. package/styles.css +0 -1
  25. /package/{cqa-lib-cqa-ui.d.ts → dist/cqa-ui/cqa-lib-cqa-ui.d.ts} +0 -0
  26. /package/{esm2020 → dist/cqa-ui/esm2020}/cqa-lib-cqa-ui.mjs +0 -0
  27. /package/{esm2020 → dist/cqa-ui/esm2020}/lib/dialog/dialog-ref.mjs +0 -0
  28. /package/{esm2020 → dist/cqa-ui/esm2020}/lib/dialog/dialog.models.mjs +0 -0
  29. /package/{esm2020 → dist/cqa-ui/esm2020}/lib/dialog/dialog.service.mjs +0 -0
  30. /package/{esm2020 → dist/cqa-ui/esm2020}/lib/dialog/dialog.tokens.mjs +0 -0
  31. /package/{esm2020 → dist/cqa-ui/esm2020}/lib/root-wrapper/root-wrapper.component.mjs +0 -0
  32. /package/{esm2020 → dist/cqa-ui/esm2020}/lib/ui-kit.module.mjs +0 -0
  33. /package/{esm2020 → dist/cqa-ui/esm2020}/public-api.mjs +0 -0
  34. /package/{lib → dist/cqa-ui/lib}/dialog/dialog-ref.d.ts +0 -0
  35. /package/{lib → dist/cqa-ui/lib}/dialog/dialog.models.d.ts +0 -0
  36. /package/{lib → dist/cqa-ui/lib}/dialog/dialog.service.d.ts +0 -0
  37. /package/{lib → dist/cqa-ui/lib}/dialog/dialog.tokens.d.ts +0 -0
  38. /package/{lib → dist/cqa-ui/lib}/root-wrapper/root-wrapper.component.d.ts +0 -0
  39. /package/{lib → dist/cqa-ui/lib}/ui-kit.module.d.ts +0 -0
  40. /package/{public-api.d.ts → dist/cqa-ui/public-api.d.ts} +0 -0
package/README.md CHANGED
@@ -7,24 +7,26 @@ Component library for Angular 13+, built with Tailwind CSS tokens and Storybook-
7
7
  ## 📦 Installation
8
8
 
9
9
  ```bash
10
- npm install cqa-ui
10
+ npm install @cqa-lib/cqa-ui
11
11
  ```
12
12
 
13
13
  ### Peer dependencies
14
14
 
15
15
  ```bash
16
- npm install @angular/common@^13.4.0 @angular/core@^13.4.0 @angular/forms@^13.4.0 @angular/material@^13.3.9 rxjs@^7.5.0
16
+ npm install @angular/common@^13.4.0 @angular/core@^13.4.0 @angular/forms@^13.4.0 @angular/material@^13.3.9 @angular/cdk@^13.3.9 rxjs@^6.6.7 || ^7.5.0
17
17
  ```
18
18
 
19
19
  ---
20
20
 
21
21
  ## 🚀 Quick Start
22
22
 
23
+ ### Step 1: Import the Module
24
+
23
25
  ```ts
24
26
  // app.module.ts
25
27
  import { NgModule } from '@angular/core';
26
28
  import { BrowserModule } from '@angular/platform-browser';
27
- import { UiKitModule } from 'cqa-ui';
29
+ import { UiKitModule } from '@cqa-lib/cqa-ui';
28
30
 
29
31
  import { AppComponent } from './app.component';
30
32
 
@@ -39,6 +41,49 @@ import { AppComponent } from './app.component';
39
41
  export class AppModule {}
40
42
  ```
41
43
 
44
+ ### Step 2: Import the Styles
45
+
46
+ **Option A: In `angular.json`** (Recommended)
47
+
48
+ ```json
49
+ {
50
+ "projects": {
51
+ "your-project": {
52
+ "architect": {
53
+ "build": {
54
+ "options": {
55
+ "styles": [
56
+ "node_modules/@cqa-lib/cqa-ui/styles.css",
57
+ "src/styles.css"
58
+ ]
59
+ }
60
+ }
61
+ }
62
+ }
63
+ }
64
+ }
65
+ ```
66
+
67
+ **Option B: In `styles.css`**
68
+
69
+ ```css
70
+ @import '@cqa-lib/cqa-ui/styles.css';
71
+ ```
72
+
73
+ ### Step 3: Add Angular Material Icon Font
74
+
75
+ Add this to your `index.html`:
76
+
77
+ ```html
78
+ <link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet">
79
+ ```
80
+
81
+ Or in `styles.css`:
82
+
83
+ ```css
84
+ @import url('https://fonts.googleapis.com/icon?family=Material+Icons');
85
+ ```
86
+
42
87
  ---
43
88
 
44
89
  ## 💡 Usage Examples
@@ -84,7 +129,7 @@ export class AppModule {}
84
129
  ### Dialog
85
130
 
86
131
  ```ts
87
- import { DialogService } from 'cqa-ui';
132
+ import { DialogService } from '@cqa-lib/cqa-ui';
88
133
 
89
134
  constructor(private readonly dialog: DialogService) {}
90
135
 
@@ -107,6 +152,47 @@ openDialog(): void {
107
152
 
108
153
  ---
109
154
 
155
+ ## 🔧 Troubleshooting
156
+
157
+ ### Component renders as empty tag (no inner HTML)
158
+
159
+ If you see `<cqa-search-bar></cqa-search-bar>` without any inner content, check:
160
+
161
+ 1. **Missing styles import** - Add to `angular.json`:
162
+ ```json
163
+ "styles": [
164
+ "node_modules/@cqa-lib/cqa-ui/styles.css",
165
+ "src/styles.css"
166
+ ]
167
+ ```
168
+
169
+ 2. **Missing Angular Material Icon font** - Add to `index.html`:
170
+ ```html
171
+ <link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet">
172
+ ```
173
+
174
+ 3. **Check browser console** for errors:
175
+ - Missing peer dependencies (`@angular/material`, `@angular/cdk`)
176
+ - Module import errors
177
+ - Template compilation errors
178
+
179
+ 4. **Verify module import** in your `app.module.ts`:
180
+ ```ts
181
+ import { UiKitModule } from '@cqa-lib/cqa-ui';
182
+
183
+ @NgModule({
184
+ imports: [UiKitModule, ...]
185
+ })
186
+ ```
187
+
188
+ 5. **Clear Angular cache and rebuild**:
189
+ ```bash
190
+ rm -rf .angular
191
+ npm start # or ng serve
192
+ ```
193
+
194
+ ---
195
+
110
196
  ## 📚 Storybook
111
197
 
112
198
  ```bash
@@ -0,0 +1,226 @@
1
+ # CQA UI
2
+
3
+ Component library for Angular 13+, built with Tailwind CSS tokens and Storybook-driven documentation.
4
+
5
+ ---
6
+
7
+ ## 📦 Installation
8
+
9
+ ```bash
10
+ npm install @cqa-lib/cqa-ui
11
+ ```
12
+
13
+ ### Peer dependencies
14
+
15
+ ```bash
16
+ npm install @angular/common@^13.4.0 @angular/core@^13.4.0 @angular/forms@^13.4.0 @angular/material@^13.3.9 @angular/cdk@^13.3.9 rxjs@^6.6.7 || ^7.5.0
17
+ ```
18
+
19
+ ---
20
+
21
+ ## 🚀 Quick Start
22
+
23
+ ### Step 1: Import the Module
24
+
25
+ ```ts
26
+ // app.module.ts
27
+ import { NgModule } from '@angular/core';
28
+ import { BrowserModule } from '@angular/platform-browser';
29
+ import { UiKitModule } from '@cqa-lib/cqa-ui';
30
+
31
+ import { AppComponent } from './app.component';
32
+
33
+ @NgModule({
34
+ declarations: [AppComponent],
35
+ imports: [
36
+ BrowserModule,
37
+ UiKitModule
38
+ ],
39
+ bootstrap: [AppComponent]
40
+ })
41
+ export class AppModule {}
42
+ ```
43
+
44
+ ### Step 2: Import the Styles
45
+
46
+ **Option A: In `angular.json`** (Recommended)
47
+
48
+ ```json
49
+ {
50
+ "projects": {
51
+ "your-project": {
52
+ "architect": {
53
+ "build": {
54
+ "options": {
55
+ "styles": [
56
+ "node_modules/@cqa-lib/cqa-ui/styles.css",
57
+ "src/styles.css"
58
+ ]
59
+ }
60
+ }
61
+ }
62
+ }
63
+ }
64
+ }
65
+ ```
66
+
67
+ **Option B: In `styles.css`**
68
+
69
+ ```css
70
+ @import '@cqa-lib/cqa-ui/styles.css';
71
+ ```
72
+
73
+ ### Step 3: Add Angular Material Icon Font
74
+
75
+ Add this to your `index.html`:
76
+
77
+ ```html
78
+ <link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet">
79
+ ```
80
+
81
+ Or in `styles.css`:
82
+
83
+ ```css
84
+ @import url('https://fonts.googleapis.com/icon?family=Material+Icons');
85
+ ```
86
+
87
+ ---
88
+
89
+ ## 💡 Usage Examples
90
+
91
+ ### Button
92
+
93
+ ```html
94
+ <cqa-button variant="filled" icon="save" (clicked)="onSave()">
95
+ Save changes
96
+ </cqa-button>
97
+
98
+ <cqa-button variant="outlined" [disabled]="isSubmitting">
99
+ Cancel
100
+ </cqa-button>
101
+ ```
102
+
103
+ ### Search bar
104
+
105
+ ```html
106
+ <cqa-search-bar
107
+ placeholder="Search components"
108
+ [value]="query"
109
+ [showClear]="true"
110
+ (valueChange)="query = $event"
111
+ (search)="onSearch($event)"
112
+ ></cqa-search-bar>
113
+ ```
114
+
115
+ ### Segment control
116
+
117
+ ```html
118
+ <cqa-segment-control
119
+ [segments]="[
120
+ { label: 'Overview', value: 'overview' },
121
+ { label: 'Analytics', value: 'analytics' },
122
+ { label: 'Settings', value: 'settings', disabled: true }
123
+ ]"
124
+ [value]="currentTab"
125
+ (valueChange)="currentTab = $event"
126
+ ></cqa-segment-control>
127
+ ```
128
+
129
+ ### Dialog
130
+
131
+ ```ts
132
+ import { DialogService } from '@cqa-lib/cqa-ui';
133
+
134
+ constructor(private readonly dialog: DialogService) {}
135
+
136
+ openDialog(): void {
137
+ this.dialog.open({
138
+ title: 'Delete dashboard',
139
+ description: 'Deleting this dashboard will remove it for all collaborators.',
140
+ warning: 'This action cannot be undone.',
141
+ content: {
142
+ type: 'text',
143
+ text: 'Are you sure you want to continue?'
144
+ },
145
+ buttons: [
146
+ { label: 'Cancel', role: 'secondary' },
147
+ { label: 'Delete', role: 'warn', handler: () => 'delete' }
148
+ ]
149
+ });
150
+ }
151
+ ```
152
+
153
+ ---
154
+
155
+ ## 🔧 Troubleshooting
156
+
157
+ ### Component renders as empty tag (no inner HTML)
158
+
159
+ If you see `<cqa-search-bar></cqa-search-bar>` without any inner content, check:
160
+
161
+ 1. **Missing styles import** - Add to `angular.json`:
162
+ ```json
163
+ "styles": [
164
+ "node_modules/@cqa-lib/cqa-ui/styles.css",
165
+ "src/styles.css"
166
+ ]
167
+ ```
168
+
169
+ 2. **Missing Angular Material Icon font** - Add to `index.html`:
170
+ ```html
171
+ <link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet">
172
+ ```
173
+
174
+ 3. **Check browser console** for errors:
175
+ - Missing peer dependencies (`@angular/material`, `@angular/cdk`)
176
+ - Module import errors
177
+ - Template compilation errors
178
+
179
+ 4. **Verify module import** in your `app.module.ts`:
180
+ ```ts
181
+ import { UiKitModule } from '@cqa-lib/cqa-ui';
182
+
183
+ @NgModule({
184
+ imports: [UiKitModule, ...]
185
+ })
186
+ ```
187
+
188
+ 5. **Clear Angular cache and rebuild**:
189
+ ```bash
190
+ rm -rf .angular
191
+ npm start # or ng serve
192
+ ```
193
+
194
+ ---
195
+
196
+ ## 📚 Storybook
197
+
198
+ ```bash
199
+ npm run storybook
200
+ ```
201
+
202
+ View component documentation and interactive examples at `http://localhost:6006`.
203
+
204
+ ---
205
+
206
+ ## 🛠 Development
207
+
208
+ ```bash
209
+ npm install
210
+ npm run build:cqa-ui # build the library bundle
211
+ npm run storybook # start Storybook
212
+ npm run build-storybook # static Storybook build
213
+ ```
214
+
215
+ ### Publish workflow
216
+
217
+ 1. Update version in `package.json`
218
+ 2. `npm run build:cqa-ui`
219
+ 3. `npm publish dist/cqa-ui`
220
+
221
+ ---
222
+
223
+ ## 📄 License
224
+
225
+ MIT — contributions and issues welcome!
226
+
@@ -0,0 +1,257 @@
1
+ import { Component, Input, Output, EventEmitter, HostListener } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/material/icon";
4
+ import * as i2 from "@angular/common";
5
+ export class ButtonComponent {
6
+ constructor() {
7
+ this.variant = 'filled';
8
+ this.disabled = false;
9
+ this.iconPosition = 'start';
10
+ this.type = 'button';
11
+ this.clicked = new EventEmitter();
12
+ // Internal state tracking
13
+ this.isHovered = false;
14
+ this.isFocused = false;
15
+ this.isPressed = false;
16
+ }
17
+ get hasIcon() {
18
+ return !!this.icon;
19
+ }
20
+ get buttonClasses() {
21
+ const baseClasses = [
22
+ 'flex',
23
+ 'flex-col',
24
+ 'justify-center',
25
+ 'items-center',
26
+ 'p-0',
27
+ 'gap-2',
28
+ 'rounded-lg',
29
+ 'cursor-pointer',
30
+ 'font-inter',
31
+ 'font-semibold',
32
+ 'text-sm',
33
+ 'leading-[14px]',
34
+ 'transition-all',
35
+ 'duration-200',
36
+ 'outline-none'
37
+ ];
38
+ if (this.disabled) {
39
+ baseClasses.push('cursor-not-allowed');
40
+ }
41
+ // Add variant and state specific classes
42
+ const variantClasses = this.getVariantClasses();
43
+ return [...baseClasses, ...variantClasses].join(' ');
44
+ }
45
+ get stateLayerClasses() {
46
+ const classes = [
47
+ 'flex',
48
+ 'flex-row',
49
+ 'justify-center',
50
+ 'items-center',
51
+ 'gap-2',
52
+ 'w-full',
53
+ 'h-full',
54
+ 'py-[10px]',
55
+ 'px-6',
56
+ ];
57
+ return classes.join(' ');
58
+ }
59
+ get labelClasses() {
60
+ const classes = [
61
+ 'flex',
62
+ 'items-center',
63
+ 'text-center',
64
+ 'font-inter',
65
+ 'font-semibold',
66
+ 'text-sm',
67
+ 'leading-[14px]',
68
+ 'flex-none',
69
+ this.textClass,
70
+ ];
71
+ if (this.disabled) {
72
+ classes.push('opacity-[0.38]');
73
+ }
74
+ return classes.join(' ');
75
+ }
76
+ get iconClasses() {
77
+ const classes = [
78
+ 'flex',
79
+ 'items-center',
80
+ 'justify-center',
81
+ 'w-[14px]',
82
+ 'h-[14px]',
83
+ 'shrink-0',
84
+ 'flex-none'
85
+ ];
86
+ if (this.disabled) {
87
+ classes.push('opacity-[0.38]');
88
+ }
89
+ return classes.join(' ');
90
+ }
91
+ getVariantClasses() {
92
+ const classes = [];
93
+ if (this.variant === 'filled') {
94
+ if (this.disabled) {
95
+ classes.push('bg-primary-muted');
96
+ }
97
+ else {
98
+ classes.push('bg-primary');
99
+ if (this.isHovered) {
100
+ classes.push('shadow-[0px_1px_2px_rgba(0,0,0,0.3),0px_1px_3px_1px_rgba(0,0,0,0.15)]');
101
+ }
102
+ }
103
+ }
104
+ else if (this.variant === 'outlined') {
105
+ if (this.disabled) {
106
+ classes.push('bg-transparent', 'border', 'border-primary-muted');
107
+ }
108
+ else {
109
+ if (this.isFocused) {
110
+ classes.push('bg-primary-surface-alt', 'border', 'border-primary-hover', 'shadow-[0px_4px_4px_rgba(0,0,0,0.25)]');
111
+ }
112
+ else if (this.isHovered || this.isPressed) {
113
+ classes.push('bg-primary-surface', 'border', 'border-primary');
114
+ }
115
+ else {
116
+ classes.push('bg-transparent', 'border', 'border-slate');
117
+ }
118
+ }
119
+ }
120
+ else if (this.variant === 'text') {
121
+ if (this.disabled) {
122
+ classes.push('bg-transparent');
123
+ }
124
+ else {
125
+ classes.push('bg-transparent');
126
+ if (this.isHovered || this.isFocused || this.isPressed) {
127
+ classes.push('bg-primary-surface');
128
+ }
129
+ }
130
+ }
131
+ else if (this.variant === 'elevated') {
132
+ if (this.disabled) {
133
+ classes.push('bg-primary-muted', 'shadow-none');
134
+ }
135
+ else {
136
+ if (this.isFocused) {
137
+ classes.push('bg-primary-surface-alt', 'shadow-[0px_4px_4px_rgba(0,0,0,0.25)]');
138
+ }
139
+ else if (this.isPressed) {
140
+ classes.push('bg-primary-surface', 'shadow-[0px_1px_2px_rgba(0,0,0,0.3),0px_1px_3px_1px_rgba(0,0,0,0.15)]');
141
+ }
142
+ else if (this.isHovered) {
143
+ classes.push('bg-primary-surface-alt', 'shadow-[0px_1px_2px_rgba(0,0,0,0.3),0px_2px_6px_2px_rgba(0,0,0,0.15)]');
144
+ }
145
+ else {
146
+ classes.push('bg-primary-surface', 'shadow-[0px_1px_2px_rgba(0,0,0,0.3),0px_1px_3px_1px_rgba(0,0,0,0.15)]');
147
+ }
148
+ }
149
+ }
150
+ else if (this.variant === 'tonal') {
151
+ if (this.disabled) {
152
+ classes.push('bg-primary-muted');
153
+ }
154
+ else {
155
+ if (this.isHovered) {
156
+ classes.push('bg-tonal-hover', 'shadow-[0px_1px_2px_rgba(0,0,0,0.3),0px_1px_3px_1px_rgba(0,0,0,0.15)]');
157
+ }
158
+ else {
159
+ classes.push('bg-primary-surface-alt');
160
+ }
161
+ }
162
+ }
163
+ return classes;
164
+ }
165
+ get textClass() {
166
+ if (this.disabled) {
167
+ if (this.variant === 'outlined' || this.variant === 'text') {
168
+ return 'text-ink';
169
+ }
170
+ return 'text-ink-muted';
171
+ }
172
+ switch (this.variant) {
173
+ case 'filled':
174
+ return 'text-surface-default';
175
+ case 'outlined':
176
+ if (this.isFocused || this.isHovered || this.isPressed) {
177
+ return 'text-primary-hover';
178
+ }
179
+ return 'text-slate';
180
+ case 'text':
181
+ case 'elevated':
182
+ return 'text-primary-hover';
183
+ case 'tonal':
184
+ return 'text-ink';
185
+ default:
186
+ return '';
187
+ }
188
+ }
189
+ onMouseEnter() {
190
+ if (!this.disabled) {
191
+ this.isHovered = true;
192
+ }
193
+ }
194
+ onMouseLeave() {
195
+ this.isHovered = false;
196
+ this.isPressed = false;
197
+ }
198
+ onMouseDown() {
199
+ if (!this.disabled) {
200
+ this.isPressed = true;
201
+ }
202
+ }
203
+ onMouseUp() {
204
+ this.isPressed = false;
205
+ }
206
+ onFocus() {
207
+ if (!this.disabled) {
208
+ this.isFocused = true;
209
+ }
210
+ }
211
+ onBlur() {
212
+ this.isFocused = false;
213
+ this.isPressed = false;
214
+ }
215
+ onClick(event) {
216
+ if (!this.disabled) {
217
+ this.clicked.emit(event);
218
+ }
219
+ }
220
+ }
221
+ ButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ButtonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
222
+ ButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.4.0", type: ButtonComponent, selector: "cqa-button", inputs: { variant: "variant", disabled: "disabled", icon: "icon", iconPosition: "iconPosition", type: "type" }, outputs: { clicked: "clicked" }, host: { listeners: { "mouseenter": "onMouseEnter()", "mouseleave": "onMouseLeave()", "mousedown": "onMouseDown()", "mouseup": "onMouseUp()", "focus": "onFocus()", "blur": "onBlur()" } }, ngImport: i0, template: "<div id=\"cqa-ui-root\" style=\"display: inline-block; width: auto;\">\n <button\n [type]=\"type\"\n [disabled]=\"disabled\"\n [attr.aria-disabled]=\"disabled\"\n [class]=\"buttonClasses\"\n (click)=\"onClick($event)\"\n >\n <span [class]=\"stateLayerClasses\">\n <span *ngIf=\"icon && iconPosition === 'start'\" [class]=\"iconClasses\" [ngClass]=\"textClass\">\n <mat-icon class=\"text-[18px] leading-[18px] w-[18px] h-[18px]\">\n {{ icon }}\n </mat-icon>\n </span>\n <span [class]=\"labelClasses\" [ngClass]=\"textClass\">\n <ng-content></ng-content>\n </span>\n <span *ngIf=\"icon && iconPosition === 'end'\" [class]=\"iconClasses\" [ngClass]=\"textClass\">\n <mat-icon class=\"text-[18px] leading-[18px] w-[18px] h-[18px]\">\n {{ icon }}\n </mat-icon>\n </span>\n </span>\n </button>\n</div>\n\n", components: [{ type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }], directives: [{ type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }] });
223
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.4.0", ngImport: i0, type: ButtonComponent, decorators: [{
224
+ type: Component,
225
+ args: [{ selector: 'cqa-button', template: "<div id=\"cqa-ui-root\" style=\"display: inline-block; width: auto;\">\n <button\n [type]=\"type\"\n [disabled]=\"disabled\"\n [attr.aria-disabled]=\"disabled\"\n [class]=\"buttonClasses\"\n (click)=\"onClick($event)\"\n >\n <span [class]=\"stateLayerClasses\">\n <span *ngIf=\"icon && iconPosition === 'start'\" [class]=\"iconClasses\" [ngClass]=\"textClass\">\n <mat-icon class=\"text-[18px] leading-[18px] w-[18px] h-[18px]\">\n {{ icon }}\n </mat-icon>\n </span>\n <span [class]=\"labelClasses\" [ngClass]=\"textClass\">\n <ng-content></ng-content>\n </span>\n <span *ngIf=\"icon && iconPosition === 'end'\" [class]=\"iconClasses\" [ngClass]=\"textClass\">\n <mat-icon class=\"text-[18px] leading-[18px] w-[18px] h-[18px]\">\n {{ icon }}\n </mat-icon>\n </span>\n </span>\n </button>\n</div>\n\n", styles: [] }]
226
+ }], propDecorators: { variant: [{
227
+ type: Input
228
+ }], disabled: [{
229
+ type: Input
230
+ }], icon: [{
231
+ type: Input
232
+ }], iconPosition: [{
233
+ type: Input
234
+ }], type: [{
235
+ type: Input
236
+ }], clicked: [{
237
+ type: Output
238
+ }], onMouseEnter: [{
239
+ type: HostListener,
240
+ args: ['mouseenter']
241
+ }], onMouseLeave: [{
242
+ type: HostListener,
243
+ args: ['mouseleave']
244
+ }], onMouseDown: [{
245
+ type: HostListener,
246
+ args: ['mousedown']
247
+ }], onMouseUp: [{
248
+ type: HostListener,
249
+ args: ['mouseup']
250
+ }], onFocus: [{
251
+ type: HostListener,
252
+ args: ['focus']
253
+ }], onBlur: [{
254
+ type: HostListener,
255
+ args: ['blur']
256
+ }] } });
257
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"button.component.js","sourceRoot":"","sources":["../../../../../src/lib/button/button.component.ts","../../../../../src/lib/button/button.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, HostListener } from '@angular/core';\n\nexport type ButtonVariant = 'filled' | 'outlined' | 'text' | 'elevated' | 'tonal';\n\n@Component({\n  selector: 'cqa-button',\n  templateUrl: './button.component.html',\n  styleUrls: []\n})\nexport class ButtonComponent {\n\n  @Input() variant: ButtonVariant = 'filled';\n  @Input() disabled: boolean = false;\n  @Input() icon?: string;\n  @Input() iconPosition: 'start' | 'end' = 'start';\n  @Input() type: 'button' | 'submit' | 'reset' = 'button';\n  @Output() clicked = new EventEmitter<MouseEvent>();\n\n  // Internal state tracking\n  isHovered: boolean = false;\n  isFocused: boolean = false;\n  isPressed: boolean = false;\n\n  get hasIcon(): boolean {\n    return !!this.icon;\n  }\n\n  get buttonClasses(): string {\n    const baseClasses = [\n      'flex',\n      'flex-col',\n      'justify-center',\n      'items-center',\n      'p-0',\n      'gap-2',\n      'rounded-lg',\n      'cursor-pointer',\n      'font-inter',\n      'font-semibold',\n      'text-sm',\n      'leading-[14px]',\n      'transition-all',\n      'duration-200',\n      'outline-none'\n    ];\n\n    if (this.disabled) {\n      baseClasses.push('cursor-not-allowed');\n    }\n\n    // Add variant and state specific classes\n    const variantClasses = this.getVariantClasses();\n    \n    return [...baseClasses, ...variantClasses].join(' ');\n  }\n\n  get stateLayerClasses(): string {\n    const classes = [\n      'flex',\n      'flex-row',\n      'justify-center',\n      'items-center',\n      'gap-2',\n      'w-full',\n      'h-full',\n      'py-[10px]',\n      'px-6',\n    ];\n\n    return classes.join(' ');\n  }\n\n  get labelClasses(): string {\n    const classes = [\n      'flex',\n      'items-center',\n      'text-center',\n      'font-inter',\n      'font-semibold',\n      'text-sm',\n      'leading-[14px]',\n      'flex-none',\n      this.textClass,\n    ];\n\n    if (this.disabled) {\n      classes.push('opacity-[0.38]');\n    }\n\n    return classes.join(' ');\n  }\n\n  get iconClasses(): string {\n    const classes = [\n      'flex',\n      'items-center',\n      'justify-center',\n      'w-[14px]',\n      'h-[14px]',\n      'shrink-0',\n      'flex-none'\n    ];\n    if (this.disabled) {\n      classes.push('opacity-[0.38]');\n    }\n    return classes.join(' ');\n  }\n\n  private getVariantClasses(): string[] {\n    const classes: string[] = [];\n\n    if (this.variant === 'filled') {\n      if (this.disabled) {\n        classes.push('bg-primary-muted');\n      } else {\n        classes.push('bg-primary');\n        if (this.isHovered) {\n          classes.push('shadow-[0px_1px_2px_rgba(0,0,0,0.3),0px_1px_3px_1px_rgba(0,0,0,0.15)]');\n        }\n      }\n    } else if (this.variant === 'outlined') {\n      if (this.disabled) {\n        classes.push('bg-transparent', 'border', 'border-primary-muted');\n      } else {\n        if (this.isFocused) {\n          classes.push('bg-primary-surface-alt', 'border', 'border-primary-hover', 'shadow-[0px_4px_4px_rgba(0,0,0,0.25)]');\n        } else if (this.isHovered || this.isPressed) {\n          classes.push('bg-primary-surface', 'border', 'border-primary');\n        } else {\n          classes.push('bg-transparent', 'border', 'border-slate');\n        }\n      }\n    } else if (this.variant === 'text') {\n      if (this.disabled) {\n        classes.push('bg-transparent');\n      } else {\n        classes.push('bg-transparent');\n        if (this.isHovered || this.isFocused || this.isPressed) {\n          classes.push('bg-primary-surface');\n        }\n      }\n    } else if (this.variant === 'elevated') {\n      if (this.disabled) {\n        classes.push('bg-primary-muted', 'shadow-none');\n      } else {\n        if (this.isFocused) {\n          classes.push('bg-primary-surface-alt', 'shadow-[0px_4px_4px_rgba(0,0,0,0.25)]');\n        } else if (this.isPressed) {\n          classes.push('bg-primary-surface', 'shadow-[0px_1px_2px_rgba(0,0,0,0.3),0px_1px_3px_1px_rgba(0,0,0,0.15)]');\n        } else if (this.isHovered) {\n          classes.push('bg-primary-surface-alt', 'shadow-[0px_1px_2px_rgba(0,0,0,0.3),0px_2px_6px_2px_rgba(0,0,0,0.15)]');\n        } else {\n          classes.push('bg-primary-surface', 'shadow-[0px_1px_2px_rgba(0,0,0,0.3),0px_1px_3px_1px_rgba(0,0,0,0.15)]');\n        }\n      }\n    } else if (this.variant === 'tonal') {\n      if (this.disabled) {\n        classes.push('bg-primary-muted');\n      } else {\n        if (this.isHovered) {\n          classes.push('bg-tonal-hover', 'shadow-[0px_1px_2px_rgba(0,0,0,0.3),0px_1px_3px_1px_rgba(0,0,0,0.15)]');\n        } else {\n          classes.push('bg-primary-surface-alt');\n        }\n      }\n    }\n\n    return classes;\n  }\n\n  get textClass(): string {\n    if (this.disabled) {\n      if (this.variant === 'outlined' || this.variant === 'text') {\n        return 'text-ink';\n      }\n      return 'text-ink-muted';\n    }\n\n    switch (this.variant) {\n      case 'filled':\n        return 'text-surface-default';\n      case 'outlined':\n        if (this.isFocused || this.isHovered || this.isPressed) {\n          return 'text-primary-hover';\n        }\n        return 'text-slate';\n      case 'text':\n      case 'elevated':\n        return 'text-primary-hover';\n      case 'tonal':\n        return 'text-ink';\n      default:\n        return '';\n    }\n  }\n\n  @HostListener('mouseenter')\n  onMouseEnter() {\n    if (!this.disabled) {\n      this.isHovered = true;\n    }\n  }\n\n  @HostListener('mouseleave')\n  onMouseLeave() {\n    this.isHovered = false;\n    this.isPressed = false;\n  }\n\n  @HostListener('mousedown')\n  onMouseDown() {\n    if (!this.disabled) {\n      this.isPressed = true;\n    }\n  }\n\n  @HostListener('mouseup')\n  onMouseUp() {\n    this.isPressed = false;\n  }\n\n  @HostListener('focus')\n  onFocus() {\n    if (!this.disabled) {\n      this.isFocused = true;\n    }\n  }\n\n  @HostListener('blur')\n  onBlur() {\n    this.isFocused = false;\n    this.isPressed = false;\n  }\n\n  onClick(event: MouseEvent) {\n    if (!this.disabled) {\n      this.clicked.emit(event);\n    }\n  }\n}\n\n","<div id=\"cqa-ui-root\" style=\"display: inline-block; width: auto;\">\n  <button\n    [type]=\"type\"\n    [disabled]=\"disabled\"\n    [attr.aria-disabled]=\"disabled\"\n    [class]=\"buttonClasses\"\n    (click)=\"onClick($event)\"\n  >\n    <span [class]=\"stateLayerClasses\">\n      <span *ngIf=\"icon && iconPosition === 'start'\" [class]=\"iconClasses\" [ngClass]=\"textClass\">\n        <mat-icon class=\"text-[18px] leading-[18px] w-[18px] h-[18px]\">\n          {{ icon }}\n        </mat-icon>\n      </span>\n      <span [class]=\"labelClasses\" [ngClass]=\"textClass\">\n        <ng-content></ng-content>\n      </span>\n      <span *ngIf=\"icon && iconPosition === 'end'\" [class]=\"iconClasses\" [ngClass]=\"textClass\">\n        <mat-icon class=\"text-[18px] leading-[18px] w-[18px] h-[18px]\">\n          {{ icon }}\n        </mat-icon>\n      </span>\n    </span>\n  </button>\n</div>\n\n"]}