@cqa-lib/cqa-ui 0.1.0 → 0.1.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.
Files changed (39) hide show
  1. package/dist/cqa-ui/README.md +226 -0
  2. package/dist/cqa-ui/esm2020/lib/button/button.component.mjs +257 -0
  3. package/dist/cqa-ui/esm2020/lib/dialog/dialog.component.mjs +127 -0
  4. package/dist/cqa-ui/esm2020/lib/search-bar/search-bar.component.mjs +95 -0
  5. package/{esm2020 → dist/cqa-ui/esm2020}/lib/segment-control/segment-control.component.mjs +3 -3
  6. package/{esm2020 → dist/cqa-ui/esm2020}/lib/ui-kit.module.mjs +6 -8
  7. package/{esm2020 → dist/cqa-ui/esm2020}/public-api.mjs +1 -2
  8. package/{fesm2015 → dist/cqa-ui/fesm2015}/cqa-lib-cqa-ui.mjs +96 -128
  9. package/dist/cqa-ui/fesm2015/cqa-lib-cqa-ui.mjs.map +1 -0
  10. package/{fesm2020 → dist/cqa-ui/fesm2020}/cqa-lib-cqa-ui.mjs +96 -128
  11. package/dist/cqa-ui/fesm2020/cqa-lib-cqa-ui.mjs.map +1 -0
  12. package/dist/cqa-ui/lib/ui-kit.module.d.ts +15 -0
  13. package/dist/cqa-ui/package.json +56 -0
  14. package/{public-api.d.ts → dist/cqa-ui/public-api.d.ts} +0 -1
  15. package/dist/cqa-ui/styles.css +1 -0
  16. package/package.json +49 -23
  17. package/esm2020/lib/button/button.component.mjs +0 -257
  18. package/esm2020/lib/dialog/dialog.component.mjs +0 -127
  19. package/esm2020/lib/root-wrapper/root-wrapper.component.mjs +0 -34
  20. package/esm2020/lib/search-bar/search-bar.component.mjs +0 -95
  21. package/fesm2015/cqa-lib-cqa-ui.mjs.map +0 -1
  22. package/fesm2020/cqa-lib-cqa-ui.mjs.map +0 -1
  23. package/lib/root-wrapper/root-wrapper.component.d.ts +0 -9
  24. package/lib/ui-kit.module.d.ts +0 -16
  25. package/styles.css +0 -1
  26. /package/{cqa-lib-cqa-ui.d.ts → dist/cqa-ui/cqa-lib-cqa-ui.d.ts} +0 -0
  27. /package/{esm2020 → dist/cqa-ui/esm2020}/cqa-lib-cqa-ui.mjs +0 -0
  28. /package/{esm2020 → dist/cqa-ui/esm2020}/lib/dialog/dialog-ref.mjs +0 -0
  29. /package/{esm2020 → dist/cqa-ui/esm2020}/lib/dialog/dialog.models.mjs +0 -0
  30. /package/{esm2020 → dist/cqa-ui/esm2020}/lib/dialog/dialog.service.mjs +0 -0
  31. /package/{esm2020 → dist/cqa-ui/esm2020}/lib/dialog/dialog.tokens.mjs +0 -0
  32. /package/{lib → dist/cqa-ui/lib}/button/button.component.d.ts +0 -0
  33. /package/{lib → dist/cqa-ui/lib}/dialog/dialog-ref.d.ts +0 -0
  34. /package/{lib → dist/cqa-ui/lib}/dialog/dialog.component.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}/search-bar/search-bar.component.d.ts +0 -0
  39. /package/{lib → dist/cqa-ui/lib}/segment-control/segment-control.component.d.ts +0 -0
@@ -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
+ 'cqa-flex',
23
+ 'cqa-flex-col',
24
+ 'cqa-justify-center',
25
+ 'cqa-items-center',
26
+ 'cqa-p-0',
27
+ 'cqa-gap-2',
28
+ 'cqa-rounded-lg',
29
+ 'cqa-cursor-pointer',
30
+ 'cqa-font-inter',
31
+ 'cqa-font-semibold',
32
+ 'cqa-text-sm',
33
+ 'cqa-leading-[14px]',
34
+ 'cqa-transition-all',
35
+ 'cqa-duration-200',
36
+ 'cqa-outline-none'
37
+ ];
38
+ if (this.disabled) {
39
+ baseClasses.push('cqa-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
+ 'cqa-flex',
48
+ 'cqa-flex-row',
49
+ 'cqa-justify-center',
50
+ 'cqa-items-center',
51
+ 'cqa-gap-2',
52
+ 'cqa-w-full',
53
+ 'cqa-h-full',
54
+ 'cqa-py-[10px]',
55
+ 'cqa-px-6',
56
+ ];
57
+ return classes.join(' ');
58
+ }
59
+ get labelClasses() {
60
+ const classes = [
61
+ 'cqa-flex',
62
+ 'cqa-items-center',
63
+ 'cqa-text-center',
64
+ 'cqa-font-inter',
65
+ 'cqa-font-semibold',
66
+ 'cqa-text-sm',
67
+ 'cqa-leading-[14px]',
68
+ 'cqa-flex-none',
69
+ this.textClass,
70
+ ];
71
+ if (this.disabled) {
72
+ classes.push('cqa-opacity-[0.38]');
73
+ }
74
+ return classes.join(' ');
75
+ }
76
+ get iconClasses() {
77
+ const classes = [
78
+ 'cqa-flex',
79
+ 'cqa-items-center',
80
+ 'cqa-justify-center',
81
+ 'cqa-w-[14px]',
82
+ 'cqa-h-[14px]',
83
+ 'cqa-shrink-0',
84
+ 'cqa-flex-none'
85
+ ];
86
+ if (this.disabled) {
87
+ classes.push('cqa-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('cqa-bg-primary-muted');
96
+ }
97
+ else {
98
+ classes.push('cqa-bg-primary');
99
+ if (this.isHovered) {
100
+ classes.push('cqa-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('cqa-bg-transparent', 'cqa-border', 'cqa-border-primary-muted');
107
+ }
108
+ else {
109
+ if (this.isFocused) {
110
+ classes.push('cqa-bg-primary-surface-alt', 'cqa-border', 'cqa-border-primary-hover', 'cqa-shadow-[0px_4px_4px_rgba(0,0,0,0.25)]');
111
+ }
112
+ else if (this.isHovered || this.isPressed) {
113
+ classes.push('cqa-bg-primary-surface', 'cqa-border', 'cqa-border-primary');
114
+ }
115
+ else {
116
+ classes.push('cqa-bg-transparent', 'cqa-border', 'cqa-border-slate');
117
+ }
118
+ }
119
+ }
120
+ else if (this.variant === 'text') {
121
+ if (this.disabled) {
122
+ classes.push('cqa-bg-transparent');
123
+ }
124
+ else {
125
+ classes.push('cqa-bg-transparent');
126
+ if (this.isHovered || this.isFocused || this.isPressed) {
127
+ classes.push('cqa-bg-primary-surface');
128
+ }
129
+ }
130
+ }
131
+ else if (this.variant === 'elevated') {
132
+ if (this.disabled) {
133
+ classes.push('cqa-bg-primary-muted', 'cqa-shadow-none');
134
+ }
135
+ else {
136
+ if (this.isFocused) {
137
+ classes.push('cqa-bg-primary-surface-alt', 'cqa-shadow-[0px_4px_4px_rgba(0,0,0,0.25)]');
138
+ }
139
+ else if (this.isPressed) {
140
+ classes.push('cqa-bg-primary-surface', 'cqa-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('cqa-bg-primary-surface-alt', 'cqa-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('cqa-bg-primary-surface', 'cqa-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('cqa-bg-primary-muted');
153
+ }
154
+ else {
155
+ if (this.isHovered) {
156
+ classes.push('cqa-bg-tonal-hover', 'cqa-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('cqa-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 'cqa-text-ink';
169
+ }
170
+ return 'cqa-text-ink-muted';
171
+ }
172
+ switch (this.variant) {
173
+ case 'filled':
174
+ return 'cqa-text-surface-default';
175
+ case 'outlined':
176
+ if (this.isFocused || this.isHovered || this.isPressed) {
177
+ return 'cqa-text-primary-hover';
178
+ }
179
+ return 'cqa-text-slate';
180
+ case 'text':
181
+ case 'elevated':
182
+ return 'cqa-text-primary-hover';
183
+ case 'tonal':
184
+ return 'cqa-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=\"cqa-text-[18px] cqa-leading-[18px] cqa-w-[18px] cqa-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=\"cqa-text-[18px] cqa-leading-[18px] cqa-w-[18px] cqa-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=\"cqa-text-[18px] cqa-leading-[18px] cqa-w-[18px] cqa-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=\"cqa-text-[18px] cqa-leading-[18px] cqa-w-[18px] cqa-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      'cqa-flex',\n      'cqa-flex-col',\n      'cqa-justify-center',\n      'cqa-items-center',\n      'cqa-p-0',\n      'cqa-gap-2',\n      'cqa-rounded-lg',\n      'cqa-cursor-pointer',\n      'cqa-font-inter',\n      'cqa-font-semibold',\n      'cqa-text-sm',\n      'cqa-leading-[14px]',\n      'cqa-transition-all',\n      'cqa-duration-200',\n      'cqa-outline-none'\n    ];\n\n    if (this.disabled) {\n      baseClasses.push('cqa-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      'cqa-flex',\n      'cqa-flex-row',\n      'cqa-justify-center',\n      'cqa-items-center',\n      'cqa-gap-2',\n      'cqa-w-full',\n      'cqa-h-full',\n      'cqa-py-[10px]',\n      'cqa-px-6',\n    ];\n\n    return classes.join(' ');\n  }\n\n  get labelClasses(): string {\n    const classes = [\n      'cqa-flex',\n      'cqa-items-center',\n      'cqa-text-center',\n      'cqa-font-inter',\n      'cqa-font-semibold',\n      'cqa-text-sm',\n      'cqa-leading-[14px]',\n      'cqa-flex-none',\n      this.textClass,\n    ];\n\n    if (this.disabled) {\n      classes.push('cqa-opacity-[0.38]');\n    }\n\n    return classes.join(' ');\n  }\n\n  get iconClasses(): string {\n    const classes = [\n      'cqa-flex',\n      'cqa-items-center',\n      'cqa-justify-center',\n      'cqa-w-[14px]',\n      'cqa-h-[14px]',\n      'cqa-shrink-0',\n      'cqa-flex-none'\n    ];\n    if (this.disabled) {\n      classes.push('cqa-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('cqa-bg-primary-muted');\n      } else {\n        classes.push('cqa-bg-primary');\n        if (this.isHovered) {\n          classes.push('cqa-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('cqa-bg-transparent', 'cqa-border', 'cqa-border-primary-muted');\n      } else {\n        if (this.isFocused) {\n          classes.push('cqa-bg-primary-surface-alt', 'cqa-border', 'cqa-border-primary-hover', 'cqa-shadow-[0px_4px_4px_rgba(0,0,0,0.25)]');\n        } else if (this.isHovered || this.isPressed) {\n          classes.push('cqa-bg-primary-surface', 'cqa-border', 'cqa-border-primary');\n        } else {\n          classes.push('cqa-bg-transparent', 'cqa-border', 'cqa-border-slate');\n        }\n      }\n    } else if (this.variant === 'text') {\n      if (this.disabled) {\n        classes.push('cqa-bg-transparent');\n      } else {\n        classes.push('cqa-bg-transparent');\n        if (this.isHovered || this.isFocused || this.isPressed) {\n          classes.push('cqa-bg-primary-surface');\n        }\n      }\n    } else if (this.variant === 'elevated') {\n      if (this.disabled) {\n        classes.push('cqa-bg-primary-muted', 'cqa-shadow-none');\n      } else {\n        if (this.isFocused) {\n          classes.push('cqa-bg-primary-surface-alt', 'cqa-shadow-[0px_4px_4px_rgba(0,0,0,0.25)]');\n        } else if (this.isPressed) {\n          classes.push('cqa-bg-primary-surface', 'cqa-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('cqa-bg-primary-surface-alt', 'cqa-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('cqa-bg-primary-surface', 'cqa-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('cqa-bg-primary-muted');\n      } else {\n        if (this.isHovered) {\n          classes.push('cqa-bg-tonal-hover', 'cqa-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('cqa-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 'cqa-text-ink';\n      }\n      return 'cqa-text-ink-muted';\n    }\n\n    switch (this.variant) {\n      case 'filled':\n        return 'cqa-text-surface-default';\n      case 'outlined':\n        if (this.isFocused || this.isHovered || this.isPressed) {\n          return 'cqa-text-primary-hover';\n        }\n        return 'cqa-text-slate';\n      case 'text':\n      case 'elevated':\n        return 'cqa-text-primary-hover';\n      case 'tonal':\n        return 'cqa-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=\"cqa-text-[18px] cqa-leading-[18px] cqa-w-[18px] cqa-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=\"cqa-text-[18px] cqa-leading-[18px] cqa-w-[18px] cqa-h-[18px]\">\n          {{ icon }}\n        </mat-icon>\n      </span>\n    </span>\n  </button>\n</div>\n\n"]}
@@ -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
+ 'cqa-relative',
53
+ 'cqa-w-full',
54
+ 'cqa-bg-white',
55
+ 'cqa-rounded-2xl',
56
+ 'cqa-shadow-md',
57
+ 'cqa-border',
58
+ 'cqa-border-[#E5E7EB]',
59
+ 'cqa-p-6',
60
+ 'cqa-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 'cqa-justify-start';
103
+ case 'center':
104
+ return 'cqa-justify-center';
105
+ case 'right':
106
+ default:
107
+ return 'cqa-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=\"cqa-flex cqa-w-full cqa-justify-center cqa-px-4 sm:cqa-px-6\">\n <div [ngClass]=\"panelClassList\" [ngStyle]=\"panelStyles\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-5\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <h2 class=\"cqa-text-lg cqa-font-semibold cqa-text-[#111827]\">\n {{ config.title }}\n </h2>\n\n <p *ngIf=\"config.description\" class=\"cqa-text-sm cqa-leading-6 cqa-text-[#4B5563]\">\n {{ config.description }}\n </p>\n\n <div\n *ngIf=\"config.warning\"\n class=\"cqa-rounded-xl cqa-border cqa-border-red-200 cqa-bg-red-50 cqa-px-4 cqa-py-3 cqa-text-sm cqa-leading-5 cqa-text-red-700\"\n >\n {{ config.warning }}\n </div>\n </div>\n\n <div class=\"cqa-text-sm cqa-text-[#111827]\" [class.hidden]=\"!contentAttached\">\n <ng-template cdkPortalOutlet></ng-template>\n </div>\n\n <div class=\"cqa-mt-4 cqa-flex cqa-flex-wrap cqa-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=\"cqa-flex cqa-w-full cqa-justify-center cqa-px-4 sm:cqa-px-6\">\n <div [ngClass]=\"panelClassList\" [ngStyle]=\"panelStyles\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-5\">\n <div class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n <h2 class=\"cqa-text-lg cqa-font-semibold cqa-text-[#111827]\">\n {{ config.title }}\n </h2>\n\n <p *ngIf=\"config.description\" class=\"cqa-text-sm cqa-leading-6 cqa-text-[#4B5563]\">\n {{ config.description }}\n </p>\n\n <div\n *ngIf=\"config.warning\"\n class=\"cqa-rounded-xl cqa-border cqa-border-red-200 cqa-bg-red-50 cqa-px-4 cqa-py-3 cqa-text-sm cqa-leading-5 cqa-text-red-700\"\n >\n {{ config.warning }}\n </div>\n </div>\n\n <div class=\"cqa-text-sm cqa-text-[#111827]\" [class.hidden]=\"!contentAttached\">\n <ng-template cdkPortalOutlet></ng-template>\n </div>\n\n <div class=\"cqa-mt-4 cqa-flex cqa-flex-wrap cqa-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      'cqa-relative',\n      'cqa-w-full',\n      'cqa-bg-white',\n      'cqa-rounded-2xl',\n      'cqa-shadow-md',\n      'cqa-border',\n      'cqa-border-[#E5E7EB]',\n      'cqa-p-6',\n      'cqa-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 'cqa-justify-start';\n      case 'center':\n        return 'cqa-justify-center';\n      case 'right':\n      default:\n        return 'cqa-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=\"cqa-flex cqa-w-full cqa-justify-center cqa-px-4 sm:cqa-px-6\">\n    <div [ngClass]=\"panelClassList\" [ngStyle]=\"panelStyles\">\n      <div class=\"cqa-flex cqa-flex-col cqa-gap-5\">\n        <div class=\"cqa-flex cqa-flex-col cqa-gap-3\">\n          <h2 class=\"cqa-text-lg cqa-font-semibold cqa-text-[#111827]\">\n            {{ config.title }}\n          </h2>\n\n          <p *ngIf=\"config.description\" class=\"cqa-text-sm cqa-leading-6 cqa-text-[#4B5563]\">\n            {{ config.description }}\n          </p>\n\n          <div\n            *ngIf=\"config.warning\"\n            class=\"cqa-rounded-xl cqa-border cqa-border-red-200 cqa-bg-red-50 cqa-px-4 cqa-py-3 cqa-text-sm cqa-leading-5 cqa-text-red-700\"\n          >\n            {{ config.warning }}\n          </div>\n        </div>\n\n        <div class=\"cqa-text-sm cqa-text-[#111827]\" [class.hidden]=\"!contentAttached\">\n          <ng-template cdkPortalOutlet></ng-template>\n        </div>\n\n        <div class=\"cqa-mt-4 cqa-flex cqa-flex-wrap cqa-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"]}