@schenkerjon/ng-govbr-tw 0.4.0 → 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/animations/animate.directive.mjs +239 -0
- package/esm2022/lib/animations/animation.service.mjs +178 -0
- package/esm2022/lib/animations/animation.types.mjs +34 -0
- package/esm2022/lib/animations/index.mjs +13 -0
- package/esm2022/lib/animations/success-check.component.mjs +129 -0
- package/esm2022/lib/components/accordion/br-accordion.component.mjs +98 -47
- package/esm2022/lib/components/button/br-button.component.mjs +27 -13
- package/esm2022/lib/components/card/br-card.component.mjs +51 -32
- package/esm2022/lib/components/chip/br-chip.component.mjs +141 -0
- package/esm2022/lib/components/divider/br-divider.component.mjs +104 -0
- package/esm2022/lib/components/input/br-input.component.mjs +7 -11
- package/esm2022/lib/components/input-cep/br-input-cep.component.mjs +296 -0
- package/esm2022/lib/components/input-cnpj/br-input-cnpj.component.mjs +7 -11
- package/esm2022/lib/components/input-cpf/br-input-cpf.component.mjs +7 -11
- package/esm2022/lib/components/input-money/br-input-money.component.mjs +325 -0
- package/esm2022/lib/components/input-telefone/br-input-telefone.component.mjs +264 -0
- package/esm2022/lib/components/modal/br-modal.component.mjs +35 -21
- package/esm2022/lib/components/notification/br-notification.component.mjs +131 -0
- package/esm2022/lib/components/pagination/br-pagination.component.mjs +228 -132
- package/esm2022/lib/components/select/br-select.component.mjs +7 -11
- package/esm2022/lib/components/signin/br-signin.component.mjs +169 -0
- package/esm2022/lib/components/skeleton/br-skeleton.component.mjs +41 -25
- package/esm2022/lib/components/stepper/br-stepper.component.mjs +241 -0
- package/esm2022/lib/components/switch/br-switch.component.mjs +157 -0
- package/esm2022/lib/components/table/br-table.component.mjs +1 -1
- package/esm2022/lib/components/textarea/br-textarea.component.mjs +7 -11
- package/esm2022/lib/components/timeline/br-timeline.component.mjs +123 -0
- package/esm2022/lib/components/tooltip/br-tooltip.component.mjs +31 -14
- package/esm2022/lib/components/upload/br-upload.component.mjs +7 -7
- package/esm2022/public-api.mjs +13 -1
- package/fesm2022/schenkerjon-ng-govbr-tw.mjs +3024 -322
- package/fesm2022/schenkerjon-ng-govbr-tw.mjs.map +1 -1
- package/lib/animations/animate.directive.d.ts +84 -0
- package/lib/animations/animation.service.d.ts +90 -0
- package/lib/animations/animation.types.d.ts +117 -0
- package/lib/animations/index.d.ts +8 -0
- package/lib/animations/success-check.component.d.ts +28 -0
- package/lib/components/accordion/br-accordion.component.d.ts +10 -2
- package/lib/components/button/br-button.component.d.ts +7 -1
- package/lib/components/card/br-card.component.d.ts +5 -1
- package/lib/components/chip/br-chip.component.d.ts +19 -0
- package/lib/components/divider/br-divider.component.d.ts +14 -0
- package/lib/components/input/br-input.component.d.ts +4 -4
- package/lib/components/input-cep/br-input-cep.component.d.ts +55 -0
- package/lib/components/input-cnpj/br-input-cnpj.component.d.ts +4 -4
- package/lib/components/input-cpf/br-input-cpf.component.d.ts +4 -4
- package/lib/components/input-money/br-input-money.component.d.ts +44 -0
- package/lib/components/input-telefone/br-input-telefone.component.d.ts +36 -0
- package/lib/components/modal/br-modal.component.d.ts +4 -0
- package/lib/components/notification/br-notification.component.d.ts +26 -0
- package/lib/components/pagination/br-pagination.component.d.ts +12 -1
- package/lib/components/select/br-select.component.d.ts +4 -4
- package/lib/components/signin/br-signin.component.d.ts +32 -0
- package/lib/components/skeleton/br-skeleton.component.d.ts +4 -1
- package/lib/components/stepper/br-stepper.component.d.ts +23 -0
- package/lib/components/switch/br-switch.component.d.ts +24 -0
- package/lib/components/textarea/br-textarea.component.d.ts +4 -4
- package/lib/components/timeline/br-timeline.component.d.ts +15 -0
- package/lib/components/tooltip/br-tooltip.component.d.ts +2 -0
- package/package.json +1 -1
- package/public-api.d.ts +14 -0
- package/src/lib/styles/govbr-components.css +91 -0
- package/src/lib/styles/govbr-contrast.css +252 -0
|
@@ -0,0 +1,239 @@
|
|
|
1
|
+
import { Directive, ElementRef, Input, Output, EventEmitter, inject, effect, } from '@angular/core';
|
|
2
|
+
import { BrAnimationService } from './animation.service';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
/**
|
|
5
|
+
* Diretiva para aplicar animações a qualquer elemento
|
|
6
|
+
*
|
|
7
|
+
* @example
|
|
8
|
+
* <!-- Animação imediata -->
|
|
9
|
+
* <div brAnimate="fadeInUp">Conteúdo</div>
|
|
10
|
+
*
|
|
11
|
+
* <!-- Animação quando visível -->
|
|
12
|
+
* <div brAnimate="fadeInUp" brAnimateTrigger="visible">Conteúdo</div>
|
|
13
|
+
*
|
|
14
|
+
* <!-- Com configurações -->
|
|
15
|
+
* <div brAnimate="scaleIn" [brAnimateDuration]="300" brAnimateEasing="easeOutBack">
|
|
16
|
+
* Conteúdo
|
|
17
|
+
* </div>
|
|
18
|
+
*
|
|
19
|
+
* <!-- Stagger em lista -->
|
|
20
|
+
* <ul>
|
|
21
|
+
* <li *ngFor="let item of items; let i = index"
|
|
22
|
+
* brAnimate="fadeInUp"
|
|
23
|
+
* [brAnimateDelay]="i * 50">
|
|
24
|
+
* {{ item }}
|
|
25
|
+
* </li>
|
|
26
|
+
* </ul>
|
|
27
|
+
*/
|
|
28
|
+
export class BrAnimateDirective {
|
|
29
|
+
constructor() {
|
|
30
|
+
this.el = inject(ElementRef);
|
|
31
|
+
this.animationService = inject(BrAnimationService);
|
|
32
|
+
/** Tipo de animação */
|
|
33
|
+
this.animation = 'fadeIn';
|
|
34
|
+
/** Trigger que inicia a animação */
|
|
35
|
+
this.brAnimateTrigger = 'immediate';
|
|
36
|
+
/** Threshold para Intersection Observer (0-1) */
|
|
37
|
+
this.brAnimateThreshold = 0.1;
|
|
38
|
+
/** Se deve animar apenas uma vez */
|
|
39
|
+
this.brAnimateOnce = true;
|
|
40
|
+
/** Se a animação está habilitada */
|
|
41
|
+
this.brAnimateEnabled = true;
|
|
42
|
+
/** Evento emitido no início da animação */
|
|
43
|
+
this.brAnimateStart = new EventEmitter();
|
|
44
|
+
/** Evento emitido no fim da animação */
|
|
45
|
+
this.brAnimateEnd = new EventEmitter();
|
|
46
|
+
this.hasAnimated = false;
|
|
47
|
+
this.boundHandleAnimationEnd = this.handleAnimationEnd.bind(this);
|
|
48
|
+
this.boundHandleHover = this.handleHover.bind(this);
|
|
49
|
+
this.boundHandleHoverOut = this.handleHoverOut.bind(this);
|
|
50
|
+
this.boundHandleFocus = this.handleFocus.bind(this);
|
|
51
|
+
this.boundHandleBlur = this.handleBlur.bind(this);
|
|
52
|
+
this.boundHandleClick = this.handleClick.bind(this);
|
|
53
|
+
// Reage a mudanças no serviço de animação
|
|
54
|
+
effect(() => {
|
|
55
|
+
if (!this.animationService.isEnabled()) {
|
|
56
|
+
this.clearAnimation();
|
|
57
|
+
}
|
|
58
|
+
});
|
|
59
|
+
}
|
|
60
|
+
ngOnInit() {
|
|
61
|
+
if (!this.brAnimateEnabled)
|
|
62
|
+
return;
|
|
63
|
+
this.setupTrigger();
|
|
64
|
+
}
|
|
65
|
+
ngOnDestroy() {
|
|
66
|
+
this.cleanup();
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Inicia a animação programaticamente
|
|
70
|
+
*/
|
|
71
|
+
play() {
|
|
72
|
+
if (!this.canAnimate())
|
|
73
|
+
return;
|
|
74
|
+
this.applyAnimation();
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Reseta a animação para poder rodar novamente
|
|
78
|
+
*/
|
|
79
|
+
reset() {
|
|
80
|
+
this.hasAnimated = false;
|
|
81
|
+
this.clearAnimation();
|
|
82
|
+
}
|
|
83
|
+
setupTrigger() {
|
|
84
|
+
const element = this.el.nativeElement;
|
|
85
|
+
switch (this.brAnimateTrigger) {
|
|
86
|
+
case 'immediate':
|
|
87
|
+
// Pequeno delay para garantir que o elemento está no DOM
|
|
88
|
+
requestAnimationFrame(() => this.play());
|
|
89
|
+
break;
|
|
90
|
+
case 'visible':
|
|
91
|
+
this.setupIntersectionObserver();
|
|
92
|
+
break;
|
|
93
|
+
case 'hover':
|
|
94
|
+
element.addEventListener('mouseenter', this.boundHandleHover);
|
|
95
|
+
element.addEventListener('mouseleave', this.boundHandleHoverOut);
|
|
96
|
+
break;
|
|
97
|
+
case 'focus':
|
|
98
|
+
element.addEventListener('focus', this.boundHandleFocus);
|
|
99
|
+
element.addEventListener('blur', this.boundHandleBlur);
|
|
100
|
+
break;
|
|
101
|
+
case 'click':
|
|
102
|
+
element.addEventListener('click', this.boundHandleClick);
|
|
103
|
+
break;
|
|
104
|
+
case 'manual':
|
|
105
|
+
// Não faz nada, espera chamada programática
|
|
106
|
+
break;
|
|
107
|
+
}
|
|
108
|
+
// Listener para fim da animação
|
|
109
|
+
element.addEventListener('animationend', this.boundHandleAnimationEnd);
|
|
110
|
+
}
|
|
111
|
+
setupIntersectionObserver() {
|
|
112
|
+
if (typeof IntersectionObserver === 'undefined') {
|
|
113
|
+
// Fallback: anima imediatamente
|
|
114
|
+
this.play();
|
|
115
|
+
return;
|
|
116
|
+
}
|
|
117
|
+
this.observer = new IntersectionObserver((entries) => {
|
|
118
|
+
entries.forEach((entry) => {
|
|
119
|
+
if (entry.isIntersecting && this.canAnimate()) {
|
|
120
|
+
this.applyAnimation();
|
|
121
|
+
if (this.brAnimateOnce) {
|
|
122
|
+
this.observer?.disconnect();
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
});
|
|
126
|
+
}, { threshold: this.brAnimateThreshold });
|
|
127
|
+
this.observer.observe(this.el.nativeElement);
|
|
128
|
+
}
|
|
129
|
+
canAnimate() {
|
|
130
|
+
if (!this.brAnimateEnabled)
|
|
131
|
+
return false;
|
|
132
|
+
if (!this.animationService.isEnabled())
|
|
133
|
+
return false;
|
|
134
|
+
if (this.brAnimateOnce && this.hasAnimated)
|
|
135
|
+
return false;
|
|
136
|
+
return true;
|
|
137
|
+
}
|
|
138
|
+
applyAnimation() {
|
|
139
|
+
const element = this.el.nativeElement;
|
|
140
|
+
const duration = this.animationService.resolveDuration(this.brAnimateDuration);
|
|
141
|
+
const easing = this.animationService.resolveEasing(this.brAnimateEasing);
|
|
142
|
+
const delay = this.brAnimateDelay || 0;
|
|
143
|
+
// Emite evento de início
|
|
144
|
+
this.brAnimateStart.emit({
|
|
145
|
+
type: this.animation,
|
|
146
|
+
phase: 'start',
|
|
147
|
+
element,
|
|
148
|
+
timestamp: Date.now(),
|
|
149
|
+
});
|
|
150
|
+
// Aplica a animação
|
|
151
|
+
element.style.animation = `br-${this.animation} ${duration}ms ${easing} ${delay}ms both`;
|
|
152
|
+
this.hasAnimated = true;
|
|
153
|
+
}
|
|
154
|
+
clearAnimation() {
|
|
155
|
+
const element = this.el.nativeElement;
|
|
156
|
+
element.style.animation = '';
|
|
157
|
+
}
|
|
158
|
+
handleAnimationEnd() {
|
|
159
|
+
const element = this.el.nativeElement;
|
|
160
|
+
this.brAnimateEnd.emit({
|
|
161
|
+
type: this.animation,
|
|
162
|
+
phase: 'end',
|
|
163
|
+
element,
|
|
164
|
+
timestamp: Date.now(),
|
|
165
|
+
});
|
|
166
|
+
// Para triggers de hover/focus, limpa para poder repetir
|
|
167
|
+
if (this.brAnimateTrigger === 'hover' || this.brAnimateTrigger === 'focus') {
|
|
168
|
+
this.clearAnimation();
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
handleHover() {
|
|
172
|
+
if (this.canAnimate() || this.brAnimateTrigger === 'hover') {
|
|
173
|
+
this.hasAnimated = false; // Permite repetir no hover
|
|
174
|
+
this.applyAnimation();
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
handleHoverOut() {
|
|
178
|
+
this.clearAnimation();
|
|
179
|
+
}
|
|
180
|
+
handleFocus() {
|
|
181
|
+
if (this.canAnimate() || this.brAnimateTrigger === 'focus') {
|
|
182
|
+
this.hasAnimated = false;
|
|
183
|
+
this.applyAnimation();
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
handleBlur() {
|
|
187
|
+
this.clearAnimation();
|
|
188
|
+
}
|
|
189
|
+
handleClick() {
|
|
190
|
+
if (this.canAnimate() || this.brAnimateTrigger === 'click') {
|
|
191
|
+
this.hasAnimated = false;
|
|
192
|
+
this.clearAnimation();
|
|
193
|
+
// Força reflow para reiniciar a animação
|
|
194
|
+
this.el.nativeElement.getBoundingClientRect();
|
|
195
|
+
this.applyAnimation();
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
cleanup() {
|
|
199
|
+
const element = this.el.nativeElement;
|
|
200
|
+
this.observer?.disconnect();
|
|
201
|
+
element.removeEventListener('animationend', this.boundHandleAnimationEnd);
|
|
202
|
+
element.removeEventListener('mouseenter', this.boundHandleHover);
|
|
203
|
+
element.removeEventListener('mouseleave', this.boundHandleHoverOut);
|
|
204
|
+
element.removeEventListener('focus', this.boundHandleFocus);
|
|
205
|
+
element.removeEventListener('blur', this.boundHandleBlur);
|
|
206
|
+
element.removeEventListener('click', this.boundHandleClick);
|
|
207
|
+
}
|
|
208
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrAnimateDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
209
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.3.12", type: BrAnimateDirective, isStandalone: true, selector: "[brAnimate]", inputs: { animation: ["brAnimate", "animation"], brAnimateTrigger: "brAnimateTrigger", brAnimateDuration: "brAnimateDuration", brAnimateDelay: "brAnimateDelay", brAnimateEasing: "brAnimateEasing", brAnimateThreshold: "brAnimateThreshold", brAnimateOnce: "brAnimateOnce", brAnimateEnabled: "brAnimateEnabled" }, outputs: { brAnimateStart: "brAnimateStart", brAnimateEnd: "brAnimateEnd" }, ngImport: i0 }); }
|
|
210
|
+
}
|
|
211
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrAnimateDirective, decorators: [{
|
|
212
|
+
type: Directive,
|
|
213
|
+
args: [{
|
|
214
|
+
selector: '[brAnimate]',
|
|
215
|
+
standalone: true,
|
|
216
|
+
}]
|
|
217
|
+
}], ctorParameters: () => [], propDecorators: { animation: [{
|
|
218
|
+
type: Input,
|
|
219
|
+
args: ['brAnimate']
|
|
220
|
+
}], brAnimateTrigger: [{
|
|
221
|
+
type: Input
|
|
222
|
+
}], brAnimateDuration: [{
|
|
223
|
+
type: Input
|
|
224
|
+
}], brAnimateDelay: [{
|
|
225
|
+
type: Input
|
|
226
|
+
}], brAnimateEasing: [{
|
|
227
|
+
type: Input
|
|
228
|
+
}], brAnimateThreshold: [{
|
|
229
|
+
type: Input
|
|
230
|
+
}], brAnimateOnce: [{
|
|
231
|
+
type: Input
|
|
232
|
+
}], brAnimateEnabled: [{
|
|
233
|
+
type: Input
|
|
234
|
+
}], brAnimateStart: [{
|
|
235
|
+
type: Output
|
|
236
|
+
}], brAnimateEnd: [{
|
|
237
|
+
type: Output
|
|
238
|
+
}] } });
|
|
239
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"animate.directive.js","sourceRoot":"","sources":["../../../../../projects/ng-govbr-tw/src/lib/animations/animate.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Directive,\r\n  ElementRef,\r\n  Input,\r\n  OnInit,\r\n  OnDestroy,\r\n  Output,\r\n  EventEmitter,\r\n  inject,\r\n  effect,\r\n} from '@angular/core';\r\nimport { BrAnimationService } from './animation.service';\r\nimport {\r\n  BrAnimationType,\r\n  BrAnimationDuration,\r\n  BrAnimationEasing,\r\n  BrAnimationTrigger,\r\n  BrAnimationEvent,\r\n} from './animation.types';\r\n\r\n/**\r\n * Diretiva para aplicar animações a qualquer elemento\r\n *\r\n * @example\r\n * <!-- Animação imediata -->\r\n * <div brAnimate=\"fadeInUp\">Conteúdo</div>\r\n *\r\n * <!-- Animação quando visível -->\r\n * <div brAnimate=\"fadeInUp\" brAnimateTrigger=\"visible\">Conteúdo</div>\r\n *\r\n * <!-- Com configurações -->\r\n * <div brAnimate=\"scaleIn\" [brAnimateDuration]=\"300\" brAnimateEasing=\"easeOutBack\">\r\n *   Conteúdo\r\n * </div>\r\n *\r\n * <!-- Stagger em lista -->\r\n * <ul>\r\n *   <li *ngFor=\"let item of items; let i = index\"\r\n *       brAnimate=\"fadeInUp\"\r\n *       [brAnimateDelay]=\"i * 50\">\r\n *     {{ item }}\r\n *   </li>\r\n * </ul>\r\n */\r\n@Directive({\r\n  selector: '[brAnimate]',\r\n  standalone: true,\r\n})\r\nexport class BrAnimateDirective implements OnInit, OnDestroy {\r\n  private readonly el = inject(ElementRef);\r\n  private readonly animationService = inject(BrAnimationService);\r\n\r\n  /** Tipo de animação */\r\n  @Input('brAnimate') animation: BrAnimationType = 'fadeIn';\r\n\r\n  /** Trigger que inicia a animação */\r\n  @Input() brAnimateTrigger: BrAnimationTrigger = 'immediate';\r\n\r\n  /** Duração da animação */\r\n  @Input() brAnimateDuration?: number | BrAnimationDuration;\r\n\r\n  /** Delay antes de iniciar */\r\n  @Input() brAnimateDelay?: number;\r\n\r\n  /** Curva de easing */\r\n  @Input() brAnimateEasing?: BrAnimationEasing;\r\n\r\n  /** Threshold para Intersection Observer (0-1) */\r\n  @Input() brAnimateThreshold = 0.1;\r\n\r\n  /** Se deve animar apenas uma vez */\r\n  @Input() brAnimateOnce = true;\r\n\r\n  /** Se a animação está habilitada */\r\n  @Input() brAnimateEnabled = true;\r\n\r\n  /** Evento emitido no início da animação */\r\n  @Output() brAnimateStart = new EventEmitter<BrAnimationEvent>();\r\n\r\n  /** Evento emitido no fim da animação */\r\n  @Output() brAnimateEnd = new EventEmitter<BrAnimationEvent>();\r\n\r\n  private observer?: IntersectionObserver;\r\n  private hasAnimated = false;\r\n  private boundHandleAnimationEnd = this.handleAnimationEnd.bind(this);\r\n  private boundHandleHover = this.handleHover.bind(this);\r\n  private boundHandleHoverOut = this.handleHoverOut.bind(this);\r\n  private boundHandleFocus = this.handleFocus.bind(this);\r\n  private boundHandleBlur = this.handleBlur.bind(this);\r\n  private boundHandleClick = this.handleClick.bind(this);\r\n\r\n  constructor() {\r\n    // Reage a mudanças no serviço de animação\r\n    effect(() => {\r\n      if (!this.animationService.isEnabled()) {\r\n        this.clearAnimation();\r\n      }\r\n    });\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    if (!this.brAnimateEnabled) return;\r\n\r\n    this.setupTrigger();\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.cleanup();\r\n  }\r\n\r\n  /**\r\n   * Inicia a animação programaticamente\r\n   */\r\n  play(): void {\r\n    if (!this.canAnimate()) return;\r\n    this.applyAnimation();\r\n  }\r\n\r\n  /**\r\n   * Reseta a animação para poder rodar novamente\r\n   */\r\n  reset(): void {\r\n    this.hasAnimated = false;\r\n    this.clearAnimation();\r\n  }\r\n\r\n  private setupTrigger(): void {\r\n    const element = this.el.nativeElement as HTMLElement;\r\n\r\n    switch (this.brAnimateTrigger) {\r\n      case 'immediate':\r\n        // Pequeno delay para garantir que o elemento está no DOM\r\n        requestAnimationFrame(() => this.play());\r\n        break;\r\n\r\n      case 'visible':\r\n        this.setupIntersectionObserver();\r\n        break;\r\n\r\n      case 'hover':\r\n        element.addEventListener('mouseenter', this.boundHandleHover);\r\n        element.addEventListener('mouseleave', this.boundHandleHoverOut);\r\n        break;\r\n\r\n      case 'focus':\r\n        element.addEventListener('focus', this.boundHandleFocus);\r\n        element.addEventListener('blur', this.boundHandleBlur);\r\n        break;\r\n\r\n      case 'click':\r\n        element.addEventListener('click', this.boundHandleClick);\r\n        break;\r\n\r\n      case 'manual':\r\n        // Não faz nada, espera chamada programática\r\n        break;\r\n    }\r\n\r\n    // Listener para fim da animação\r\n    element.addEventListener('animationend', this.boundHandleAnimationEnd);\r\n  }\r\n\r\n  private setupIntersectionObserver(): void {\r\n    if (typeof IntersectionObserver === 'undefined') {\r\n      // Fallback: anima imediatamente\r\n      this.play();\r\n      return;\r\n    }\r\n\r\n    this.observer = new IntersectionObserver(\r\n      (entries) => {\r\n        entries.forEach((entry) => {\r\n          if (entry.isIntersecting && this.canAnimate()) {\r\n            this.applyAnimation();\r\n\r\n            if (this.brAnimateOnce) {\r\n              this.observer?.disconnect();\r\n            }\r\n          }\r\n        });\r\n      },\r\n      { threshold: this.brAnimateThreshold }\r\n    );\r\n\r\n    this.observer.observe(this.el.nativeElement);\r\n  }\r\n\r\n  private canAnimate(): boolean {\r\n    if (!this.brAnimateEnabled) return false;\r\n    if (!this.animationService.isEnabled()) return false;\r\n    if (this.brAnimateOnce && this.hasAnimated) return false;\r\n    return true;\r\n  }\r\n\r\n  private applyAnimation(): void {\r\n    const element = this.el.nativeElement as HTMLElement;\r\n    const duration = this.animationService.resolveDuration(this.brAnimateDuration);\r\n    const easing = this.animationService.resolveEasing(this.brAnimateEasing);\r\n    const delay = this.brAnimateDelay || 0;\r\n\r\n    // Emite evento de início\r\n    this.brAnimateStart.emit({\r\n      type: this.animation,\r\n      phase: 'start',\r\n      element,\r\n      timestamp: Date.now(),\r\n    });\r\n\r\n    // Aplica a animação\r\n    element.style.animation = `br-${this.animation} ${duration}ms ${easing} ${delay}ms both`;\r\n\r\n    this.hasAnimated = true;\r\n  }\r\n\r\n  private clearAnimation(): void {\r\n    const element = this.el.nativeElement as HTMLElement;\r\n    element.style.animation = '';\r\n  }\r\n\r\n  private handleAnimationEnd(): void {\r\n    const element = this.el.nativeElement as HTMLElement;\r\n\r\n    this.brAnimateEnd.emit({\r\n      type: this.animation,\r\n      phase: 'end',\r\n      element,\r\n      timestamp: Date.now(),\r\n    });\r\n\r\n    // Para triggers de hover/focus, limpa para poder repetir\r\n    if (this.brAnimateTrigger === 'hover' || this.brAnimateTrigger === 'focus') {\r\n      this.clearAnimation();\r\n    }\r\n  }\r\n\r\n  private handleHover(): void {\r\n    if (this.canAnimate() || this.brAnimateTrigger === 'hover') {\r\n      this.hasAnimated = false; // Permite repetir no hover\r\n      this.applyAnimation();\r\n    }\r\n  }\r\n\r\n  private handleHoverOut(): void {\r\n    this.clearAnimation();\r\n  }\r\n\r\n  private handleFocus(): void {\r\n    if (this.canAnimate() || this.brAnimateTrigger === 'focus') {\r\n      this.hasAnimated = false;\r\n      this.applyAnimation();\r\n    }\r\n  }\r\n\r\n  private handleBlur(): void {\r\n    this.clearAnimation();\r\n  }\r\n\r\n  private handleClick(): void {\r\n    if (this.canAnimate() || this.brAnimateTrigger === 'click') {\r\n      this.hasAnimated = false;\r\n      this.clearAnimation();\r\n      // Força reflow para reiniciar a animação\r\n      this.el.nativeElement.getBoundingClientRect();\r\n      this.applyAnimation();\r\n    }\r\n  }\r\n\r\n  private cleanup(): void {\r\n    const element = this.el.nativeElement as HTMLElement;\r\n\r\n    this.observer?.disconnect();\r\n\r\n    element.removeEventListener('animationend', this.boundHandleAnimationEnd);\r\n    element.removeEventListener('mouseenter', this.boundHandleHover);\r\n    element.removeEventListener('mouseleave', this.boundHandleHoverOut);\r\n    element.removeEventListener('focus', this.boundHandleFocus);\r\n    element.removeEventListener('blur', this.boundHandleBlur);\r\n    element.removeEventListener('click', this.boundHandleClick);\r\n  }\r\n}\r\n"]}
|
|
@@ -0,0 +1,178 @@
|
|
|
1
|
+
import { Injectable, signal, computed, inject, InjectionToken } from '@angular/core';
|
|
2
|
+
import { BR_ANIMATION_DEFAULT_CONFIG, BR_ANIMATION_DURATION_MAP, BR_ANIMATION_EASING_MAP, } from './animation.types';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
/** Token para injetar configuração customizada */
|
|
5
|
+
export const BR_ANIMATION_CONFIG = new InjectionToken('BR_ANIMATION_CONFIG');
|
|
6
|
+
/**
|
|
7
|
+
* Serviço de animações Gov.br
|
|
8
|
+
* @description Gerencia configuração global e estado das animações
|
|
9
|
+
*
|
|
10
|
+
* @example
|
|
11
|
+
* // No módulo/componente principal
|
|
12
|
+
* providers: [
|
|
13
|
+
* {
|
|
14
|
+
* provide: BR_ANIMATION_CONFIG,
|
|
15
|
+
* useValue: { enabled: true, defaultDuration: 'fast' }
|
|
16
|
+
* }
|
|
17
|
+
* ]
|
|
18
|
+
*
|
|
19
|
+
* // No componente
|
|
20
|
+
* constructor(private animationService: BrAnimationService) {
|
|
21
|
+
* this.animationService.enable();
|
|
22
|
+
* }
|
|
23
|
+
*/
|
|
24
|
+
export class BrAnimationService {
|
|
25
|
+
constructor() {
|
|
26
|
+
/** Configuração global reativa */
|
|
27
|
+
this._config = signal({ ...BR_ANIMATION_DEFAULT_CONFIG });
|
|
28
|
+
/** Se animações estão habilitadas (considera reduced motion) */
|
|
29
|
+
this.isEnabled = computed(() => {
|
|
30
|
+
const config = this._config();
|
|
31
|
+
if (!config.enabled)
|
|
32
|
+
return false;
|
|
33
|
+
if (config.respectReducedMotion && this._prefersReducedMotion())
|
|
34
|
+
return false;
|
|
35
|
+
return true;
|
|
36
|
+
});
|
|
37
|
+
/** Configuração atual (readonly) */
|
|
38
|
+
this.config = computed(() => this._config());
|
|
39
|
+
/** Cache de media query */
|
|
40
|
+
this._reducedMotionQuery = null;
|
|
41
|
+
/** Custom config injected optionally */
|
|
42
|
+
this._customConfig = inject(BR_ANIMATION_CONFIG, { optional: true });
|
|
43
|
+
// Merge com configuração customizada
|
|
44
|
+
if (this._customConfig) {
|
|
45
|
+
this._config.update(current => ({ ...current, ...this._customConfig }));
|
|
46
|
+
}
|
|
47
|
+
// Setup listener para prefers-reduced-motion
|
|
48
|
+
this._setupReducedMotionListener();
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Habilita animações globalmente
|
|
52
|
+
*/
|
|
53
|
+
enable() {
|
|
54
|
+
this._config.update(config => ({ ...config, enabled: true }));
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Desabilita animações globalmente
|
|
58
|
+
*/
|
|
59
|
+
disable() {
|
|
60
|
+
this._config.update(config => ({ ...config, enabled: false }));
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Toggle do estado de animações
|
|
64
|
+
*/
|
|
65
|
+
toggle() {
|
|
66
|
+
this._config.update(config => ({ ...config, enabled: !config.enabled }));
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Atualiza configuração global
|
|
70
|
+
*/
|
|
71
|
+
configure(config) {
|
|
72
|
+
this._config.update(current => ({ ...current, ...config }));
|
|
73
|
+
}
|
|
74
|
+
/**
|
|
75
|
+
* Define duração padrão
|
|
76
|
+
*/
|
|
77
|
+
setDefaultDuration(duration) {
|
|
78
|
+
this._config.update(config => ({ ...config, defaultDuration: duration }));
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* Define easing padrão
|
|
82
|
+
*/
|
|
83
|
+
setDefaultEasing(easing) {
|
|
84
|
+
this._config.update(config => ({ ...config, defaultEasing: easing }));
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* Resolve duração para valor em ms
|
|
88
|
+
*/
|
|
89
|
+
resolveDuration(duration) {
|
|
90
|
+
if (duration === undefined) {
|
|
91
|
+
const defaultDur = this._config().defaultDuration;
|
|
92
|
+
return typeof defaultDur === 'number' ? defaultDur : BR_ANIMATION_DURATION_MAP[defaultDur];
|
|
93
|
+
}
|
|
94
|
+
return typeof duration === 'number' ? duration : BR_ANIMATION_DURATION_MAP[duration];
|
|
95
|
+
}
|
|
96
|
+
/**
|
|
97
|
+
* Resolve easing para valor CSS
|
|
98
|
+
*/
|
|
99
|
+
resolveEasing(easing) {
|
|
100
|
+
const resolvedEasing = easing || this._config().defaultEasing;
|
|
101
|
+
return BR_ANIMATION_EASING_MAP[resolvedEasing];
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Gera string de animação CSS
|
|
105
|
+
*/
|
|
106
|
+
getAnimationString(config) {
|
|
107
|
+
if (!this.isEnabled())
|
|
108
|
+
return 'none';
|
|
109
|
+
const duration = this.resolveDuration(config.duration);
|
|
110
|
+
const easing = this.resolveEasing(config.easing);
|
|
111
|
+
const delay = config.delay || 0;
|
|
112
|
+
const iterations = config.iterations === 'infinite' ? 'infinite' : (config.iterations || 1);
|
|
113
|
+
const direction = config.direction || 'normal';
|
|
114
|
+
const fillMode = config.fillMode || 'both';
|
|
115
|
+
return `br-${config.type} ${duration}ms ${easing} ${delay}ms ${iterations} ${direction} ${fillMode}`;
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* Aplica animação a um elemento
|
|
119
|
+
*/
|
|
120
|
+
animate(element, config) {
|
|
121
|
+
return new Promise((resolve, reject) => {
|
|
122
|
+
if (!this.isEnabled()) {
|
|
123
|
+
resolve();
|
|
124
|
+
return;
|
|
125
|
+
}
|
|
126
|
+
const animationString = this.getAnimationString(config);
|
|
127
|
+
element.style.animation = animationString;
|
|
128
|
+
const handleEnd = () => {
|
|
129
|
+
element.removeEventListener('animationend', handleEnd);
|
|
130
|
+
element.removeEventListener('animationcancel', handleCancel);
|
|
131
|
+
resolve();
|
|
132
|
+
};
|
|
133
|
+
const handleCancel = () => {
|
|
134
|
+
element.removeEventListener('animationend', handleEnd);
|
|
135
|
+
element.removeEventListener('animationcancel', handleCancel);
|
|
136
|
+
reject(new Error('Animation cancelled'));
|
|
137
|
+
};
|
|
138
|
+
element.addEventListener('animationend', handleEnd);
|
|
139
|
+
element.addEventListener('animationcancel', handleCancel);
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
/**
|
|
143
|
+
* Remove animação de um elemento
|
|
144
|
+
*/
|
|
145
|
+
clearAnimation(element) {
|
|
146
|
+
element.style.animation = '';
|
|
147
|
+
}
|
|
148
|
+
/**
|
|
149
|
+
* Verifica se usuário prefere movimento reduzido
|
|
150
|
+
*/
|
|
151
|
+
_prefersReducedMotion() {
|
|
152
|
+
if (typeof window === 'undefined')
|
|
153
|
+
return false;
|
|
154
|
+
return this._reducedMotionQuery?.matches ?? false;
|
|
155
|
+
}
|
|
156
|
+
/**
|
|
157
|
+
* Configura listener para mudanças de preferência
|
|
158
|
+
*/
|
|
159
|
+
_setupReducedMotionListener() {
|
|
160
|
+
if (typeof window === 'undefined')
|
|
161
|
+
return;
|
|
162
|
+
this._reducedMotionQuery = window.matchMedia('(prefers-reduced-motion: reduce)');
|
|
163
|
+
// Força re-avaliação quando preferência muda
|
|
164
|
+
this._reducedMotionQuery.addEventListener('change', () => {
|
|
165
|
+
// Trigger recomputation via signal update
|
|
166
|
+
this._config.update(config => ({ ...config }));
|
|
167
|
+
});
|
|
168
|
+
}
|
|
169
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrAnimationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
170
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrAnimationService, providedIn: 'root' }); }
|
|
171
|
+
}
|
|
172
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrAnimationService, decorators: [{
|
|
173
|
+
type: Injectable,
|
|
174
|
+
args: [{
|
|
175
|
+
providedIn: 'root',
|
|
176
|
+
}]
|
|
177
|
+
}], ctorParameters: () => [] });
|
|
178
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"animation.service.js","sourceRoot":"","sources":["../../../../../projects/ng-govbr-tw/src/lib/animations/animation.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, signal, computed, inject, InjectionToken } from '@angular/core';\r\nimport {\r\n  BrAnimationGlobalConfig,\r\n  BrAnimationDuration,\r\n  BrAnimationEasing,\r\n  BrAnimationConfig,\r\n  BR_ANIMATION_DEFAULT_CONFIG,\r\n  BR_ANIMATION_DURATION_MAP,\r\n  BR_ANIMATION_EASING_MAP,\r\n} from './animation.types';\r\n\r\n/** Token para injetar configuração customizada */\r\nexport const BR_ANIMATION_CONFIG = new InjectionToken<Partial<BrAnimationGlobalConfig>>(\r\n  'BR_ANIMATION_CONFIG'\r\n);\r\n\r\n/**\r\n * Serviço de animações Gov.br\r\n * @description Gerencia configuração global e estado das animações\r\n *\r\n * @example\r\n * // No módulo/componente principal\r\n * providers: [\r\n *   {\r\n *     provide: BR_ANIMATION_CONFIG,\r\n *     useValue: { enabled: true, defaultDuration: 'fast' }\r\n *   }\r\n * ]\r\n *\r\n * // No componente\r\n * constructor(private animationService: BrAnimationService) {\r\n *   this.animationService.enable();\r\n * }\r\n */\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class BrAnimationService {\r\n  /** Configuração global reativa */\r\n  private readonly _config = signal<BrAnimationGlobalConfig>({ ...BR_ANIMATION_DEFAULT_CONFIG });\r\n\r\n  /** Se animações estão habilitadas (considera reduced motion) */\r\n  readonly isEnabled = computed(() => {\r\n    const config = this._config();\r\n    if (!config.enabled) return false;\r\n    if (config.respectReducedMotion && this._prefersReducedMotion()) return false;\r\n    return true;\r\n  });\r\n\r\n  /** Configuração atual (readonly) */\r\n  readonly config = computed(() => this._config());\r\n\r\n  /** Cache de media query */\r\n  private _reducedMotionQuery: MediaQueryList | null = null;\r\n\r\n  /** Custom config injected optionally */\r\n  private readonly _customConfig = inject(BR_ANIMATION_CONFIG, { optional: true });\r\n\r\n  constructor() {\r\n    // Merge com configuração customizada\r\n    if (this._customConfig) {\r\n      this._config.update(current => ({ ...current, ...this._customConfig }));\r\n    }\r\n\r\n    // Setup listener para prefers-reduced-motion\r\n    this._setupReducedMotionListener();\r\n  }\r\n\r\n  /**\r\n   * Habilita animações globalmente\r\n   */\r\n  enable(): void {\r\n    this._config.update(config => ({ ...config, enabled: true }));\r\n  }\r\n\r\n  /**\r\n   * Desabilita animações globalmente\r\n   */\r\n  disable(): void {\r\n    this._config.update(config => ({ ...config, enabled: false }));\r\n  }\r\n\r\n  /**\r\n   * Toggle do estado de animações\r\n   */\r\n  toggle(): void {\r\n    this._config.update(config => ({ ...config, enabled: !config.enabled }));\r\n  }\r\n\r\n  /**\r\n   * Atualiza configuração global\r\n   */\r\n  configure(config: Partial<BrAnimationGlobalConfig>): void {\r\n    this._config.update(current => ({ ...current, ...config }));\r\n  }\r\n\r\n  /**\r\n   * Define duração padrão\r\n   */\r\n  setDefaultDuration(duration: number | BrAnimationDuration): void {\r\n    this._config.update(config => ({ ...config, defaultDuration: duration }));\r\n  }\r\n\r\n  /**\r\n   * Define easing padrão\r\n   */\r\n  setDefaultEasing(easing: BrAnimationEasing): void {\r\n    this._config.update(config => ({ ...config, defaultEasing: easing }));\r\n  }\r\n\r\n  /**\r\n   * Resolve duração para valor em ms\r\n   */\r\n  resolveDuration(duration?: number | BrAnimationDuration): number {\r\n    if (duration === undefined) {\r\n      const defaultDur = this._config().defaultDuration;\r\n      return typeof defaultDur === 'number' ? defaultDur : BR_ANIMATION_DURATION_MAP[defaultDur];\r\n    }\r\n    return typeof duration === 'number' ? duration : BR_ANIMATION_DURATION_MAP[duration];\r\n  }\r\n\r\n  /**\r\n   * Resolve easing para valor CSS\r\n   */\r\n  resolveEasing(easing?: BrAnimationEasing): string {\r\n    const resolvedEasing = easing || this._config().defaultEasing;\r\n    return BR_ANIMATION_EASING_MAP[resolvedEasing];\r\n  }\r\n\r\n  /**\r\n   * Gera string de animação CSS\r\n   */\r\n  getAnimationString(config: BrAnimationConfig): string {\r\n    if (!this.isEnabled()) return 'none';\r\n\r\n    const duration = this.resolveDuration(config.duration);\r\n    const easing = this.resolveEasing(config.easing);\r\n    const delay = config.delay || 0;\r\n    const iterations = config.iterations === 'infinite' ? 'infinite' : (config.iterations || 1);\r\n    const direction = config.direction || 'normal';\r\n    const fillMode = config.fillMode || 'both';\r\n\r\n    return `br-${config.type} ${duration}ms ${easing} ${delay}ms ${iterations} ${direction} ${fillMode}`;\r\n  }\r\n\r\n  /**\r\n   * Aplica animação a um elemento\r\n   */\r\n  animate(element: HTMLElement, config: BrAnimationConfig): Promise<void> {\r\n    return new Promise((resolve, reject) => {\r\n      if (!this.isEnabled()) {\r\n        resolve();\r\n        return;\r\n      }\r\n\r\n      const animationString = this.getAnimationString(config);\r\n      element.style.animation = animationString;\r\n\r\n      const handleEnd = () => {\r\n        element.removeEventListener('animationend', handleEnd);\r\n        element.removeEventListener('animationcancel', handleCancel);\r\n        resolve();\r\n      };\r\n\r\n      const handleCancel = () => {\r\n        element.removeEventListener('animationend', handleEnd);\r\n        element.removeEventListener('animationcancel', handleCancel);\r\n        reject(new Error('Animation cancelled'));\r\n      };\r\n\r\n      element.addEventListener('animationend', handleEnd);\r\n      element.addEventListener('animationcancel', handleCancel);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Remove animação de um elemento\r\n   */\r\n  clearAnimation(element: HTMLElement): void {\r\n    element.style.animation = '';\r\n  }\r\n\r\n  /**\r\n   * Verifica se usuário prefere movimento reduzido\r\n   */\r\n  private _prefersReducedMotion(): boolean {\r\n    if (typeof window === 'undefined') return false;\r\n    return this._reducedMotionQuery?.matches ?? false;\r\n  }\r\n\r\n  /**\r\n   * Configura listener para mudanças de preferência\r\n   */\r\n  private _setupReducedMotionListener(): void {\r\n    if (typeof window === 'undefined') return;\r\n\r\n    this._reducedMotionQuery = window.matchMedia('(prefers-reduced-motion: reduce)');\r\n\r\n    // Força re-avaliação quando preferência muda\r\n    this._reducedMotionQuery.addEventListener('change', () => {\r\n      // Trigger recomputation via signal update\r\n      this._config.update(config => ({ ...config }));\r\n    });\r\n  }\r\n}\r\n"]}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tipos e interfaces para o sistema de animações Gov.br
|
|
3
|
+
* @description Sistema de animações configurável e acessível
|
|
4
|
+
*/
|
|
5
|
+
/** Mapeamento de easing para valores CSS */
|
|
6
|
+
export const BR_ANIMATION_EASING_MAP = {
|
|
7
|
+
linear: 'linear',
|
|
8
|
+
ease: 'ease',
|
|
9
|
+
easeIn: 'ease-in',
|
|
10
|
+
easeOut: 'ease-out',
|
|
11
|
+
easeInOut: 'ease-in-out',
|
|
12
|
+
easeInBack: 'cubic-bezier(0.6, -0.28, 0.735, 0.045)',
|
|
13
|
+
easeOutBack: 'cubic-bezier(0.175, 0.885, 0.32, 1.275)',
|
|
14
|
+
easeInOutBack: 'cubic-bezier(0.68, -0.55, 0.265, 1.55)',
|
|
15
|
+
easeInElastic: 'cubic-bezier(0.5, -0.5, 0.75, 1.25)',
|
|
16
|
+
easeOutElastic: 'cubic-bezier(0.25, -0.25, 0.25, 1.5)',
|
|
17
|
+
easeOutBounce: 'cubic-bezier(0.34, 1.56, 0.64, 1)',
|
|
18
|
+
};
|
|
19
|
+
/** Mapeamento de duração para valores em ms */
|
|
20
|
+
export const BR_ANIMATION_DURATION_MAP = {
|
|
21
|
+
instant: 100,
|
|
22
|
+
fast: 150,
|
|
23
|
+
normal: 250,
|
|
24
|
+
slow: 400,
|
|
25
|
+
slower: 600,
|
|
26
|
+
};
|
|
27
|
+
/** Configuração padrão global */
|
|
28
|
+
export const BR_ANIMATION_DEFAULT_CONFIG = {
|
|
29
|
+
enabled: false, // Desabilitado por padrão
|
|
30
|
+
respectReducedMotion: true,
|
|
31
|
+
defaultDuration: 'normal',
|
|
32
|
+
defaultEasing: 'easeOut',
|
|
33
|
+
};
|
|
34
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"animation.types.js","sourceRoot":"","sources":["../../../../../projects/ng-govbr-tw/src/lib/animations/animation.types.ts"],"names":[],"mappings":"AAAA;;;GAGG;AA0FH,4CAA4C;AAC5C,MAAM,CAAC,MAAM,uBAAuB,GAAsC;IACxE,MAAM,EAAE,QAAQ;IAChB,IAAI,EAAE,MAAM;IACZ,MAAM,EAAE,SAAS;IACjB,OAAO,EAAE,UAAU;IACnB,SAAS,EAAE,aAAa;IACxB,UAAU,EAAE,wCAAwC;IACpD,WAAW,EAAE,yCAAyC;IACtD,aAAa,EAAE,wCAAwC;IACvD,aAAa,EAAE,qCAAqC;IACpD,cAAc,EAAE,sCAAsC;IACtD,aAAa,EAAE,mCAAmC;CAC1C,CAAC;AAKX,+CAA+C;AAC/C,MAAM,CAAC,MAAM,yBAAyB,GAAwC;IAC5E,OAAO,EAAE,GAAG;IACZ,IAAI,EAAE,GAAG;IACT,MAAM,EAAE,GAAG;IACX,IAAI,EAAE,GAAG;IACT,MAAM,EAAE,GAAG;CACH,CAAC;AAwCX,iCAAiC;AACjC,MAAM,CAAC,MAAM,2BAA2B,GAA4B;IAClE,OAAO,EAAE,KAAK,EAAE,0BAA0B;IAC1C,oBAAoB,EAAE,IAAI;IAC1B,eAAe,EAAE,QAAQ;IACzB,aAAa,EAAE,SAAS;CAChB,CAAC","sourcesContent":["/**\r\n * Tipos e interfaces para o sistema de animações Gov.br\r\n * @description Sistema de animações configurável e acessível\r\n */\r\n\r\n/** Tipos de animação de entrada */\r\nexport type BrAnimationEnter =\r\n  | 'fadeIn'\r\n  | 'fadeInUp'\r\n  | 'fadeInDown'\r\n  | 'fadeInLeft'\r\n  | 'fadeInRight'\r\n  | 'scaleIn'\r\n  | 'slideInUp'\r\n  | 'slideInDown'\r\n  | 'slideInLeft'\r\n  | 'slideInRight'\r\n  | 'expandHeight'\r\n  | 'expandWidth';\r\n\r\n/** Tipos de animação de saída */\r\nexport type BrAnimationLeave =\r\n  | 'fadeOut'\r\n  | 'fadeOutUp'\r\n  | 'fadeOutDown'\r\n  | 'fadeOutLeft'\r\n  | 'fadeOutRight'\r\n  | 'scaleOut'\r\n  | 'slideOutUp'\r\n  | 'slideOutDown'\r\n  | 'slideOutLeft'\r\n  | 'slideOutRight'\r\n  | 'collapseHeight'\r\n  | 'collapseWidth';\r\n\r\n/** Tipos de animação de atenção/feedback */\r\nexport type BrAnimationAttention =\r\n  | 'pulse'\r\n  | 'shake'\r\n  | 'bounce'\r\n  | 'wiggle'\r\n  | 'flash'\r\n  | 'heartbeat'\r\n  | 'swing';\r\n\r\n/** Tipos de animação de loading */\r\nexport type BrAnimationLoading =\r\n  | 'spin'\r\n  | 'dots'\r\n  | 'bars'\r\n  | 'shimmer'\r\n  | 'wave'\r\n  | 'progressIndeterminate';\r\n\r\n/** Tipos de animação de hover */\r\nexport type BrAnimationHover =\r\n  | 'lift'\r\n  | 'grow'\r\n  | 'shrink'\r\n  | 'glow'\r\n  | 'underline'\r\n  | 'ripple';\r\n\r\n/** Tipos de animação de sucesso/feedback */\r\nexport type BrAnimationFeedback =\r\n  | 'checkmark'\r\n  | 'successPop'\r\n  | 'errorShake'\r\n  | 'warningPulse';\r\n\r\n/** União de todos os tipos de animação */\r\nexport type BrAnimationType =\r\n  | BrAnimationEnter\r\n  | BrAnimationLeave\r\n  | BrAnimationAttention\r\n  | BrAnimationLoading\r\n  | BrAnimationHover\r\n  | BrAnimationFeedback;\r\n\r\n/** Curvas de easing disponíveis */\r\nexport type BrAnimationEasing =\r\n  | 'linear'\r\n  | 'ease'\r\n  | 'easeIn'\r\n  | 'easeOut'\r\n  | 'easeInOut'\r\n  | 'easeInBack'\r\n  | 'easeOutBack'\r\n  | 'easeInOutBack'\r\n  | 'easeInElastic'\r\n  | 'easeOutElastic'\r\n  | 'easeOutBounce';\r\n\r\n/** Mapeamento de easing para valores CSS */\r\nexport const BR_ANIMATION_EASING_MAP: Record<BrAnimationEasing, string> = {\r\n  linear: 'linear',\r\n  ease: 'ease',\r\n  easeIn: 'ease-in',\r\n  easeOut: 'ease-out',\r\n  easeInOut: 'ease-in-out',\r\n  easeInBack: 'cubic-bezier(0.6, -0.28, 0.735, 0.045)',\r\n  easeOutBack: 'cubic-bezier(0.175, 0.885, 0.32, 1.275)',\r\n  easeInOutBack: 'cubic-bezier(0.68, -0.55, 0.265, 1.55)',\r\n  easeInElastic: 'cubic-bezier(0.5, -0.5, 0.75, 1.25)',\r\n  easeOutElastic: 'cubic-bezier(0.25, -0.25, 0.25, 1.5)',\r\n  easeOutBounce: 'cubic-bezier(0.34, 1.56, 0.64, 1)',\r\n} as const;\r\n\r\n/** Durações predefinidas */\r\nexport type BrAnimationDuration = 'instant' | 'fast' | 'normal' | 'slow' | 'slower';\r\n\r\n/** Mapeamento de duração para valores em ms */\r\nexport const BR_ANIMATION_DURATION_MAP: Record<BrAnimationDuration, number> = {\r\n  instant: 100,\r\n  fast: 150,\r\n  normal: 250,\r\n  slow: 400,\r\n  slower: 600,\r\n} as const;\r\n\r\n/** Configuração de uma animação individual */\r\nexport interface BrAnimationConfig {\r\n  /** Tipo da animação */\r\n  type: BrAnimationType;\r\n  /** Duração em ms ou predefinida */\r\n  duration?: number | BrAnimationDuration;\r\n  /** Delay antes de iniciar em ms */\r\n  delay?: number;\r\n  /** Curva de easing */\r\n  easing?: BrAnimationEasing;\r\n  /** Número de iterações (Infinity para loop) */\r\n  iterations?: number | 'infinite';\r\n  /** Direção da animação */\r\n  direction?: 'normal' | 'reverse' | 'alternate' | 'alternate-reverse';\r\n  /** Estado final após animação */\r\n  fillMode?: 'none' | 'forwards' | 'backwards' | 'both';\r\n}\r\n\r\n/** Configuração de animação de entrada/saída */\r\nexport interface BrTransitionConfig {\r\n  /** Animação de entrada */\r\n  enter?: BrAnimationConfig;\r\n  /** Animação de saída */\r\n  leave?: BrAnimationConfig;\r\n}\r\n\r\n/** Configuração global de animações */\r\nexport interface BrAnimationGlobalConfig {\r\n  /** Habilita/desabilita todas as animações globalmente */\r\n  enabled: boolean;\r\n  /** Respeita preferência do usuário (prefers-reduced-motion) */\r\n  respectReducedMotion: boolean;\r\n  /** Duração padrão para todas as animações */\r\n  defaultDuration: number | BrAnimationDuration;\r\n  /** Easing padrão para todas as animações */\r\n  defaultEasing: BrAnimationEasing;\r\n}\r\n\r\n/** Configuração padrão global */\r\nexport const BR_ANIMATION_DEFAULT_CONFIG: BrAnimationGlobalConfig = {\r\n  enabled: false, // Desabilitado por padrão\r\n  respectReducedMotion: true,\r\n  defaultDuration: 'normal',\r\n  defaultEasing: 'easeOut',\r\n} as const;\r\n\r\n/** Estado de uma animação em execução */\r\nexport interface BrAnimationState {\r\n  /** Se está animando */\r\n  isAnimating: boolean;\r\n  /** Tipo atual */\r\n  currentAnimation: BrAnimationType | null;\r\n  /** Se foi concluída */\r\n  isCompleted: boolean;\r\n  /** Se foi cancelada */\r\n  isCancelled: boolean;\r\n}\r\n\r\n/** Evento emitido durante animação */\r\nexport interface BrAnimationEvent {\r\n  /** Tipo da animação */\r\n  type: BrAnimationType;\r\n  /** Fase: início, fim, cancelamento */\r\n  phase: 'start' | 'end' | 'cancel';\r\n  /** Elemento alvo */\r\n  element: HTMLElement;\r\n  /** Timestamp do evento */\r\n  timestamp: number;\r\n}\r\n\r\n/** Trigger para iniciar animação */\r\nexport type BrAnimationTrigger =\r\n  | 'immediate'     // Imediatamente ao montar\r\n  | 'visible'       // Quando visível no viewport (Intersection Observer)\r\n  | 'hover'         // No hover\r\n  | 'focus'         // No focus\r\n  | 'click'         // No click\r\n  | 'manual';       // Controlado programaticamente\r\n\r\n/** Configuração para diretiva brAnimate */\r\nexport interface BrAnimateDirectiveConfig {\r\n  /** Animação a executar */\r\n  animation: BrAnimationType;\r\n  /** Trigger que inicia a animação */\r\n  trigger?: BrAnimationTrigger;\r\n  /** Duração */\r\n  duration?: number | BrAnimationDuration;\r\n  /** Delay */\r\n  delay?: number;\r\n  /** Easing */\r\n  easing?: BrAnimationEasing;\r\n  /** Threshold para Intersection Observer (0-1) */\r\n  visibilityThreshold?: number;\r\n  /** Se deve animar apenas uma vez */\r\n  once?: boolean;\r\n}\r\n\r\n/** Configuração para stagger (animação em sequência) */\r\nexport interface BrStaggerConfig {\r\n  /** Animação base */\r\n  animation: BrAnimationType;\r\n  /** Delay entre cada item em ms */\r\n  staggerDelay: number;\r\n  /** Delay inicial antes do primeiro item */\r\n  initialDelay?: number;\r\n  /** Duração de cada animação */\r\n  duration?: number | BrAnimationDuration;\r\n  /** Easing */\r\n  easing?: BrAnimationEasing;\r\n}\r\n"]}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Sistema de Animações Gov.br
|
|
3
|
+
* @description Animações configuráveis e acessíveis para componentes Angular
|
|
4
|
+
*/
|
|
5
|
+
// Types e interfaces
|
|
6
|
+
export * from './animation.types';
|
|
7
|
+
// Serviço principal
|
|
8
|
+
export { BrAnimationService, BR_ANIMATION_CONFIG } from './animation.service';
|
|
9
|
+
// Diretiva de animação
|
|
10
|
+
export { BrAnimateDirective } from './animate.directive';
|
|
11
|
+
// Componentes de feedback
|
|
12
|
+
export { BrSuccessCheckComponent } from './success-check.component';
|
|
13
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1nb3Zici10dy9zcmMvbGliL2FuaW1hdGlvbnMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7OztHQUdHO0FBRUgscUJBQXFCO0FBQ3JCLGNBQWMsbUJBQW1CLENBQUM7QUFFbEMsb0JBQW9CO0FBQ3BCLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxtQkFBbUIsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBRTlFLHVCQUF1QjtBQUN2QixPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQztBQUV6RCwwQkFBMEI7QUFDMUIsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sMkJBQTJCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcclxuICogU2lzdGVtYSBkZSBBbmltYcOnw7VlcyBHb3YuYnJcclxuICogQGRlc2NyaXB0aW9uIEFuaW1hw6fDtWVzIGNvbmZpZ3Vyw6F2ZWlzIGUgYWNlc3PDrXZlaXMgcGFyYSBjb21wb25lbnRlcyBBbmd1bGFyXHJcbiAqL1xyXG5cclxuLy8gVHlwZXMgZSBpbnRlcmZhY2VzXHJcbmV4cG9ydCAqIGZyb20gJy4vYW5pbWF0aW9uLnR5cGVzJztcclxuXHJcbi8vIFNlcnZpw6dvIHByaW5jaXBhbFxyXG5leHBvcnQgeyBCckFuaW1hdGlvblNlcnZpY2UsIEJSX0FOSU1BVElPTl9DT05GSUcgfSBmcm9tICcuL2FuaW1hdGlvbi5zZXJ2aWNlJztcclxuXHJcbi8vIERpcmV0aXZhIGRlIGFuaW1hw6fDo29cclxuZXhwb3J0IHsgQnJBbmltYXRlRGlyZWN0aXZlIH0gZnJvbSAnLi9hbmltYXRlLmRpcmVjdGl2ZSc7XHJcblxyXG4vLyBDb21wb25lbnRlcyBkZSBmZWVkYmFja1xyXG5leHBvcnQgeyBCclN1Y2Nlc3NDaGVja0NvbXBvbmVudCB9IGZyb20gJy4vc3VjY2Vzcy1jaGVjay5jb21wb25lbnQnO1xyXG4iXX0=
|