@schenkerjon/ng-govbr-tw 0.0.4 → 0.0.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,137 @@
1
+ import { Component, Input, ViewEncapsulation } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class BrAvatarComponent {
4
+ constructor() {
5
+ this.size = 'medium';
6
+ this.variant = 'primary';
7
+ this.imageFailed = false;
8
+ }
9
+ onImageError() {
10
+ this.imageFailed = true;
11
+ this.src = undefined;
12
+ }
13
+ get initials() {
14
+ if (!this.name)
15
+ return '?';
16
+ const parts = this.name.trim().split(/\s+/);
17
+ if (parts.length === 1)
18
+ return parts[0].charAt(0).toUpperCase();
19
+ return (parts[0].charAt(0) + parts[parts.length - 1].charAt(0)).toUpperCase();
20
+ }
21
+ get containerClasses() {
22
+ const sizeMap = {
23
+ xsmall: 'w-6 h-6',
24
+ small: 'w-8 h-8',
25
+ medium: 'w-10 h-10',
26
+ large: 'w-14 h-14',
27
+ xlarge: 'w-20 h-20',
28
+ };
29
+ const variantMap = {
30
+ primary: 'bg-govbr-primary text-white',
31
+ neutral: 'bg-govbr-gray-20 text-govbr-gray-80',
32
+ success: 'bg-govbr-success text-white',
33
+ danger: 'bg-govbr-danger text-white',
34
+ warning: 'bg-govbr-warning text-govbr-gray-80',
35
+ };
36
+ return [
37
+ 'relative inline-flex items-center justify-center rounded-full overflow-hidden flex-shrink-0 font-rawline font-semibold',
38
+ sizeMap[this.size],
39
+ this.src ? '' : variantMap[this.variant],
40
+ ].filter(Boolean).join(' ');
41
+ }
42
+ get initialsClasses() {
43
+ const fontMap = {
44
+ xsmall: 'text-[10px]',
45
+ small: 'text-govbr-xs',
46
+ medium: 'text-govbr-sm',
47
+ large: 'text-govbr-lg',
48
+ xlarge: 'text-govbr-2xl',
49
+ };
50
+ return fontMap[this.size];
51
+ }
52
+ get statusClasses() {
53
+ const colorMap = {
54
+ online: 'bg-govbr-success',
55
+ offline: 'bg-govbr-gray-40',
56
+ busy: 'bg-govbr-danger',
57
+ away: 'bg-govbr-warning',
58
+ };
59
+ const dotSize = {
60
+ xsmall: 'w-1.5 h-1.5 border',
61
+ small: 'w-2 h-2 border',
62
+ medium: 'w-2.5 h-2.5 border-2',
63
+ large: 'w-3 h-3 border-2',
64
+ xlarge: 'w-4 h-4 border-2',
65
+ };
66
+ return [
67
+ 'absolute bottom-0 right-0 rounded-full border-white',
68
+ dotSize[this.size],
69
+ colorMap[this.status],
70
+ ].join(' ');
71
+ }
72
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrAvatarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
73
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: BrAvatarComponent, isStandalone: true, selector: "br-avatar", inputs: { src: "src", name: "name", size: "size", variant: "variant", status: "status" }, ngImport: i0, template: `
74
+ <div [class]="containerClasses" [attr.aria-label]="name || 'Avatar'" role="img">
75
+ @if (src) {
76
+ <img
77
+ [src]="src"
78
+ [alt]="name || 'Avatar'"
79
+ class="w-full h-full object-cover"
80
+ (error)="onImageError()"
81
+ />
82
+ } @else {
83
+ <span [class]="initialsClasses">{{ initials }}</span>
84
+ }
85
+
86
+ @if (status) {
87
+ <span
88
+ [class]="statusClasses"
89
+ [attr.aria-label]="'Status: ' + status"
90
+ role="status"
91
+ ></span>
92
+ }
93
+ </div>
94
+ `, isInline: true, encapsulation: i0.ViewEncapsulation.None }); }
95
+ }
96
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrAvatarComponent, decorators: [{
97
+ type: Component,
98
+ args: [{
99
+ selector: 'br-avatar',
100
+ standalone: true,
101
+ imports: [],
102
+ encapsulation: ViewEncapsulation.None,
103
+ template: `
104
+ <div [class]="containerClasses" [attr.aria-label]="name || 'Avatar'" role="img">
105
+ @if (src) {
106
+ <img
107
+ [src]="src"
108
+ [alt]="name || 'Avatar'"
109
+ class="w-full h-full object-cover"
110
+ (error)="onImageError()"
111
+ />
112
+ } @else {
113
+ <span [class]="initialsClasses">{{ initials }}</span>
114
+ }
115
+
116
+ @if (status) {
117
+ <span
118
+ [class]="statusClasses"
119
+ [attr.aria-label]="'Status: ' + status"
120
+ role="status"
121
+ ></span>
122
+ }
123
+ </div>
124
+ `,
125
+ }]
126
+ }], propDecorators: { src: [{
127
+ type: Input
128
+ }], name: [{
129
+ type: Input
130
+ }], size: [{
131
+ type: Input
132
+ }], variant: [{
133
+ type: Input
134
+ }], status: [{
135
+ type: Input
136
+ }] } });
137
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnItYXZhdGFyLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25nLWdvdmJyLXR3L3NyYy9saWIvY29tcG9uZW50cy9hdmF0YXIvYnItYXZhdGFyLmNvbXBvbmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGVBQWUsQ0FBQzs7QUErQnBFLE1BQU0sT0FBTyxpQkFBaUI7SUE1QjlCO1FBK0JXLFNBQUksR0FBdUQsUUFBUSxDQUFDO1FBQ3BFLFlBQU8sR0FBNkQsU0FBUyxDQUFDO1FBR3ZGLGdCQUFXLEdBQUcsS0FBSyxDQUFDO0tBd0VyQjtJQXRFQyxZQUFZO1FBQ1YsSUFBSSxDQUFDLFdBQVcsR0FBRyxJQUFJLENBQUM7UUFDeEIsSUFBSSxDQUFDLEdBQUcsR0FBRyxTQUFTLENBQUM7SUFDdkIsQ0FBQztJQUVELElBQUksUUFBUTtRQUNWLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSTtZQUFFLE9BQU8sR0FBRyxDQUFDO1FBQzNCLE1BQU0sS0FBSyxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzVDLElBQUksS0FBSyxDQUFDLE1BQU0sS0FBSyxDQUFDO1lBQUUsT0FBTyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQ2hFLE9BQU8sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxHQUFHLEtBQUssQ0FBQyxLQUFLLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLFdBQVcsRUFBRSxDQUFDO0lBQ2hGLENBQUM7SUFFRCxJQUFJLGdCQUFnQjtRQUNsQixNQUFNLE9BQU8sR0FBRztZQUNkLE1BQU0sRUFBRSxTQUFTO1lBQ2pCLEtBQUssRUFBRSxTQUFTO1lBQ2hCLE1BQU0sRUFBRSxXQUFXO1lBQ25CLEtBQUssRUFBRSxXQUFXO1lBQ2xCLE1BQU0sRUFBRSxXQUFXO1NBQ1gsQ0FBQztRQUVYLE1BQU0sVUFBVSxHQUFHO1lBQ2pCLE9BQU8sRUFBRSw2QkFBNkI7WUFDdEMsT0FBTyxFQUFFLHFDQUFxQztZQUM5QyxPQUFPLEVBQUUsNkJBQTZCO1lBQ3RDLE1BQU0sRUFBRSw0QkFBNEI7WUFDcEMsT0FBTyxFQUFFLHFDQUFxQztTQUN0QyxDQUFDO1FBRVgsT0FBTztZQUNMLHdIQUF3SDtZQUN4SCxPQUFPLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQztZQUNsQixJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDO1NBQ3pDLENBQUMsTUFBTSxDQUFDLE9BQU8sQ0FBQyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUM5QixDQUFDO0lBRUQsSUFBSSxlQUFlO1FBQ2pCLE1BQU0sT0FBTyxHQUFHO1lBQ2QsTUFBTSxFQUFFLGFBQWE7WUFDckIsS0FBSyxFQUFFLGVBQWU7WUFDdEIsTUFBTSxFQUFFLGVBQWU7WUFDdkIsS0FBSyxFQUFFLGVBQWU7WUFDdEIsTUFBTSxFQUFFLGdCQUFnQjtTQUNoQixDQUFDO1FBRVgsT0FBTyxPQUFPLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQzVCLENBQUM7SUFFRCxJQUFJLGFBQWE7UUFDZixNQUFNLFFBQVEsR0FBRztZQUNmLE1BQU0sRUFBRSxrQkFBa0I7WUFDMUIsT0FBTyxFQUFFLGtCQUFrQjtZQUMzQixJQUFJLEVBQUUsaUJBQWlCO1lBQ3ZCLElBQUksRUFBRSxrQkFBa0I7U0FDaEIsQ0FBQztRQUVYLE1BQU0sT0FBTyxHQUFHO1lBQ2QsTUFBTSxFQUFFLG9CQUFvQjtZQUM1QixLQUFLLEVBQUUsZ0JBQWdCO1lBQ3ZCLE1BQU0sRUFBRSxzQkFBc0I7WUFDOUIsS0FBSyxFQUFFLGtCQUFrQjtZQUN6QixNQUFNLEVBQUUsa0JBQWtCO1NBQ2xCLENBQUM7UUFFWCxPQUFPO1lBQ0wscURBQXFEO1lBQ3JELE9BQU8sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDO1lBQ2xCLFFBQVEsQ0FBQyxJQUFJLENBQUMsTUFBTyxDQUFDO1NBQ3ZCLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQ2QsQ0FBQzsrR0E5RVUsaUJBQWlCO21HQUFqQixpQkFBaUIsK0pBdkJsQjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBcUJUOzs0RkFFVSxpQkFBaUI7a0JBNUI3QixTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRSxXQUFXO29CQUNyQixVQUFVLEVBQUUsSUFBSTtvQkFDaEIsT0FBTyxFQUFFLEVBQUU7b0JBQ1gsYUFBYSxFQUFFLGlCQUFpQixDQUFDLElBQUk7b0JBQ3JDLFFBQVEsRUFBRTs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBcUJUO2lCQUNGOzhCQUVVLEdBQUc7c0JBQVgsS0FBSztnQkFDRyxJQUFJO3NCQUFaLEtBQUs7Z0JBQ0csSUFBSTtzQkFBWixLQUFLO2dCQUNHLE9BQU87c0JBQWYsS0FBSztnQkFDRyxNQUFNO3NCQUFkLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0LCBWaWV3RW5jYXBzdWxhdGlvbiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5cclxuXHJcbkBDb21wb25lbnQoe1xyXG4gIHNlbGVjdG9yOiAnYnItYXZhdGFyJyxcclxuICBzdGFuZGFsb25lOiB0cnVlLFxyXG4gIGltcG9ydHM6IFtdLFxyXG4gIGVuY2Fwc3VsYXRpb246IFZpZXdFbmNhcHN1bGF0aW9uLk5vbmUsXHJcbiAgdGVtcGxhdGU6IGBcclxuICAgIDxkaXYgW2NsYXNzXT1cImNvbnRhaW5lckNsYXNzZXNcIiBbYXR0ci5hcmlhLWxhYmVsXT1cIm5hbWUgfHwgJ0F2YXRhcidcIiByb2xlPVwiaW1nXCI+XHJcbiAgICAgIEBpZiAoc3JjKSB7XHJcbiAgICAgICAgPGltZ1xyXG4gICAgICAgICAgW3NyY109XCJzcmNcIlxyXG4gICAgICAgICAgW2FsdF09XCJuYW1lIHx8ICdBdmF0YXInXCJcclxuICAgICAgICAgIGNsYXNzPVwidy1mdWxsIGgtZnVsbCBvYmplY3QtY292ZXJcIlxyXG4gICAgICAgICAgKGVycm9yKT1cIm9uSW1hZ2VFcnJvcigpXCJcclxuICAgICAgICAvPlxyXG4gICAgICB9IEBlbHNlIHtcclxuICAgICAgICA8c3BhbiBbY2xhc3NdPVwiaW5pdGlhbHNDbGFzc2VzXCI+e3sgaW5pdGlhbHMgfX08L3NwYW4+XHJcbiAgICAgIH1cclxuXHJcbiAgICAgIEBpZiAoc3RhdHVzKSB7XHJcbiAgICAgICAgPHNwYW5cclxuICAgICAgICAgIFtjbGFzc109XCJzdGF0dXNDbGFzc2VzXCJcclxuICAgICAgICAgIFthdHRyLmFyaWEtbGFiZWxdPVwiJ1N0YXR1czogJyArIHN0YXR1c1wiXHJcbiAgICAgICAgICByb2xlPVwic3RhdHVzXCJcclxuICAgICAgICA+PC9zcGFuPlxyXG4gICAgICB9XHJcbiAgICA8L2Rpdj5cclxuICBgLFxyXG59KVxyXG5leHBvcnQgY2xhc3MgQnJBdmF0YXJDb21wb25lbnQge1xyXG4gIEBJbnB1dCgpIHNyYz86IHN0cmluZztcclxuICBASW5wdXQoKSBuYW1lPzogc3RyaW5nO1xyXG4gIEBJbnB1dCgpIHNpemU6ICd4c21hbGwnIHwgJ3NtYWxsJyB8ICdtZWRpdW0nIHwgJ2xhcmdlJyB8ICd4bGFyZ2UnID0gJ21lZGl1bSc7XHJcbiAgQElucHV0KCkgdmFyaWFudDogJ3ByaW1hcnknIHwgJ25ldXRyYWwnIHwgJ3N1Y2Nlc3MnIHwgJ2RhbmdlcicgfCAnd2FybmluZycgPSAncHJpbWFyeSc7XHJcbiAgQElucHV0KCkgc3RhdHVzPzogJ29ubGluZScgfCAnb2ZmbGluZScgfCAnYnVzeScgfCAnYXdheSc7XHJcblxyXG4gIGltYWdlRmFpbGVkID0gZmFsc2U7XHJcblxyXG4gIG9uSW1hZ2VFcnJvcigpOiB2b2lkIHtcclxuICAgIHRoaXMuaW1hZ2VGYWlsZWQgPSB0cnVlO1xyXG4gICAgdGhpcy5zcmMgPSB1bmRlZmluZWQ7XHJcbiAgfVxyXG5cclxuICBnZXQgaW5pdGlhbHMoKTogc3RyaW5nIHtcclxuICAgIGlmICghdGhpcy5uYW1lKSByZXR1cm4gJz8nO1xyXG4gICAgY29uc3QgcGFydHMgPSB0aGlzLm5hbWUudHJpbSgpLnNwbGl0KC9cXHMrLyk7XHJcbiAgICBpZiAocGFydHMubGVuZ3RoID09PSAxKSByZXR1cm4gcGFydHNbMF0uY2hhckF0KDApLnRvVXBwZXJDYXNlKCk7XHJcbiAgICByZXR1cm4gKHBhcnRzWzBdLmNoYXJBdCgwKSArIHBhcnRzW3BhcnRzLmxlbmd0aCAtIDFdLmNoYXJBdCgwKSkudG9VcHBlckNhc2UoKTtcclxuICB9XHJcblxyXG4gIGdldCBjb250YWluZXJDbGFzc2VzKCk6IHN0cmluZyB7XHJcbiAgICBjb25zdCBzaXplTWFwID0ge1xyXG4gICAgICB4c21hbGw6ICd3LTYgaC02JyxcclxuICAgICAgc21hbGw6ICd3LTggaC04JyxcclxuICAgICAgbWVkaXVtOiAndy0xMCBoLTEwJyxcclxuICAgICAgbGFyZ2U6ICd3LTE0IGgtMTQnLFxyXG4gICAgICB4bGFyZ2U6ICd3LTIwIGgtMjAnLFxyXG4gICAgfSBhcyBjb25zdDtcclxuXHJcbiAgICBjb25zdCB2YXJpYW50TWFwID0ge1xyXG4gICAgICBwcmltYXJ5OiAnYmctZ292YnItcHJpbWFyeSB0ZXh0LXdoaXRlJyxcclxuICAgICAgbmV1dHJhbDogJ2JnLWdvdmJyLWdyYXktMjAgdGV4dC1nb3Zici1ncmF5LTgwJyxcclxuICAgICAgc3VjY2VzczogJ2JnLWdvdmJyLXN1Y2Nlc3MgdGV4dC13aGl0ZScsXHJcbiAgICAgIGRhbmdlcjogJ2JnLWdvdmJyLWRhbmdlciB0ZXh0LXdoaXRlJyxcclxuICAgICAgd2FybmluZzogJ2JnLWdvdmJyLXdhcm5pbmcgdGV4dC1nb3Zici1ncmF5LTgwJyxcclxuICAgIH0gYXMgY29uc3Q7XHJcblxyXG4gICAgcmV0dXJuIFtcclxuICAgICAgJ3JlbGF0aXZlIGlubGluZS1mbGV4IGl0ZW1zLWNlbnRlciBqdXN0aWZ5LWNlbnRlciByb3VuZGVkLWZ1bGwgb3ZlcmZsb3ctaGlkZGVuIGZsZXgtc2hyaW5rLTAgZm9udC1yYXdsaW5lIGZvbnQtc2VtaWJvbGQnLFxyXG4gICAgICBzaXplTWFwW3RoaXMuc2l6ZV0sXHJcbiAgICAgIHRoaXMuc3JjID8gJycgOiB2YXJpYW50TWFwW3RoaXMudmFyaWFudF0sXHJcbiAgICBdLmZpbHRlcihCb29sZWFuKS5qb2luKCcgJyk7XHJcbiAgfVxyXG5cclxuICBnZXQgaW5pdGlhbHNDbGFzc2VzKCk6IHN0cmluZyB7XHJcbiAgICBjb25zdCBmb250TWFwID0ge1xyXG4gICAgICB4c21hbGw6ICd0ZXh0LVsxMHB4XScsXHJcbiAgICAgIHNtYWxsOiAndGV4dC1nb3Zici14cycsXHJcbiAgICAgIG1lZGl1bTogJ3RleHQtZ292YnItc20nLFxyXG4gICAgICBsYXJnZTogJ3RleHQtZ292YnItbGcnLFxyXG4gICAgICB4bGFyZ2U6ICd0ZXh0LWdvdmJyLTJ4bCcsXHJcbiAgICB9IGFzIGNvbnN0O1xyXG5cclxuICAgIHJldHVybiBmb250TWFwW3RoaXMuc2l6ZV07XHJcbiAgfVxyXG5cclxuICBnZXQgc3RhdHVzQ2xhc3NlcygpOiBzdHJpbmcge1xyXG4gICAgY29uc3QgY29sb3JNYXAgPSB7XHJcbiAgICAgIG9ubGluZTogJ2JnLWdvdmJyLXN1Y2Nlc3MnLFxyXG4gICAgICBvZmZsaW5lOiAnYmctZ292YnItZ3JheS00MCcsXHJcbiAgICAgIGJ1c3k6ICdiZy1nb3Zici1kYW5nZXInLFxyXG4gICAgICBhd2F5OiAnYmctZ292YnItd2FybmluZycsXHJcbiAgICB9IGFzIGNvbnN0O1xyXG5cclxuICAgIGNvbnN0IGRvdFNpemUgPSB7XHJcbiAgICAgIHhzbWFsbDogJ3ctMS41IGgtMS41IGJvcmRlcicsXHJcbiAgICAgIHNtYWxsOiAndy0yIGgtMiBib3JkZXInLFxyXG4gICAgICBtZWRpdW06ICd3LTIuNSBoLTIuNSBib3JkZXItMicsXHJcbiAgICAgIGxhcmdlOiAndy0zIGgtMyBib3JkZXItMicsXHJcbiAgICAgIHhsYXJnZTogJ3ctNCBoLTQgYm9yZGVyLTInLFxyXG4gICAgfSBhcyBjb25zdDtcclxuXHJcbiAgICByZXR1cm4gW1xyXG4gICAgICAnYWJzb2x1dGUgYm90dG9tLTAgcmlnaHQtMCByb3VuZGVkLWZ1bGwgYm9yZGVyLXdoaXRlJyxcclxuICAgICAgZG90U2l6ZVt0aGlzLnNpemVdLFxyXG4gICAgICBjb2xvck1hcFt0aGlzLnN0YXR1cyFdLFxyXG4gICAgXS5qb2luKCcgJyk7XHJcbiAgfVxyXG59XHJcbiJdfQ==
@@ -5,6 +5,7 @@ export class BrModalComponent {
5
5
  this.isOpen = false;
6
6
  this.title = '';
7
7
  this.size = 'medium';
8
+ this.variant = 'default';
8
9
  this.showCloseButton = true;
9
10
  this.showFooter = true;
10
11
  this.showCancelButton = true;
@@ -22,6 +23,44 @@ export class BrModalComponent {
22
23
  this.titleId = `br-modal-title-${BrModalComponent.nextId++}`;
23
24
  }
24
25
  static { this.nextId = 0; }
26
+ get variantIcon() {
27
+ const icons = {
28
+ default: null,
29
+ danger: 'fa-solid fa-triangle-exclamation',
30
+ warning: 'fa-solid fa-circle-exclamation',
31
+ success: 'fa-solid fa-circle-check',
32
+ };
33
+ return icons[this.variant];
34
+ }
35
+ get variantIconColor() {
36
+ const colors = {
37
+ default: '',
38
+ danger: 'text-govbr-danger',
39
+ warning: 'text-govbr-warning',
40
+ success: 'text-govbr-success',
41
+ };
42
+ return colors[this.variant];
43
+ }
44
+ get headerClasses() {
45
+ const base = 'flex items-center justify-between px-6 py-4 border-b';
46
+ const borderColors = {
47
+ default: 'border-govbr-gray-10',
48
+ danger: 'border-govbr-danger/20',
49
+ warning: 'border-govbr-warning/30',
50
+ success: 'border-govbr-success/20',
51
+ };
52
+ return `${base} ${borderColors[this.variant]}`;
53
+ }
54
+ get primaryButtonClasses() {
55
+ const base = 'px-5 py-2 text-govbr-base font-semibold font-rawline rounded-govbr-pill transition-colors govbr-focus-ring disabled:opacity-50 disabled:cursor-not-allowed';
56
+ const variantStyles = {
57
+ default: 'text-white bg-govbr-primary hover:bg-govbr-primary-dark',
58
+ danger: 'text-white bg-govbr-danger hover:bg-red-700',
59
+ warning: 'text-govbr-gray-80 bg-govbr-warning hover:bg-yellow-500',
60
+ success: 'text-white bg-govbr-success hover:bg-green-700',
61
+ };
62
+ return `${base} ${variantStyles[this.variant]}`;
63
+ }
25
64
  get modalClasses() {
26
65
  const sizes = {
27
66
  xsmall: 'max-w-[220px]',
@@ -57,7 +96,7 @@ export class BrModalComponent {
57
96
  this.canceled.emit();
58
97
  }
59
98
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrModalComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
60
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: BrModalComponent, isStandalone: true, selector: "br-modal", inputs: { isOpen: "isOpen", title: "title", size: "size", showCloseButton: "showCloseButton", showFooter: "showFooter", showCancelButton: "showCancelButton", showPrimaryButton: "showPrimaryButton", primaryText: "primaryText", cancelText: "cancelText", primaryDisabled: "primaryDisabled", loading: "loading", closeOnScrim: "closeOnScrim", closeOnEscape: "closeOnEscape", hasCustomFooter: "hasCustomFooter" }, outputs: { closed: "closed", confirmed: "confirmed", canceled: "canceled" }, host: { listeners: { "document:keydown.escape": "onEscapeKey()" } }, ngImport: i0, template: `
99
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: BrModalComponent, isStandalone: true, selector: "br-modal", inputs: { isOpen: "isOpen", title: "title", size: "size", variant: "variant", showCloseButton: "showCloseButton", showFooter: "showFooter", showCancelButton: "showCancelButton", showPrimaryButton: "showPrimaryButton", primaryText: "primaryText", cancelText: "cancelText", primaryDisabled: "primaryDisabled", loading: "loading", closeOnScrim: "closeOnScrim", closeOnEscape: "closeOnEscape", hasCustomFooter: "hasCustomFooter" }, outputs: { closed: "closed", confirmed: "confirmed", canceled: "canceled" }, host: { listeners: { "document:keydown.escape": "onEscapeKey()" } }, ngImport: i0, template: `
61
100
  <!-- Scrim / Overlay -->
62
101
  @if (isOpen) {
63
102
  <div
@@ -81,13 +120,18 @@ export class BrModalComponent {
81
120
  role="document"
82
121
  >
83
122
  <!-- Header -->
84
- <div class="flex items-center justify-between px-6 py-4 border-b border-govbr-gray-10">
85
- <h4
86
- [id]="titleId"
87
- class="text-govbr-lg font-semibold text-govbr-gray-80 font-rawline truncate pr-4"
88
- >
89
- {{ title }}
90
- </h4>
123
+ <div [class]="headerClasses">
124
+ <div class="flex items-center gap-3 min-w-0 flex-1">
125
+ @if (variantIcon) {
126
+ <i [class]="variantIcon + ' text-xl flex-shrink-0 ' + variantIconColor"></i>
127
+ }
128
+ <h4
129
+ [id]="titleId"
130
+ class="text-govbr-lg font-semibold text-govbr-gray-80 font-rawline truncate"
131
+ >
132
+ {{ title }}
133
+ </h4>
134
+ </div>
91
135
  @if (showCloseButton) {
92
136
  <button
93
137
  type="button"
@@ -131,7 +175,7 @@ export class BrModalComponent {
131
175
  <button
132
176
  type="button"
133
177
  [disabled]="primaryDisabled"
134
- class="px-5 py-2 text-govbr-base font-semibold font-rawline text-white bg-govbr-primary rounded-govbr-pill hover:bg-govbr-primary-dark transition-colors govbr-focus-ring disabled:opacity-50 disabled:cursor-not-allowed"
178
+ [class]="primaryButtonClasses"
135
179
  (click)="confirm()"
136
180
  >
137
181
  {{ primaryText }}
@@ -176,13 +220,18 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
176
220
  role="document"
177
221
  >
178
222
  <!-- Header -->
179
- <div class="flex items-center justify-between px-6 py-4 border-b border-govbr-gray-10">
180
- <h4
181
- [id]="titleId"
182
- class="text-govbr-lg font-semibold text-govbr-gray-80 font-rawline truncate pr-4"
183
- >
184
- {{ title }}
185
- </h4>
223
+ <div [class]="headerClasses">
224
+ <div class="flex items-center gap-3 min-w-0 flex-1">
225
+ @if (variantIcon) {
226
+ <i [class]="variantIcon + ' text-xl flex-shrink-0 ' + variantIconColor"></i>
227
+ }
228
+ <h4
229
+ [id]="titleId"
230
+ class="text-govbr-lg font-semibold text-govbr-gray-80 font-rawline truncate"
231
+ >
232
+ {{ title }}
233
+ </h4>
234
+ </div>
186
235
  @if (showCloseButton) {
187
236
  <button
188
237
  type="button"
@@ -226,7 +275,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
226
275
  <button
227
276
  type="button"
228
277
  [disabled]="primaryDisabled"
229
- class="px-5 py-2 text-govbr-base font-semibold font-rawline text-white bg-govbr-primary rounded-govbr-pill hover:bg-govbr-primary-dark transition-colors govbr-focus-ring disabled:opacity-50 disabled:cursor-not-allowed"
278
+ [class]="primaryButtonClasses"
230
279
  (click)="confirm()"
231
280
  >
232
281
  {{ primaryText }}
@@ -246,6 +295,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
246
295
  type: Input
247
296
  }], size: [{
248
297
  type: Input
298
+ }], variant: [{
299
+ type: Input
249
300
  }], showCloseButton: [{
250
301
  type: Input
251
302
  }], showFooter: [{
@@ -278,4 +329,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
278
329
  type: HostListener,
279
330
  args: ['document:keydown.escape']
280
331
  }] } });
281
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"br-modal.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-govbr-tw/src/lib/components/modal/br-modal.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  HostListener,\n  Input,\n  Output,\n  ViewEncapsulation,\n} from '@angular/core';\n\n\n@Component({\n  selector: 'br-modal',\n  standalone: true,\n  imports: [],\n  encapsulation: ViewEncapsulation.None,\n  template: `\n    <!-- Scrim / Overlay -->\n    @if (isOpen) {\n      <div\n        class=\"fixed inset-0 z-[1000] flex items-center justify-center\"\n        role=\"dialog\"\n        [attr.aria-modal]=\"true\"\n        [attr.aria-labelledby]=\"titleId\"\n        >\n        <!-- Backdrop -->\n        <div\n          class=\"absolute inset-0 bg-govbr-primary-darkest/60 transition-opacity\"\n          (click)=\"onScrimClick()\"\n          (keydown.enter)=\"onScrimClick()\"\n          role=\"button\"\n          tabindex=\"0\"\n          aria-label=\"Fechar modal\"\n        ></div>\n        <!-- Modal container -->\n        <div\n          [class]=\"modalClasses\"\n          role=\"document\"\n          >\n          <!-- Header -->\n          <div class=\"flex items-center justify-between px-6 py-4 border-b border-govbr-gray-10\">\n            <h4\n              [id]=\"titleId\"\n              class=\"text-govbr-lg font-semibold text-govbr-gray-80 font-rawline truncate pr-4\"\n              >\n              {{ title }}\n            </h4>\n            @if (showCloseButton) {\n              <button\n                type=\"button\"\n                class=\"flex-shrink-0 w-8 h-8 rounded-full flex items-center justify-center text-govbr-primary hover:bg-govbr-primary-lightest govbr-focus-ring transition-colors\"\n                aria-label=\"Fechar\"\n                (click)=\"close()\"\n                >\n                <i class=\"fa-solid fa-xmark\"></i>\n              </button>\n            }\n          </div>\n          <!-- Body -->\n          <div class=\"flex-1 overflow-y-auto px-6 py-4 font-rawline text-govbr-base text-govbr-gray-80\">\n            <!-- Loading state -->\n            @if (loading) {\n              <div class=\"flex items-center justify-center py-10\">\n                <i class=\"fa-solid fa-spinner fa-spin text-govbr-primary text-2xl\"></i>\n              </div>\n            }\n            @if (!loading) {\n              <ng-content></ng-content>\n            }\n          </div>\n          <!-- Footer -->\n          @if (!loading && showFooter) {\n            <div\n              class=\"flex flex-wrap items-center justify-end gap-3 px-6 py-4 border-t border-govbr-gray-10\"\n              >\n              <ng-content select=\"[modal-footer]\"></ng-content>\n              @if (!hasCustomFooter) {\n                @if (showCancelButton) {\n                  <button\n                    type=\"button\"\n                    class=\"px-5 py-2 text-govbr-base font-semibold font-rawline text-govbr-primary bg-transparent border border-govbr-primary rounded-govbr-pill hover:bg-govbr-primary hover:text-white transition-colors govbr-focus-ring\"\n                    (click)=\"cancel()\"\n                    >\n                    {{ cancelText }}\n                  </button>\n                }\n                @if (showPrimaryButton) {\n                  <button\n                    type=\"button\"\n                    [disabled]=\"primaryDisabled\"\n                    class=\"px-5 py-2 text-govbr-base font-semibold font-rawline text-white bg-govbr-primary rounded-govbr-pill hover:bg-govbr-primary-dark transition-colors govbr-focus-ring disabled:opacity-50 disabled:cursor-not-allowed\"\n                    (click)=\"confirm()\"\n                    >\n                    {{ primaryText }}\n                  </button>\n                }\n              }\n            </div>\n          }\n        </div>\n      </div>\n    }\n    `,\n})\nexport class BrModalComponent {\n  @Input() isOpen = false;\n  @Input() title = '';\n  @Input() size: 'xsmall' | 'small' | 'medium' | 'large' | 'auto' = 'medium';\n  @Input() showCloseButton = true;\n  @Input() showFooter = true;\n  @Input() showCancelButton = true;\n  @Input() showPrimaryButton = true;\n  @Input() primaryText = 'Confirmar';\n  @Input() cancelText = 'Cancelar';\n  @Input() primaryDisabled = false;\n  @Input() loading = false;\n  @Input() closeOnScrim = true;\n  @Input() closeOnEscape = true;\n  @Input() hasCustomFooter = false;\n\n  @Output() closed = new EventEmitter<void>();\n  @Output() confirmed = new EventEmitter<void>();\n  @Output() canceled = new EventEmitter<void>();\n\n  private static nextId = 0;\n  titleId = `br-modal-title-${BrModalComponent.nextId++}`;\n\n  get modalClasses(): string {\n    const sizes = {\n      xsmall: 'max-w-[220px]',\n      small: 'max-w-[300px]',\n      medium: 'max-w-[500px]',\n      large: 'max-w-[640px]',\n      auto: 'max-w-fit',\n    } as const;\n\n    const base = [\n      'relative flex flex-col bg-white shadow-lg z-10 w-full mx-4',\n      'max-h-[90vh]',\n      'animate-modal-in',\n    ].join(' ');\n\n    return [base, sizes[this.size]].filter(Boolean).join(' ');\n  }\n\n  @HostListener('document:keydown.escape')\n  onEscapeKey(): void {\n    if (this.isOpen && this.closeOnEscape) {\n      this.close();\n    }\n  }\n\n  onScrimClick(): void {\n    if (this.closeOnScrim) {\n      this.close();\n    }\n  }\n\n  close(): void {\n    this.closed.emit();\n  }\n\n  confirm(): void {\n    this.confirmed.emit();\n  }\n\n  cancel(): void {\n    this.canceled.emit();\n  }\n}\n"]}
332
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"br-modal.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-govbr-tw/src/lib/components/modal/br-modal.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  HostListener,\n  Input,\n  Output,\n  ViewEncapsulation,\n} from '@angular/core';\n\n\n@Component({\n  selector: 'br-modal',\n  standalone: true,\n  imports: [],\n  encapsulation: ViewEncapsulation.None,\n  template: `\n    <!-- Scrim / Overlay -->\n    @if (isOpen) {\n      <div\n        class=\"fixed inset-0 z-[1000] flex items-center justify-center\"\n        role=\"dialog\"\n        [attr.aria-modal]=\"true\"\n        [attr.aria-labelledby]=\"titleId\"\n        >\n        <!-- Backdrop -->\n        <div\n          class=\"absolute inset-0 bg-govbr-primary-darkest/60 transition-opacity\"\n          (click)=\"onScrimClick()\"\n          (keydown.enter)=\"onScrimClick()\"\n          role=\"button\"\n          tabindex=\"0\"\n          aria-label=\"Fechar modal\"\n        ></div>\n        <!-- Modal container -->\n        <div\n          [class]=\"modalClasses\"\n          role=\"document\"\n          >\n          <!-- Header -->\n          <div [class]=\"headerClasses\">\n            <div class=\"flex items-center gap-3 min-w-0 flex-1\">\n              @if (variantIcon) {\n                <i [class]=\"variantIcon + ' text-xl flex-shrink-0 ' + variantIconColor\"></i>\n              }\n              <h4\n                [id]=\"titleId\"\n                class=\"text-govbr-lg font-semibold text-govbr-gray-80 font-rawline truncate\"\n                >\n                {{ title }}\n              </h4>\n            </div>\n            @if (showCloseButton) {\n              <button\n                type=\"button\"\n                class=\"flex-shrink-0 w-8 h-8 rounded-full flex items-center justify-center text-govbr-primary hover:bg-govbr-primary-lightest govbr-focus-ring transition-colors\"\n                aria-label=\"Fechar\"\n                (click)=\"close()\"\n                >\n                <i class=\"fa-solid fa-xmark\"></i>\n              </button>\n            }\n          </div>\n          <!-- Body -->\n          <div class=\"flex-1 overflow-y-auto px-6 py-4 font-rawline text-govbr-base text-govbr-gray-80\">\n            <!-- Loading state -->\n            @if (loading) {\n              <div class=\"flex items-center justify-center py-10\">\n                <i class=\"fa-solid fa-spinner fa-spin text-govbr-primary text-2xl\"></i>\n              </div>\n            }\n            @if (!loading) {\n              <ng-content></ng-content>\n            }\n          </div>\n          <!-- Footer -->\n          @if (!loading && showFooter) {\n            <div\n              class=\"flex flex-wrap items-center justify-end gap-3 px-6 py-4 border-t border-govbr-gray-10\"\n              >\n              <ng-content select=\"[modal-footer]\"></ng-content>\n              @if (!hasCustomFooter) {\n                @if (showCancelButton) {\n                  <button\n                    type=\"button\"\n                    class=\"px-5 py-2 text-govbr-base font-semibold font-rawline text-govbr-primary bg-transparent border border-govbr-primary rounded-govbr-pill hover:bg-govbr-primary hover:text-white transition-colors govbr-focus-ring\"\n                    (click)=\"cancel()\"\n                    >\n                    {{ cancelText }}\n                  </button>\n                }\n                @if (showPrimaryButton) {\n                  <button\n                    type=\"button\"\n                    [disabled]=\"primaryDisabled\"\n                    [class]=\"primaryButtonClasses\"\n                    (click)=\"confirm()\"\n                    >\n                    {{ primaryText }}\n                  </button>\n                }\n              }\n            </div>\n          }\n        </div>\n      </div>\n    }\n    `,\n})\nexport class BrModalComponent {\n  @Input() isOpen = false;\n  @Input() title = '';\n  @Input() size: 'xsmall' | 'small' | 'medium' | 'large' | 'auto' = 'medium';\n  @Input() variant: 'default' | 'danger' | 'warning' | 'success' = 'default';\n  @Input() showCloseButton = true;\n  @Input() showFooter = true;\n  @Input() showCancelButton = true;\n  @Input() showPrimaryButton = true;\n  @Input() primaryText = 'Confirmar';\n  @Input() cancelText = 'Cancelar';\n  @Input() primaryDisabled = false;\n  @Input() loading = false;\n  @Input() closeOnScrim = true;\n  @Input() closeOnEscape = true;\n  @Input() hasCustomFooter = false;\n\n  @Output() closed = new EventEmitter<void>();\n  @Output() confirmed = new EventEmitter<void>();\n  @Output() canceled = new EventEmitter<void>();\n\n  private static nextId = 0;\n  titleId = `br-modal-title-${BrModalComponent.nextId++}`;\n\n  get variantIcon(): string | null {\n    const icons = {\n      default: null,\n      danger: 'fa-solid fa-triangle-exclamation',\n      warning: 'fa-solid fa-circle-exclamation',\n      success: 'fa-solid fa-circle-check',\n    } as const;\n    return icons[this.variant];\n  }\n\n  get variantIconColor(): string {\n    const colors = {\n      default: '',\n      danger: 'text-govbr-danger',\n      warning: 'text-govbr-warning',\n      success: 'text-govbr-success',\n    } as const;\n    return colors[this.variant];\n  }\n\n  get headerClasses(): string {\n    const base = 'flex items-center justify-between px-6 py-4 border-b';\n    const borderColors = {\n      default: 'border-govbr-gray-10',\n      danger: 'border-govbr-danger/20',\n      warning: 'border-govbr-warning/30',\n      success: 'border-govbr-success/20',\n    } as const;\n    return `${base} ${borderColors[this.variant]}`;\n  }\n\n  get primaryButtonClasses(): string {\n    const base = 'px-5 py-2 text-govbr-base font-semibold font-rawline rounded-govbr-pill transition-colors govbr-focus-ring disabled:opacity-50 disabled:cursor-not-allowed';\n    const variantStyles = {\n      default: 'text-white bg-govbr-primary hover:bg-govbr-primary-dark',\n      danger: 'text-white bg-govbr-danger hover:bg-red-700',\n      warning: 'text-govbr-gray-80 bg-govbr-warning hover:bg-yellow-500',\n      success: 'text-white bg-govbr-success hover:bg-green-700',\n    } as const;\n    return `${base} ${variantStyles[this.variant]}`;\n  }\n\n  get modalClasses(): string {\n    const sizes = {\n      xsmall: 'max-w-[220px]',\n      small: 'max-w-[300px]',\n      medium: 'max-w-[500px]',\n      large: 'max-w-[640px]',\n      auto: 'max-w-fit',\n    } as const;\n\n    const base = [\n      'relative flex flex-col bg-white shadow-lg z-10 w-full mx-4',\n      'max-h-[90vh]',\n      'animate-modal-in',\n    ].join(' ');\n\n    return [base, sizes[this.size]].filter(Boolean).join(' ');\n  }\n\n  @HostListener('document:keydown.escape')\n  onEscapeKey(): void {\n    if (this.isOpen && this.closeOnEscape) {\n      this.close();\n    }\n  }\n\n  onScrimClick(): void {\n    if (this.closeOnScrim) {\n      this.close();\n    }\n  }\n\n  close(): void {\n    this.closed.emit();\n  }\n\n  confirm(): void {\n    this.confirmed.emit();\n  }\n\n  cancel(): void {\n    this.canceled.emit();\n  }\n}\n"]}
@@ -0,0 +1,156 @@
1
+ import { Component, Input, ViewEncapsulation } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class BrSkeletonComponent {
4
+ constructor() {
5
+ this.variant = 'line';
6
+ this.rows = 3;
7
+ this.cols = 4;
8
+ }
9
+ get lineClasses() {
10
+ const w = this.width || 'w-full';
11
+ const h = this.height || 'h-4';
12
+ return `br-skeleton-pulse bg-govbr-gray-10 rounded ${w} ${h}`;
13
+ }
14
+ get circleClasses() {
15
+ const size = this.width || 'w-10 h-10';
16
+ return `br-skeleton-pulse bg-govbr-gray-10 rounded-full ${size}`;
17
+ }
18
+ get tableRows() {
19
+ return Array.from({ length: this.rows }, (_, i) => i);
20
+ }
21
+ get tableCols() {
22
+ return Array.from({ length: this.cols }, (_, i) => i);
23
+ }
24
+ get listRows() {
25
+ return Array.from({ length: this.rows }, (_, i) => i);
26
+ }
27
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrSkeletonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
28
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: BrSkeletonComponent, isStandalone: true, selector: "br-skeleton", inputs: { variant: "variant", width: "width", height: "height", rows: "rows", cols: "cols" }, ngImport: i0, template: `
29
+ @switch (variant) {
30
+ @case ('circle') {
31
+ <div [class]="circleClasses" role="status" aria-label="Carregando">
32
+ <span class="sr-only">Carregando...</span>
33
+ </div>
34
+ }
35
+ @case ('card') {
36
+ <div class="rounded-md overflow-hidden border border-govbr-gray-10" role="status" aria-label="Carregando">
37
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-36"></div>
38
+ <div class="p-4 space-y-3">
39
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-4 rounded w-3/4"></div>
40
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-3 rounded w-full"></div>
41
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-3 rounded w-5/6"></div>
42
+ </div>
43
+ <span class="sr-only">Carregando...</span>
44
+ </div>
45
+ }
46
+ @case ('table') {
47
+ <div class="border border-govbr-gray-10 rounded-md overflow-hidden" role="status" aria-label="Carregando">
48
+ <div class="bg-govbr-gray-5 px-4 py-3 flex gap-4">
49
+ @for (c of tableCols; track c) {
50
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-3 rounded flex-1"></div>
51
+ }
52
+ </div>
53
+ @for (r of tableRows; track r) {
54
+ <div class="px-4 py-3 border-t border-govbr-gray-5 flex gap-4">
55
+ @for (c of tableCols; track c) {
56
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-3 rounded flex-1"></div>
57
+ }
58
+ </div>
59
+ }
60
+ <span class="sr-only">Carregando...</span>
61
+ </div>
62
+ }
63
+ @case ('list') {
64
+ <div class="space-y-3" role="status" aria-label="Carregando">
65
+ @for (r of listRows; track r) {
66
+ <div class="flex items-center gap-3">
67
+ <div class="br-skeleton-pulse bg-govbr-gray-10 w-10 h-10 rounded-full flex-shrink-0"></div>
68
+ <div class="flex-1 space-y-2">
69
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-3.5 rounded w-2/3"></div>
70
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-2.5 rounded w-1/2"></div>
71
+ </div>
72
+ </div>
73
+ }
74
+ <span class="sr-only">Carregando...</span>
75
+ </div>
76
+ }
77
+ @default {
78
+ <!-- line variant -->
79
+ <div [class]="lineClasses" role="status" aria-label="Carregando">
80
+ <span class="sr-only">Carregando...</span>
81
+ </div>
82
+ }
83
+ }
84
+ `, isInline: true, styles: ["@keyframes br-skeleton-pulse{0%,to{opacity:1}50%{opacity:.4}}.br-skeleton-pulse{animation:br-skeleton-pulse 1.5s ease-in-out infinite}\n"], encapsulation: i0.ViewEncapsulation.None }); }
85
+ }
86
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrSkeletonComponent, decorators: [{
87
+ type: Component,
88
+ args: [{ selector: 'br-skeleton', standalone: true, imports: [], encapsulation: ViewEncapsulation.None, template: `
89
+ @switch (variant) {
90
+ @case ('circle') {
91
+ <div [class]="circleClasses" role="status" aria-label="Carregando">
92
+ <span class="sr-only">Carregando...</span>
93
+ </div>
94
+ }
95
+ @case ('card') {
96
+ <div class="rounded-md overflow-hidden border border-govbr-gray-10" role="status" aria-label="Carregando">
97
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-36"></div>
98
+ <div class="p-4 space-y-3">
99
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-4 rounded w-3/4"></div>
100
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-3 rounded w-full"></div>
101
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-3 rounded w-5/6"></div>
102
+ </div>
103
+ <span class="sr-only">Carregando...</span>
104
+ </div>
105
+ }
106
+ @case ('table') {
107
+ <div class="border border-govbr-gray-10 rounded-md overflow-hidden" role="status" aria-label="Carregando">
108
+ <div class="bg-govbr-gray-5 px-4 py-3 flex gap-4">
109
+ @for (c of tableCols; track c) {
110
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-3 rounded flex-1"></div>
111
+ }
112
+ </div>
113
+ @for (r of tableRows; track r) {
114
+ <div class="px-4 py-3 border-t border-govbr-gray-5 flex gap-4">
115
+ @for (c of tableCols; track c) {
116
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-3 rounded flex-1"></div>
117
+ }
118
+ </div>
119
+ }
120
+ <span class="sr-only">Carregando...</span>
121
+ </div>
122
+ }
123
+ @case ('list') {
124
+ <div class="space-y-3" role="status" aria-label="Carregando">
125
+ @for (r of listRows; track r) {
126
+ <div class="flex items-center gap-3">
127
+ <div class="br-skeleton-pulse bg-govbr-gray-10 w-10 h-10 rounded-full flex-shrink-0"></div>
128
+ <div class="flex-1 space-y-2">
129
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-3.5 rounded w-2/3"></div>
130
+ <div class="br-skeleton-pulse bg-govbr-gray-10 h-2.5 rounded w-1/2"></div>
131
+ </div>
132
+ </div>
133
+ }
134
+ <span class="sr-only">Carregando...</span>
135
+ </div>
136
+ }
137
+ @default {
138
+ <!-- line variant -->
139
+ <div [class]="lineClasses" role="status" aria-label="Carregando">
140
+ <span class="sr-only">Carregando...</span>
141
+ </div>
142
+ }
143
+ }
144
+ `, styles: ["@keyframes br-skeleton-pulse{0%,to{opacity:1}50%{opacity:.4}}.br-skeleton-pulse{animation:br-skeleton-pulse 1.5s ease-in-out infinite}\n"] }]
145
+ }], propDecorators: { variant: [{
146
+ type: Input
147
+ }], width: [{
148
+ type: Input
149
+ }], height: [{
150
+ type: Input
151
+ }], rows: [{
152
+ type: Input
153
+ }], cols: [{
154
+ type: Input
155
+ }] } });
156
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnItc2tlbGV0b24uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmctZ292YnItdHcvc3JjL2xpYi9jb21wb25lbnRzL3NrZWxldG9uL2JyLXNrZWxldG9uLmNvbXBvbmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGVBQWUsQ0FBQzs7QUEyRXBFLE1BQU0sT0FBTyxtQkFBbUI7SUF4RWhDO1FBeUVXLFlBQU8sR0FBa0QsTUFBTSxDQUFDO1FBR2hFLFNBQUksR0FBRyxDQUFDLENBQUM7UUFDVCxTQUFJLEdBQUcsQ0FBQyxDQUFDO0tBd0JuQjtJQXRCQyxJQUFJLFdBQVc7UUFDYixNQUFNLENBQUMsR0FBRyxJQUFJLENBQUMsS0FBSyxJQUFJLFFBQVEsQ0FBQztRQUNqQyxNQUFNLENBQUMsR0FBRyxJQUFJLENBQUMsTUFBTSxJQUFJLEtBQUssQ0FBQztRQUMvQixPQUFPLDhDQUE4QyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUM7SUFDaEUsQ0FBQztJQUVELElBQUksYUFBYTtRQUNmLE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxLQUFLLElBQUksV0FBVyxDQUFDO1FBQ3ZDLE9BQU8sbURBQW1ELElBQUksRUFBRSxDQUFDO0lBQ25FLENBQUM7SUFFRCxJQUFJLFNBQVM7UUFDWCxPQUFPLEtBQUssQ0FBQyxJQUFJLENBQUMsRUFBRSxNQUFNLEVBQUUsSUFBSSxDQUFDLElBQUksRUFBRSxFQUFFLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDeEQsQ0FBQztJQUVELElBQUksU0FBUztRQUNYLE9BQU8sS0FBSyxDQUFDLElBQUksQ0FBQyxFQUFFLE1BQU0sRUFBRSxJQUFJLENBQUMsSUFBSSxFQUFFLEVBQUUsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUN4RCxDQUFDO0lBRUQsSUFBSSxRQUFRO1FBQ1YsT0FBTyxLQUFLLENBQUMsSUFBSSxDQUFDLEVBQUUsTUFBTSxFQUFFLElBQUksQ0FBQyxJQUFJLEVBQUUsRUFBRSxDQUFDLENBQUMsRUFBRSxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ3hELENBQUM7K0dBNUJVLG1CQUFtQjttR0FBbkIsbUJBQW1CLHFLQW5FcEI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBd0RUOzs0RkFXVSxtQkFBbUI7a0JBeEUvQixTQUFTOytCQUNFLGFBQWEsY0FDWCxJQUFJLFdBQ1AsRUFBRSxpQkFDSSxpQkFBaUIsQ0FBQyxJQUFJLFlBQzNCOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXdEVDs4QkFZUSxPQUFPO3NCQUFmLEtBQUs7Z0JBQ0csS0FBSztzQkFBYixLQUFLO2dCQUNHLE1BQU07c0JBQWQsS0FBSztnQkFDRyxJQUFJO3NCQUFaLEtBQUs7Z0JBQ0csSUFBSTtzQkFBWixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBJbnB1dCwgVmlld0VuY2Fwc3VsYXRpb24gfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuXHJcblxyXG5AQ29tcG9uZW50KHtcclxuICBzZWxlY3RvcjogJ2JyLXNrZWxldG9uJyxcclxuICBzdGFuZGFsb25lOiB0cnVlLFxyXG4gIGltcG9ydHM6IFtdLFxyXG4gIGVuY2Fwc3VsYXRpb246IFZpZXdFbmNhcHN1bGF0aW9uLk5vbmUsXHJcbiAgdGVtcGxhdGU6IGBcclxuICAgIEBzd2l0Y2ggKHZhcmlhbnQpIHtcclxuICAgICAgQGNhc2UgKCdjaXJjbGUnKSB7XHJcbiAgICAgICAgPGRpdiBbY2xhc3NdPVwiY2lyY2xlQ2xhc3Nlc1wiIHJvbGU9XCJzdGF0dXNcIiBhcmlhLWxhYmVsPVwiQ2FycmVnYW5kb1wiPlxyXG4gICAgICAgICAgPHNwYW4gY2xhc3M9XCJzci1vbmx5XCI+Q2FycmVnYW5kby4uLjwvc3Bhbj5cclxuICAgICAgICA8L2Rpdj5cclxuICAgICAgfVxyXG4gICAgICBAY2FzZSAoJ2NhcmQnKSB7XHJcbiAgICAgICAgPGRpdiBjbGFzcz1cInJvdW5kZWQtbWQgb3ZlcmZsb3ctaGlkZGVuIGJvcmRlciBib3JkZXItZ292YnItZ3JheS0xMFwiIHJvbGU9XCJzdGF0dXNcIiBhcmlhLWxhYmVsPVwiQ2FycmVnYW5kb1wiPlxyXG4gICAgICAgICAgPGRpdiBjbGFzcz1cImJyLXNrZWxldG9uLXB1bHNlIGJnLWdvdmJyLWdyYXktMTAgaC0zNlwiPjwvZGl2PlxyXG4gICAgICAgICAgPGRpdiBjbGFzcz1cInAtNCBzcGFjZS15LTNcIj5cclxuICAgICAgICAgICAgPGRpdiBjbGFzcz1cImJyLXNrZWxldG9uLXB1bHNlIGJnLWdvdmJyLWdyYXktMTAgaC00IHJvdW5kZWQgdy0zLzRcIj48L2Rpdj5cclxuICAgICAgICAgICAgPGRpdiBjbGFzcz1cImJyLXNrZWxldG9uLXB1bHNlIGJnLWdvdmJyLWdyYXktMTAgaC0zIHJvdW5kZWQgdy1mdWxsXCI+PC9kaXY+XHJcbiAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJici1za2VsZXRvbi1wdWxzZSBiZy1nb3Zici1ncmF5LTEwIGgtMyByb3VuZGVkIHctNS82XCI+PC9kaXY+XHJcbiAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgIDxzcGFuIGNsYXNzPVwic3Itb25seVwiPkNhcnJlZ2FuZG8uLi48L3NwYW4+XHJcbiAgICAgICAgPC9kaXY+XHJcbiAgICAgIH1cclxuICAgICAgQGNhc2UgKCd0YWJsZScpIHtcclxuICAgICAgICA8ZGl2IGNsYXNzPVwiYm9yZGVyIGJvcmRlci1nb3Zici1ncmF5LTEwIHJvdW5kZWQtbWQgb3ZlcmZsb3ctaGlkZGVuXCIgcm9sZT1cInN0YXR1c1wiIGFyaWEtbGFiZWw9XCJDYXJyZWdhbmRvXCI+XHJcbiAgICAgICAgICA8ZGl2IGNsYXNzPVwiYmctZ292YnItZ3JheS01IHB4LTQgcHktMyBmbGV4IGdhcC00XCI+XHJcbiAgICAgICAgICAgIEBmb3IgKGMgb2YgdGFibGVDb2xzOyB0cmFjayBjKSB7XHJcbiAgICAgICAgICAgICAgPGRpdiBjbGFzcz1cImJyLXNrZWxldG9uLXB1bHNlIGJnLWdvdmJyLWdyYXktMTAgaC0zIHJvdW5kZWQgZmxleC0xXCI+PC9kaXY+XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgIDwvZGl2PlxyXG4gICAgICAgICAgQGZvciAociBvZiB0YWJsZVJvd3M7IHRyYWNrIHIpIHtcclxuICAgICAgICAgICAgPGRpdiBjbGFzcz1cInB4LTQgcHktMyBib3JkZXItdCBib3JkZXItZ292YnItZ3JheS01IGZsZXggZ2FwLTRcIj5cclxuICAgICAgICAgICAgICBAZm9yIChjIG9mIHRhYmxlQ29sczsgdHJhY2sgYykge1xyXG4gICAgICAgICAgICAgICAgPGRpdiBjbGFzcz1cImJyLXNrZWxldG9uLXB1bHNlIGJnLWdvdmJyLWdyYXktMTAgaC0zIHJvdW5kZWQgZmxleC0xXCI+PC9kaXY+XHJcbiAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgIH1cclxuICAgICAgICAgIDxzcGFuIGNsYXNzPVwic3Itb25seVwiPkNhcnJlZ2FuZG8uLi48L3NwYW4+XHJcbiAgICAgICAgPC9kaXY+XHJcbiAgICAgIH1cclxuICAgICAgQGNhc2UgKCdsaXN0Jykge1xyXG4gICAgICAgIDxkaXYgY2xhc3M9XCJzcGFjZS15LTNcIiByb2xlPVwic3RhdHVzXCIgYXJpYS1sYWJlbD1cIkNhcnJlZ2FuZG9cIj5cclxuICAgICAgICAgIEBmb3IgKHIgb2YgbGlzdFJvd3M7IHRyYWNrIHIpIHtcclxuICAgICAgICAgICAgPGRpdiBjbGFzcz1cImZsZXggaXRlbXMtY2VudGVyIGdhcC0zXCI+XHJcbiAgICAgICAgICAgICAgPGRpdiBjbGFzcz1cImJyLXNrZWxldG9uLXB1bHNlIGJnLWdvdmJyLWdyYXktMTAgdy0xMCBoLTEwIHJvdW5kZWQtZnVsbCBmbGV4LXNocmluay0wXCI+PC9kaXY+XHJcbiAgICAgICAgICAgICAgPGRpdiBjbGFzcz1cImZsZXgtMSBzcGFjZS15LTJcIj5cclxuICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJici1za2VsZXRvbi1wdWxzZSBiZy1nb3Zici1ncmF5LTEwIGgtMy41IHJvdW5kZWQgdy0yLzNcIj48L2Rpdj5cclxuICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJici1za2VsZXRvbi1wdWxzZSBiZy1nb3Zici1ncmF5LTEwIGgtMi41IHJvdW5kZWQgdy0xLzJcIj48L2Rpdj5cclxuICAgICAgICAgICAgICA8L2Rpdj5cclxuICAgICAgICAgICAgPC9kaXY+XHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgICA8c3BhbiBjbGFzcz1cInNyLW9ubHlcIj5DYXJyZWdhbmRvLi4uPC9zcGFuPlxyXG4gICAgICAgIDwvZGl2PlxyXG4gICAgICB9XHJcbiAgICAgIEBkZWZhdWx0IHtcclxuICAgICAgICA8IS0tIGxpbmUgdmFyaWFudCAtLT5cclxuICAgICAgICA8ZGl2IFtjbGFzc109XCJsaW5lQ2xhc3Nlc1wiIHJvbGU9XCJzdGF0dXNcIiBhcmlhLWxhYmVsPVwiQ2FycmVnYW5kb1wiPlxyXG4gICAgICAgICAgPHNwYW4gY2xhc3M9XCJzci1vbmx5XCI+Q2FycmVnYW5kby4uLjwvc3Bhbj5cclxuICAgICAgICA8L2Rpdj5cclxuICAgICAgfVxyXG4gICAgfVxyXG4gIGAsXHJcbiAgc3R5bGVzOiBbYFxyXG4gICAgQGtleWZyYW1lcyBici1za2VsZXRvbi1wdWxzZSB7XHJcbiAgICAgIDAlLCAxMDAlIHsgb3BhY2l0eTogMTsgfVxyXG4gICAgICA1MCUgeyBvcGFjaXR5OiAwLjQ7IH1cclxuICAgIH1cclxuICAgIC5ici1za2VsZXRvbi1wdWxzZSB7XHJcbiAgICAgIGFuaW1hdGlvbjogYnItc2tlbGV0b24tcHVsc2UgMS41cyBlYXNlLWluLW91dCBpbmZpbml0ZTtcclxuICAgIH1cclxuICBgXSxcclxufSlcclxuZXhwb3J0IGNsYXNzIEJyU2tlbGV0b25Db21wb25lbnQge1xyXG4gIEBJbnB1dCgpIHZhcmlhbnQ6ICdsaW5lJyB8ICdjaXJjbGUnIHwgJ2NhcmQnIHwgJ3RhYmxlJyB8ICdsaXN0JyA9ICdsaW5lJztcclxuICBASW5wdXQoKSB3aWR0aD86IHN0cmluZztcclxuICBASW5wdXQoKSBoZWlnaHQ/OiBzdHJpbmc7XHJcbiAgQElucHV0KCkgcm93cyA9IDM7XHJcbiAgQElucHV0KCkgY29scyA9IDQ7XHJcblxyXG4gIGdldCBsaW5lQ2xhc3NlcygpOiBzdHJpbmcge1xyXG4gICAgY29uc3QgdyA9IHRoaXMud2lkdGggfHwgJ3ctZnVsbCc7XHJcbiAgICBjb25zdCBoID0gdGhpcy5oZWlnaHQgfHwgJ2gtNCc7XHJcbiAgICByZXR1cm4gYGJyLXNrZWxldG9uLXB1bHNlIGJnLWdvdmJyLWdyYXktMTAgcm91bmRlZCAke3d9ICR7aH1gO1xyXG4gIH1cclxuXHJcbiAgZ2V0IGNpcmNsZUNsYXNzZXMoKTogc3RyaW5nIHtcclxuICAgIGNvbnN0IHNpemUgPSB0aGlzLndpZHRoIHx8ICd3LTEwIGgtMTAnO1xyXG4gICAgcmV0dXJuIGBici1za2VsZXRvbi1wdWxzZSBiZy1nb3Zici1ncmF5LTEwIHJvdW5kZWQtZnVsbCAke3NpemV9YDtcclxuICB9XHJcblxyXG4gIGdldCB0YWJsZVJvd3MoKTogbnVtYmVyW10ge1xyXG4gICAgcmV0dXJuIEFycmF5LmZyb20oeyBsZW5ndGg6IHRoaXMucm93cyB9LCAoXywgaSkgPT4gaSk7XHJcbiAgfVxyXG5cclxuICBnZXQgdGFibGVDb2xzKCk6IG51bWJlcltdIHtcclxuICAgIHJldHVybiBBcnJheS5mcm9tKHsgbGVuZ3RoOiB0aGlzLmNvbHMgfSwgKF8sIGkpID0+IGkpO1xyXG4gIH1cclxuXHJcbiAgZ2V0IGxpc3RSb3dzKCk6IG51bWJlcltdIHtcclxuICAgIHJldHVybiBBcnJheS5mcm9tKHsgbGVuZ3RoOiB0aGlzLnJvd3MgfSwgKF8sIGkpID0+IGkpO1xyXG4gIH1cclxufVxyXG4iXX0=