@seniorsistemas/angular-components 19.0.9 → 19.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/breadcrumb/lib/breadcrumb/breadcrumb-models.d.ts +2 -0
- package/button/lib/button/button.component.d.ts +8 -2
- package/checkbox/lib/checkbox/checkbox.component.d.ts +5 -1
- package/dialog/lib/src/dialog/dialog.component.d.ts +7 -1
- package/esm2022/accordion/lib/accordion/components/accordion-panel/accordion-panel.component.mjs +2 -2
- package/esm2022/breadcrumb/lib/breadcrumb/breadcrumb-models.mjs +1 -1
- package/esm2022/breadcrumb/lib/breadcrumb/breadcrumb.component.mjs +3 -3
- package/esm2022/breadcrumb/lib/dynamic-breadcrumb/dynamic-breadcrumb.component.mjs +8 -6
- package/esm2022/button/lib/button/button.component.mjs +20 -8
- package/esm2022/checkbox/lib/checkbox/checkbox.component.mjs +8 -5
- package/esm2022/confirm-dialog/lib/popup-confirm-dialog/popup-confirm-dialog.component.mjs +2 -2
- package/esm2022/content-generator/lib/content-generator/content-generator.component.mjs +1 -1
- package/esm2022/dialog/lib/src/dialog/dialog.component.mjs +9 -3
- package/esm2022/dynamic-form/dynamic-form/components/grid/row/row.component.mjs +1 -1
- package/esm2022/dynamic-form/dynamic-form/components/lookup/lookup.component.mjs +2 -2
- package/esm2022/dynamic-form/dynamic-form/form-field/fields/bignumber/bignumber-field.component.mjs +1 -1
- package/esm2022/dynamic-form/dynamic-form/form-field/fields/boolean/boolean-radio-field/boolean-field.component.mjs +1 -1
- package/esm2022/dynamic-form/dynamic-form/form-field/fields/boolean/boolean-switch-field/boolean-switch-field.component.mjs +2 -2
- package/esm2022/dynamic-form/dynamic-form/form-field/fields/button-field/button-field.component.mjs +1 -1
- package/esm2022/dynamic-form/dynamic-form/form-field/fields/number/number-field.component.mjs +1 -1
- package/esm2022/dynamic-form/dynamic-form/form-field/fields/password/password-field.component.mjs +1 -1
- package/esm2022/dynamic-form/dynamic-form/form-field/fields/radio-button/radio-button-field.component.mjs +1 -1
- package/esm2022/dynamic-form/dynamic-form/form-field/fields/select/select-field.component.mjs +2 -2
- package/esm2022/dynamic-form/dynamic-form/form-field/fields/text/text-field.component.mjs +1 -1
- package/esm2022/empty-state/lib/empty-state/empty-state.component.mjs +1 -1
- package/esm2022/file-picker/lib/file-picker/file-picker.component.mjs +1 -1
- package/esm2022/global-search/lib/global-search/components/dropdown/global-search-dropdown-item.component.mjs +1 -1
- package/esm2022/global-search/lib/global-search/global-search.component.mjs +1 -1
- package/esm2022/help-popover/lib/help-popover/help-popover/help-popover.component.mjs +1 -1
- package/esm2022/image-cropper/lib/image-cropper/image-cropper.component.mjs +1 -1
- package/esm2022/inline-edit/lib/inline-edit/components/fields/inline-edit-number/inline-edit-number.component.mjs +1 -1
- package/esm2022/inline-edit/lib/inline-edit/components/inline-edit-item/inline-edit-item.component.mjs +1 -1
- package/esm2022/kanban/lib/kanban/components/kanban-column/kanban-column.component.mjs +1 -1
- package/esm2022/kanban/lib/kanban/components/kanban-item/kanban-item.component.mjs +1 -1
- package/esm2022/lib/locale/fallback.mjs +13 -2
- package/esm2022/navigation-button/lib/navigation-button/navigation-button.component.mjs +1 -1
- package/esm2022/object-card/lib/object-card/elements/field/object-card-field.component.mjs +1 -1
- package/esm2022/object-card/lib/object-card/elements/main/object-card-main.component.mjs +1 -1
- package/esm2022/paginator/lib/paginator/paginator.component.mjs +2 -2
- package/esm2022/picklist/lib/picklist/picklist.component.mjs +1 -1
- package/esm2022/profile-picture-picker/lib/profile-picture-picker/profile-picture-picker.component.mjs +1 -1
- package/esm2022/select/lib/select/components/select-option/select-option.component.mjs +3 -3
- package/esm2022/select/lib/select/select.component.mjs +132 -20
- package/esm2022/shared/lib/accessibility/paginator-accessibility.directive.mjs +182 -0
- package/esm2022/shared/lib/accessibility/table-checkbox-accessibility.directive.mjs +174 -0
- package/esm2022/shared/public-api.mjs +6 -4
- package/esm2022/switch/lib/switch/switch.component.mjs +19 -8
- package/esm2022/table/lib/table/table-paging/table-paging.component.mjs +1 -1
- package/esm2022/table/public-api.mjs +2 -1
- package/esm2022/tiered-menu/lib/tiered-menu/components/tiered-menu/tiered-menu.component.mjs +18 -8
- package/esm2022/tiered-menu/lib/tiered-menu/components/tiered-menu-item/tiered-menu-item.component.mjs +1 -1
- package/esm2022/tiered-menu/lib/tiered-menu/services/tiered-menu.service.mjs +1 -1
- package/esm2022/tiered-menu/lib/tiered-menu/tiered-menu.directive.mjs +11 -3
- package/esm2022/tooltip/lib/tooltip/tooltip.component.mjs +17 -11
- package/esm2022/tooltip/lib/tooltip/tooltip.directive.mjs +108 -241
- package/esm2022/tooltip/lib/tooltip/tooltip.module.mjs +4 -6
- package/esm2022/topbar/lib/topbar/topbar.component.mjs +2 -2
- package/fesm2022/seniorsistemas-angular-components-accordion.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-accordion.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-breadcrumb.mjs +9 -7
- package/fesm2022/seniorsistemas-angular-components-breadcrumb.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-button.mjs +19 -7
- package/fesm2022/seniorsistemas-angular-components-button.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-checkbox.mjs +7 -4
- package/fesm2022/seniorsistemas-angular-components-checkbox.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-confirm-dialog.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-confirm-dialog.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-content-generator.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-content-generator.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-dialog.mjs +8 -2
- package/fesm2022/seniorsistemas-angular-components-dialog.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-dynamic-form.mjs +11 -11
- package/fesm2022/seniorsistemas-angular-components-dynamic-form.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-empty-state.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-empty-state.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-file-picker.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-file-picker.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-global-search.mjs +2 -2
- package/fesm2022/seniorsistemas-angular-components-global-search.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-help-popover.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-help-popover.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-image-cropper.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-image-cropper.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-inline-edit.mjs +2 -2
- package/fesm2022/seniorsistemas-angular-components-inline-edit.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-kanban.mjs +2 -2
- package/fesm2022/seniorsistemas-angular-components-kanban.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-navigation-button.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-navigation-button.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-object-card.mjs +2 -2
- package/fesm2022/seniorsistemas-angular-components-object-card.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-paginator.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-paginator.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-picklist.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-picklist.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-profile-picture-picker.mjs +1 -1
- package/fesm2022/seniorsistemas-angular-components-profile-picture-picker.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-select.mjs +133 -21
- package/fesm2022/seniorsistemas-angular-components-select.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-shared.mjs +354 -2
- package/fesm2022/seniorsistemas-angular-components-shared.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-switch.mjs +18 -7
- package/fesm2022/seniorsistemas-angular-components-switch.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-table.mjs +2 -2
- package/fesm2022/seniorsistemas-angular-components-table.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-tiered-menu.mjs +26 -8
- package/fesm2022/seniorsistemas-angular-components-tiered-menu.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-tooltip.mjs +125 -254
- package/fesm2022/seniorsistemas-angular-components-tooltip.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-topbar.mjs +2 -2
- package/fesm2022/seniorsistemas-angular-components-topbar.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components.mjs +12 -1
- package/fesm2022/seniorsistemas-angular-components.mjs.map +1 -1
- package/package.json +1 -1
- package/select/lib/select/select.component.d.ts +23 -2
- package/shared/lib/accessibility/paginator-accessibility.directive.d.ts +30 -0
- package/shared/lib/accessibility/table-checkbox-accessibility.directive.d.ts +46 -0
- package/shared/public-api.d.ts +7 -5
- package/switch/lib/switch/switch.component.d.ts +9 -5
- package/table/public-api.d.ts +1 -0
- package/tiered-menu/lib/tiered-menu/components/tiered-menu/tiered-menu.component.d.ts +1 -0
- package/tiered-menu/lib/tiered-menu/tiered-menu.directive.d.ts +4 -2
- package/tooltip/lib/tooltip/tooltip.component.d.ts +13 -7
- package/tooltip/lib/tooltip/tooltip.directive.d.ts +27 -40
- package/tooltip/lib/tooltip/tooltip.module.d.ts +3 -4
|
@@ -1,11 +1,12 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import {
|
|
1
|
+
import { Overlay } from '@angular/cdk/overlay';
|
|
2
|
+
import { ComponentPortal } from '@angular/cdk/portal';
|
|
3
|
+
import { Directive, ElementRef, HostListener, Renderer2, inject, input, } from '@angular/core';
|
|
4
|
+
import { DebounceUtils } from '@seniorsistemas/angular-components/common';
|
|
3
5
|
import { TooltipComponent } from './tooltip.component';
|
|
4
6
|
import * as i0 from "@angular/core";
|
|
5
|
-
import * as i1 from "@seniorsistemas/angular-components/common";
|
|
6
7
|
/**
|
|
7
8
|
* @description Diretiva que exibe um tooltip (dica de ferramenta) flutuante ao interagir com o elemento hospedeiro.
|
|
8
|
-
* Posiciona-se automaticamente considerando o espaço disponível na tela.
|
|
9
|
+
* Posiciona-se automaticamente considerando o espaço disponível na tela, utilizando o CDK Overlay.
|
|
9
10
|
* Suporta eventos de hover, click e focus, além de comportamentos customizados para mobile.
|
|
10
11
|
*
|
|
11
12
|
* @example
|
|
@@ -16,33 +17,28 @@ import * as i1 from "@seniorsistemas/angular-components/common";
|
|
|
16
17
|
* @category Overlay
|
|
17
18
|
*/
|
|
18
19
|
export class TooltipDirective {
|
|
19
|
-
elementRef;
|
|
20
|
-
appRef;
|
|
21
|
-
debounceUtils;
|
|
22
|
-
renderer;
|
|
23
|
-
viewContainerRef;
|
|
24
20
|
/**
|
|
25
21
|
* @description Texto exibido no tooltip. Quando vazio ou `undefined`, o tooltip não é renderizado.
|
|
26
22
|
*/
|
|
27
|
-
tooltip;
|
|
23
|
+
tooltip = input(undefined, { alias: 'sTooltip' });
|
|
28
24
|
/**
|
|
29
25
|
* @description Posição preferencial do tooltip em relação ao elemento hospedeiro.
|
|
30
26
|
* O tooltip se reposiciona automaticamente se não houver espaço suficiente.
|
|
31
27
|
*
|
|
32
28
|
* @default 'top'
|
|
33
29
|
*/
|
|
34
|
-
tooltipPosition = 'top';
|
|
30
|
+
tooltipPosition = input('top');
|
|
35
31
|
/**
|
|
36
32
|
* @description Tempo em milissegundos antes do tooltip aparecer após o evento de ativação.
|
|
37
33
|
*
|
|
38
34
|
* @default 500
|
|
39
35
|
*/
|
|
40
|
-
showDelay = 500;
|
|
36
|
+
showDelay = input(500);
|
|
41
37
|
/**
|
|
42
38
|
* @description Tempo em milissegundos que o tooltip permanece visível após aparecer.
|
|
43
39
|
* Quando não informado, o tooltip permanece até o usuário sair do elemento.
|
|
44
40
|
*/
|
|
45
|
-
displayTime;
|
|
41
|
+
displayTime = input(undefined);
|
|
46
42
|
/**
|
|
47
43
|
* @description Evento do elemento hospedeiro que dispara a exibição do tooltip.
|
|
48
44
|
* - `'hover'`: exibe ao passar o mouse (padrão).
|
|
@@ -51,20 +47,20 @@ export class TooltipDirective {
|
|
|
51
47
|
*
|
|
52
48
|
* @default 'hover'
|
|
53
49
|
*/
|
|
54
|
-
tooltipEvent = 'hover';
|
|
50
|
+
tooltipEvent = input('hover');
|
|
55
51
|
/**
|
|
56
52
|
* @description Quando `true`, o conteúdo HTML do tooltip é escapado antes de ser renderizado.
|
|
57
53
|
*
|
|
58
54
|
* @default false
|
|
59
55
|
*/
|
|
60
|
-
escape = false;
|
|
56
|
+
escape = input(false);
|
|
61
57
|
/**
|
|
62
58
|
* @description Quando `false`, o tooltip nunca é exibido, mesmo que haja interação do usuário.
|
|
63
59
|
* Útil para desabilitar tooltips dinamicamente.
|
|
64
60
|
*
|
|
65
61
|
* @default true
|
|
66
62
|
*/
|
|
67
|
-
visible = true;
|
|
63
|
+
visible = input(true);
|
|
68
64
|
/**
|
|
69
65
|
* @description Comportamento do tooltip em dispositivos móveis (touch).
|
|
70
66
|
* - `'pressing'`: exibe ao pressionar e manter o toque (padrão).
|
|
@@ -72,11 +68,11 @@ export class TooltipDirective {
|
|
|
72
68
|
*
|
|
73
69
|
* @default 'pressing'
|
|
74
70
|
*/
|
|
75
|
-
mobileBehavior = 'pressing';
|
|
71
|
+
mobileBehavior = input('pressing');
|
|
76
72
|
/**
|
|
77
73
|
* @description Referência a um input HTML para sincronizar a exibição do tooltip com o foco no campo.
|
|
78
74
|
*/
|
|
79
|
-
focusedInputRef = null;
|
|
75
|
+
focusedInputRef = input(null);
|
|
80
76
|
/**
|
|
81
77
|
* @description Quando `true`, o tooltip só é exibido se o texto do elemento estiver truncado (overflow).
|
|
82
78
|
* Útil para cells de tabela ou elementos com `text-overflow: ellipsis`.
|
|
@@ -84,20 +80,17 @@ export class TooltipDirective {
|
|
|
84
80
|
* @default false
|
|
85
81
|
*/
|
|
86
82
|
checkTruncatedText = input(false);
|
|
83
|
+
elementRef = inject(ElementRef);
|
|
84
|
+
overlay = inject(Overlay);
|
|
85
|
+
debounceUtils = inject(DebounceUtils);
|
|
86
|
+
renderer = inject(Renderer2);
|
|
87
|
+
overlayRef = null;
|
|
87
88
|
componentRef = null;
|
|
89
|
+
positionChangeSubscription = null;
|
|
88
90
|
showTimeout;
|
|
89
91
|
touchTimeout;
|
|
90
|
-
tooltipDivElement = null;
|
|
91
|
-
debounceCreateTooltipFunction;
|
|
92
92
|
tooltipCreatedByFocus = false;
|
|
93
|
-
|
|
94
|
-
this.elementRef = elementRef;
|
|
95
|
-
this.appRef = appRef;
|
|
96
|
-
this.debounceUtils = debounceUtils;
|
|
97
|
-
this.renderer = renderer;
|
|
98
|
-
this.viewContainerRef = viewContainerRef;
|
|
99
|
-
this.debounceCreateTooltipFunction = this.debounceUtils.debounceLeading(() => this._createTooltip(true, true));
|
|
100
|
-
}
|
|
93
|
+
debounceCreateTooltipFunction = this.debounceUtils.debounceLeading(() => this._createTooltip(true, true));
|
|
101
94
|
ngOnInit() {
|
|
102
95
|
this.validatePosition();
|
|
103
96
|
this.updateTooltipVisibilityWhenFocusOnInput();
|
|
@@ -105,254 +98,147 @@ export class TooltipDirective {
|
|
|
105
98
|
ngOnDestroy() {
|
|
106
99
|
this.destroy();
|
|
107
100
|
}
|
|
108
|
-
/**
|
|
109
|
-
* Manipula a visibilidade do tooltip quando houver uma referência de input.
|
|
110
|
-
*/
|
|
111
|
-
updateTooltipVisibilityWhenFocusOnInput() {
|
|
112
|
-
if (this.focusedInputRef) {
|
|
113
|
-
const inputFocus = this.focusedInputRef;
|
|
114
|
-
const icon = this.getIconFromFocusedInput();
|
|
115
|
-
this.renderer.listen(inputFocus, 'focus', () => {
|
|
116
|
-
if (icon) {
|
|
117
|
-
this._createTooltip(false, false);
|
|
118
|
-
this.tooltipCreatedByFocus = true;
|
|
119
|
-
}
|
|
120
|
-
});
|
|
121
|
-
this.renderer.listen(inputFocus, 'blur', () => {
|
|
122
|
-
if (icon) {
|
|
123
|
-
this.removeTooltip(icon);
|
|
124
|
-
this.destroy();
|
|
125
|
-
this.tooltipCreatedByFocus = false;
|
|
126
|
-
}
|
|
127
|
-
});
|
|
128
|
-
}
|
|
129
|
-
}
|
|
130
|
-
/**
|
|
131
|
-
* Obtém o elemento do ícone associado ao label do input em focus.
|
|
132
|
-
* @returns O ícone do input em focus.
|
|
133
|
-
*/
|
|
134
|
-
getIconFromFocusedInput() {
|
|
135
|
-
return this.elementRef.nativeElement;
|
|
136
|
-
}
|
|
137
|
-
/**
|
|
138
|
-
* Remove o tooltip associado ao ícone do input em focus.
|
|
139
|
-
* @param icon O ícone do input em focus.
|
|
140
|
-
*/
|
|
141
|
-
removeTooltip(icon) {
|
|
142
|
-
const tooltip = icon.querySelector('.tooltip');
|
|
143
|
-
if (tooltip) {
|
|
144
|
-
tooltip.remove();
|
|
145
|
-
}
|
|
146
|
-
}
|
|
147
101
|
// whenever the component with the tooltip is clicked I destroy the tooltip.
|
|
148
102
|
// whenever a key is pressed on the component with the tooltip I destroy the tooltip.
|
|
149
103
|
onClick() {
|
|
150
|
-
if (this.tooltipEvent === 'hover' &&
|
|
104
|
+
if (this.tooltipEvent() === 'hover' &&
|
|
151
105
|
!navigator.userAgent.match(/Android/i) &&
|
|
152
106
|
!navigator.userAgent.match(/iPhone/i)) {
|
|
153
107
|
this.destroy();
|
|
154
108
|
}
|
|
155
109
|
}
|
|
156
110
|
onFocus() {
|
|
157
|
-
if (this.tooltipEvent === 'focus') {
|
|
111
|
+
if (this.tooltipEvent() === 'focus') {
|
|
158
112
|
this._createTooltip(true, true);
|
|
159
113
|
}
|
|
160
114
|
}
|
|
161
115
|
onBlur() {
|
|
162
|
-
if (this.tooltipEvent === 'focus') {
|
|
116
|
+
if (this.tooltipEvent() === 'focus') {
|
|
163
117
|
this.destroy();
|
|
164
118
|
}
|
|
165
119
|
}
|
|
166
120
|
onMouseEnter() {
|
|
167
|
-
if (this.tooltipEvent === 'hover' || this.focusedInputRef) {
|
|
121
|
+
if (this.tooltipEvent() === 'hover' || this.focusedInputRef()) {
|
|
168
122
|
this.debounceCreateTooltipFunction();
|
|
169
123
|
}
|
|
170
124
|
}
|
|
171
125
|
onMouseLeave() {
|
|
172
|
-
if ((this.tooltipEvent === 'hover' || this.focusedInputRef) && !this.tooltipCreatedByFocus) {
|
|
126
|
+
if ((this.tooltipEvent() === 'hover' || this.focusedInputRef()) && !this.tooltipCreatedByFocus) {
|
|
173
127
|
this.destroy();
|
|
174
128
|
}
|
|
175
129
|
}
|
|
176
130
|
onTouchStart() {
|
|
177
|
-
if (this.tooltipEvent === 'hover') {
|
|
131
|
+
if (this.tooltipEvent() === 'hover') {
|
|
178
132
|
window.clearTimeout(this.touchTimeout);
|
|
179
|
-
const
|
|
133
|
+
const delay = this.mobileBehavior() === 'pressing' ? this.showDelay() : 0;
|
|
180
134
|
this.touchTimeout = window.setTimeout(() => {
|
|
181
135
|
this._createTooltip(true, true);
|
|
182
|
-
},
|
|
136
|
+
}, delay);
|
|
183
137
|
}
|
|
184
138
|
}
|
|
185
139
|
onTouchEnd() {
|
|
186
|
-
if (this.tooltipEvent === 'hover') {
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
this.destroy();
|
|
190
|
-
}
|
|
140
|
+
if (this.tooltipEvent() === 'hover' && this.mobileBehavior() === 'pressing') {
|
|
141
|
+
window.clearTimeout(this.touchTimeout);
|
|
142
|
+
this.destroy();
|
|
191
143
|
}
|
|
192
144
|
}
|
|
193
145
|
validatePosition() {
|
|
194
|
-
const containsPosition = ['top', 'right', 'bottom', 'left'].includes(this.tooltipPosition);
|
|
146
|
+
const containsPosition = ['top', 'right', 'bottom', 'left'].includes(this.tooltipPosition());
|
|
195
147
|
if (!containsPosition) {
|
|
196
|
-
this.tooltipPosition
|
|
197
|
-
throw new Error(`Tooltip ${this.tooltipPosition} position is unexpected`);
|
|
148
|
+
throw new Error(`Tooltip position "${this.tooltipPosition()}" is unexpected`);
|
|
198
149
|
}
|
|
199
150
|
}
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
151
|
+
/**
|
|
152
|
+
* Manipula a visibilidade do tooltip quando houver uma referência de input.
|
|
153
|
+
*/
|
|
154
|
+
updateTooltipVisibilityWhenFocusOnInput() {
|
|
155
|
+
const inputRef = this.focusedInputRef();
|
|
156
|
+
if (!inputRef)
|
|
157
|
+
return;
|
|
158
|
+
this.renderer.listen(inputRef, 'focus', () => {
|
|
159
|
+
this._createTooltip(false, false);
|
|
160
|
+
this.tooltipCreatedByFocus = true;
|
|
161
|
+
});
|
|
162
|
+
this.renderer.listen(inputRef, 'blur', () => {
|
|
163
|
+
this.destroy();
|
|
164
|
+
this.tooltipCreatedByFocus = false;
|
|
165
|
+
});
|
|
203
166
|
}
|
|
204
167
|
_createTooltip(useShowDelay = false, autoDestroy = true) {
|
|
205
|
-
const
|
|
206
|
-
|
|
207
|
-
if (!hasText || hasComponentRef) {
|
|
168
|
+
const tooltipText = this.tooltip() ?? '';
|
|
169
|
+
if (!tooltipText.length || this.overlayRef !== null)
|
|
208
170
|
return;
|
|
209
|
-
|
|
210
|
-
const blockIfIsNotTruncatedContent = this.checkTruncatedText() ? !this.contentIsTruncated : false;
|
|
211
|
-
if (blockIfIsNotTruncatedContent) {
|
|
171
|
+
if (this.checkTruncatedText() && !this.contentIsTruncated)
|
|
212
172
|
return;
|
|
213
|
-
|
|
214
|
-
const
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
this.
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
this.
|
|
173
|
+
const positionStrategy = this.buildPositionStrategy();
|
|
174
|
+
const scrollStrategy = this.overlay.scrollStrategies.reposition();
|
|
175
|
+
this.overlayRef = this.overlay.create({ positionStrategy, scrollStrategy });
|
|
176
|
+
const portal = new ComponentPortal(TooltipComponent);
|
|
177
|
+
this.componentRef = this.overlayRef.attach(portal);
|
|
178
|
+
this.componentRef.setInput('tooltip', tooltipText);
|
|
179
|
+
this.componentRef.setInput('escape', this.escape());
|
|
180
|
+
this.componentRef.setInput('position', this.tooltipPosition());
|
|
181
|
+
this.positionChangeSubscription = positionStrategy.positionChanges.subscribe((change) => {
|
|
182
|
+
const mappedPosition = this.mapCdkPositionToTooltipPosition(change.connectionPair);
|
|
183
|
+
this.componentRef?.setInput('position', mappedPosition);
|
|
224
184
|
});
|
|
225
|
-
|
|
226
|
-
|
|
185
|
+
const delay = useShowDelay ? this.showDelay() : 0;
|
|
186
|
+
this.showTimeout = window.setTimeout(() => {
|
|
187
|
+
this.componentRef?.setInput('visible', this.visible());
|
|
188
|
+
}, delay);
|
|
189
|
+
if (autoDestroy && this.displayTime()) {
|
|
190
|
+
window.setTimeout(() => {
|
|
227
191
|
this.destroy();
|
|
228
|
-
}, this.displayTime);
|
|
192
|
+
}, this.displayTime());
|
|
229
193
|
}
|
|
230
194
|
}
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
195
|
+
buildPositionStrategy() {
|
|
196
|
+
const preferred = this.tooltipPosition();
|
|
197
|
+
const allPositions = ['top', 'bottom', 'left', 'right'];
|
|
198
|
+
const orderedPositions = [preferred, ...allPositions.filter((p) => p !== preferred)];
|
|
199
|
+
return this.overlay
|
|
200
|
+
.position()
|
|
201
|
+
.flexibleConnectedTo(this.elementRef)
|
|
202
|
+
.withPositions(orderedPositions.map((p) => this.getConnectedPosition(p)))
|
|
203
|
+
.withFlexibleDimensions(false)
|
|
204
|
+
.withPush(false);
|
|
236
205
|
}
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
const elementLeftover = getElementLeftoverContentAvailable(this.elementRef.nativeElement);
|
|
245
|
-
const elementTriggerPosition = getElementPositionInfo(this.elementRef.nativeElement);
|
|
246
|
-
const horizontalPositions = ['left', 'right'];
|
|
247
|
-
const isPositionHorizontal = horizontalPositions.includes(this.tooltipPosition);
|
|
248
|
-
const distanceToBorders = isPositionHorizontal
|
|
249
|
-
? tooltipElementPosition.elementWidth
|
|
250
|
-
: tooltipElementPosition.elementHorizontalCenter;
|
|
251
|
-
const canSetAtTop = elementLeftover.freeDistanceToTop > tooltipElementPosition.elementHeight;
|
|
252
|
-
const canSetAtLeft = elementLeftover.freeDistanceToLeft > distanceToBorders;
|
|
253
|
-
const canSetAtRight = elementLeftover.freeDistanceToRight > distanceToBorders;
|
|
254
|
-
const canSetAtBottom = elementLeftover.freeDistanceToBottom > tooltipElementPosition.elementHeight;
|
|
255
|
-
const topVertical = elementTriggerPosition.top + elementTriggerPosition.elementVerticalCenter;
|
|
256
|
-
const checkTopPosition = () => {
|
|
257
|
-
return canSetAtTop && canSetAtLeft && canSetAtRight;
|
|
206
|
+
getConnectedPosition(pos) {
|
|
207
|
+
const offset = 10;
|
|
208
|
+
const positionMap = {
|
|
209
|
+
top: { originX: 'center', originY: 'top', overlayX: 'center', overlayY: 'bottom', offsetY: -offset },
|
|
210
|
+
bottom: { originX: 'center', originY: 'bottom', overlayX: 'center', overlayY: 'top', offsetY: offset },
|
|
211
|
+
left: { originX: 'start', originY: 'center', overlayX: 'end', overlayY: 'center', offsetX: -offset },
|
|
212
|
+
right: { originX: 'end', originY: 'center', overlayX: 'start', overlayY: 'center', offsetX: offset },
|
|
258
213
|
};
|
|
259
|
-
|
|
260
|
-
return canSetAtBottom && canSetAtLeft && canSetAtRight;
|
|
261
|
-
};
|
|
262
|
-
const checkLeftPosition = () => {
|
|
263
|
-
return canSetAtLeft && canSetAtTop && canSetAtBottom;
|
|
264
|
-
};
|
|
265
|
-
const checkRightPosition = () => {
|
|
266
|
-
return canSetAtRight && canSetAtTop && canSetAtBottom;
|
|
267
|
-
};
|
|
268
|
-
const setTopPosition = () => {
|
|
269
|
-
return {
|
|
270
|
-
left: left + (right - left) / 2,
|
|
271
|
-
top: elementTriggerPosition.top - anchorSize,
|
|
272
|
-
};
|
|
273
|
-
};
|
|
274
|
-
const setBottomPosition = () => {
|
|
275
|
-
return {
|
|
276
|
-
left: left + (right - left) / 2,
|
|
277
|
-
top: elementTriggerPosition.top + elementTriggerPosition.elementHeight,
|
|
278
|
-
};
|
|
279
|
-
};
|
|
280
|
-
const setLeftPosition = () => {
|
|
281
|
-
return {
|
|
282
|
-
left: left,
|
|
283
|
-
top: topVertical,
|
|
284
|
-
};
|
|
285
|
-
};
|
|
286
|
-
const setRightPosition = () => {
|
|
287
|
-
return {
|
|
288
|
-
left: right,
|
|
289
|
-
top: topVertical,
|
|
290
|
-
};
|
|
291
|
-
};
|
|
292
|
-
const positions = [
|
|
293
|
-
{
|
|
294
|
-
pos: 'top',
|
|
295
|
-
check: checkTopPosition,
|
|
296
|
-
set: setTopPosition,
|
|
297
|
-
},
|
|
298
|
-
{
|
|
299
|
-
pos: 'bottom',
|
|
300
|
-
check: checkBottomPosition,
|
|
301
|
-
set: setBottomPosition,
|
|
302
|
-
},
|
|
303
|
-
{
|
|
304
|
-
pos: 'left',
|
|
305
|
-
check: checkLeftPosition,
|
|
306
|
-
set: setLeftPosition,
|
|
307
|
-
},
|
|
308
|
-
{
|
|
309
|
-
pos: 'right',
|
|
310
|
-
check: checkRightPosition,
|
|
311
|
-
set: setRightPosition,
|
|
312
|
-
},
|
|
313
|
-
];
|
|
314
|
-
const positionDefined = positions.find((position) => position.pos === this.tooltipPosition);
|
|
315
|
-
let validPosition = positionDefined;
|
|
316
|
-
const canUseDefinedPosition = positionDefined?.check();
|
|
317
|
-
if (canUseDefinedPosition) {
|
|
318
|
-
validPosition = positionDefined;
|
|
319
|
-
}
|
|
320
|
-
else if (!canUseDefinedPosition) {
|
|
321
|
-
const validAlternativePosition = positions.find((position) => position.check());
|
|
322
|
-
if (validAlternativePosition) {
|
|
323
|
-
validPosition = validAlternativePosition;
|
|
324
|
-
}
|
|
325
|
-
}
|
|
326
|
-
const _position = validPosition.set();
|
|
327
|
-
const positionLeft = _position.left;
|
|
328
|
-
const positionTop = _position.top;
|
|
329
|
-
this.componentRef.instance.left = Math.round(positionLeft);
|
|
330
|
-
this.componentRef.instance.top = Math.round(positionTop);
|
|
331
|
-
this.componentRef.instance.position = validPosition.pos;
|
|
214
|
+
return positionMap[pos];
|
|
332
215
|
}
|
|
333
|
-
|
|
334
|
-
if (
|
|
335
|
-
return;
|
|
336
|
-
|
|
337
|
-
|
|
216
|
+
mapCdkPositionToTooltipPosition(pair) {
|
|
217
|
+
if (pair.overlayY === 'bottom')
|
|
218
|
+
return 'top';
|
|
219
|
+
if (pair.overlayY === 'top')
|
|
220
|
+
return 'bottom';
|
|
221
|
+
if (pair.overlayX === 'end')
|
|
222
|
+
return 'left';
|
|
223
|
+
return 'right';
|
|
338
224
|
}
|
|
339
225
|
destroy() {
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
226
|
+
window.clearTimeout(this.showTimeout);
|
|
227
|
+
window.clearTimeout(this.touchTimeout);
|
|
228
|
+
this.positionChangeSubscription?.unsubscribe();
|
|
229
|
+
this.positionChangeSubscription = null;
|
|
230
|
+
if (this.overlayRef) {
|
|
231
|
+
this.overlayRef.dispose();
|
|
232
|
+
this.overlayRef = null;
|
|
344
233
|
this.componentRef = null;
|
|
345
|
-
this.tooltipDivElement = null;
|
|
346
234
|
}
|
|
347
235
|
}
|
|
348
236
|
get contentIsTruncated() {
|
|
349
237
|
const element = this.elementRef.nativeElement;
|
|
350
|
-
|
|
351
|
-
const truncatedByHorizontal = element.scrollWidth > element.clientWidth;
|
|
352
|
-
return truncatedByHorizontal || truncatedByVertical;
|
|
238
|
+
return element.scrollHeight > element.clientHeight || element.scrollWidth > element.clientWidth;
|
|
353
239
|
}
|
|
354
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TooltipDirective, deps: [
|
|
355
|
-
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.14", type: TooltipDirective, isStandalone: true, selector: "[sTooltip]", inputs: { tooltip: { classPropertyName: "tooltip", publicName: "sTooltip", isSignal:
|
|
240
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TooltipDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
241
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.14", type: TooltipDirective, isStandalone: true, selector: "[sTooltip]", inputs: { tooltip: { classPropertyName: "tooltip", publicName: "sTooltip", isSignal: true, isRequired: false, transformFunction: null }, tooltipPosition: { classPropertyName: "tooltipPosition", publicName: "tooltipPosition", isSignal: true, isRequired: false, transformFunction: null }, showDelay: { classPropertyName: "showDelay", publicName: "showDelay", isSignal: true, isRequired: false, transformFunction: null }, displayTime: { classPropertyName: "displayTime", publicName: "displayTime", isSignal: true, isRequired: false, transformFunction: null }, tooltipEvent: { classPropertyName: "tooltipEvent", publicName: "tooltipEvent", isSignal: true, isRequired: false, transformFunction: null }, escape: { classPropertyName: "escape", publicName: "escape", isSignal: true, isRequired: false, transformFunction: null }, visible: { classPropertyName: "visible", publicName: "visible", isSignal: true, isRequired: false, transformFunction: null }, mobileBehavior: { classPropertyName: "mobileBehavior", publicName: "mobileBehavior", isSignal: true, isRequired: false, transformFunction: null }, focusedInputRef: { classPropertyName: "focusedInputRef", publicName: "focusedInputRef", isSignal: true, isRequired: false, transformFunction: null }, checkTruncatedText: { classPropertyName: "checkTruncatedText", publicName: "checkTruncatedText", isSignal: true, isRequired: false, transformFunction: null } }, host: { listeners: { "click": "onClick()", "keydown": "onClick()", "focus": "onFocus()", "blur": "onBlur()", "mouseenter": "onMouseEnter()", "mouseleave": "onMouseLeave()", "wheel": "onMouseLeave()", "touchstart": "onTouchStart()", "touchend": "onTouchEnd()" } }, ngImport: i0 });
|
|
356
242
|
}
|
|
357
243
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TooltipDirective, decorators: [{
|
|
358
244
|
type: Directive,
|
|
@@ -360,26 +246,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
360
246
|
selector: '[sTooltip]',
|
|
361
247
|
standalone: true,
|
|
362
248
|
}]
|
|
363
|
-
}],
|
|
364
|
-
type: Input,
|
|
365
|
-
args: ['sTooltip']
|
|
366
|
-
}], tooltipPosition: [{
|
|
367
|
-
type: Input
|
|
368
|
-
}], showDelay: [{
|
|
369
|
-
type: Input
|
|
370
|
-
}], displayTime: [{
|
|
371
|
-
type: Input
|
|
372
|
-
}], tooltipEvent: [{
|
|
373
|
-
type: Input
|
|
374
|
-
}], escape: [{
|
|
375
|
-
type: Input
|
|
376
|
-
}], visible: [{
|
|
377
|
-
type: Input
|
|
378
|
-
}], mobileBehavior: [{
|
|
379
|
-
type: Input
|
|
380
|
-
}], focusedInputRef: [{
|
|
381
|
-
type: Input
|
|
382
|
-
}], onClick: [{
|
|
249
|
+
}], propDecorators: { onClick: [{
|
|
383
250
|
type: HostListener,
|
|
384
251
|
args: ['click']
|
|
385
252
|
}, {
|
|
@@ -407,4 +274,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
407
274
|
type: HostListener,
|
|
408
275
|
args: ['touchend']
|
|
409
276
|
}] } });
|
|
410
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip.directive.js","sourceRoot":"","sources":["../../../../../projects/angular-components/tooltip/src/lib/tooltip/tooltip.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    ApplicationRef,\n    ComponentRef,\n    Directive,\n    ElementRef,\n    HostListener,\n    input,\n    Input,\n    OnDestroy,\n    OnInit,\n    Renderer2,\n    ViewContainerRef,\n} from '@angular/core';\nimport { DebounceUtils } from '@seniorsistemas/angular-components/common';\nimport {\n    getElementLeftoverContentAvailable,\n    getElementPositionInfo,\n    isNullOrUndefined,\n} from '@seniorsistemas/angular-components/utils';\nimport { MobileBehavior } from './models/mobile-behavior';\nimport { TooltipEvent } from './models/tooltip-event';\nimport { TooltipPositions } from './models/tooltip-position';\nimport { TooltipComponent } from './tooltip.component';\n\n/**\n * @description Diretiva que exibe um tooltip (dica de ferramenta) flutuante ao interagir com o elemento hospedeiro.\n * Posiciona-se automaticamente considerando o espaço disponível na tela.\n * Suporta eventos de hover, click e focus, além de comportamentos customizados para mobile.\n *\n * @example\n * ```html\n * <button sTooltip=\"Clique aqui\" tooltipPosition=\"top\">Ação</button>\n * ```\n *\n * @category Overlay\n */\n@Directive({\n    selector: '[sTooltip]',\n    standalone: true,\n})\nexport class TooltipDirective implements OnInit, OnDestroy {\n    /**\n     * @description Texto exibido no tooltip. Quando vazio ou `undefined`, o tooltip não é renderizado.\n     */\n    @Input('sTooltip')\n    public tooltip?: string;\n\n    /**\n     * @description Posição preferencial do tooltip em relação ao elemento hospedeiro.\n     * O tooltip se reposiciona automaticamente se não houver espaço suficiente.\n     *\n     * @default 'top'\n     */\n    @Input()\n    public tooltipPosition: TooltipPositions = 'top';\n\n    /**\n     * @description Tempo em milissegundos antes do tooltip aparecer após o evento de ativação.\n     *\n     * @default 500\n     */\n    @Input()\n    public showDelay = 500;\n\n    /**\n     * @description Tempo em milissegundos que o tooltip permanece visível após aparecer.\n     * Quando não informado, o tooltip permanece até o usuário sair do elemento.\n     */\n    @Input()\n    public displayTime?: number;\n\n    /**\n     * @description Evento do elemento hospedeiro que dispara a exibição do tooltip.\n     * - `'hover'`: exibe ao passar o mouse (padrão).\n     * - `'click'`: exibe ao clicar.\n     * - `'focus'`: exibe ao focar no elemento.\n     *\n     * @default 'hover'\n     */\n    @Input()\n    public tooltipEvent: TooltipEvent = 'hover';\n\n    /**\n     * @description Quando `true`, o conteúdo HTML do tooltip é escapado antes de ser renderizado.\n     *\n     * @default false\n     */\n    @Input()\n    public escape = false;\n\n    /**\n     * @description Quando `false`, o tooltip nunca é exibido, mesmo que haja interação do usuário.\n     * Útil para desabilitar tooltips dinamicamente.\n     *\n     * @default true\n     */\n    @Input()\n    public visible = true;\n\n    /**\n     * @description Comportamento do tooltip em dispositivos móveis (touch).\n     * - `'pressing'`: exibe ao pressionar e manter o toque (padrão).\n     * - `'none'`: desabilita o tooltip em dispositivos táteis.\n     *\n     * @default 'pressing'\n     */\n    @Input()\n    public mobileBehavior: MobileBehavior = 'pressing';\n\n    /**\n     * @description Referência a um input HTML para sincronizar a exibição do tooltip com o foco no campo.\n     */\n    @Input()\n    public focusedInputRef: HTMLInputElement | null = null;\n\n    /**\n     * @description Quando `true`, o tooltip só é exibido se o texto do elemento estiver truncado (overflow).\n     * Útil para cells de tabela ou elementos com `text-overflow: ellipsis`.\n     *\n     * @default false\n     */\n    checkTruncatedText = input(false);\n\n    private componentRef: ComponentRef<TooltipComponent> | null = null;\n    private showTimeout?: number;\n    private touchTimeout?: number;\n    private tooltipDivElement: HTMLElement | null = null;\n    private readonly debounceCreateTooltipFunction: () => void;\n    private tooltipCreatedByFocus = false;\n\n    constructor(\n        private readonly elementRef: ElementRef,\n        private readonly appRef: ApplicationRef,\n        private readonly debounceUtils: DebounceUtils,\n        private readonly renderer: Renderer2,\n        private readonly viewContainerRef: ViewContainerRef,\n    ) {\n        this.debounceCreateTooltipFunction = this.debounceUtils.debounceLeading(() => this._createTooltip(true, true));\n    }\n\n    public ngOnInit(): void {\n        this.validatePosition();\n        this.updateTooltipVisibilityWhenFocusOnInput();\n    }\n\n    public ngOnDestroy(): void {\n        this.destroy();\n    }\n\n    /**\n     * Manipula a visibilidade do tooltip quando houver uma referência de input.\n     */\n    private updateTooltipVisibilityWhenFocusOnInput() {\n        if (this.focusedInputRef) {\n            const inputFocus = this.focusedInputRef;\n            const icon = this.getIconFromFocusedInput();\n            this.renderer.listen(inputFocus, 'focus', () => {\n                if (icon) {\n                    this._createTooltip(false, false);\n                    this.tooltipCreatedByFocus = true;\n                }\n            });\n            this.renderer.listen(inputFocus, 'blur', () => {\n                if (icon) {\n                    this.removeTooltip(icon);\n                    this.destroy();\n                    this.tooltipCreatedByFocus = false;\n                }\n            });\n        }\n    }\n\n    /**\n     * Obtém o elemento do ícone associado ao label do input em focus.\n     * @returns O ícone do input em focus.\n     */\n    private getIconFromFocusedInput(): HTMLElement {\n        return this.elementRef.nativeElement;\n    }\n\n    /**\n     * Remove o tooltip associado ao ícone do input em focus.\n     * @param icon O ícone do input em focus.\n     */\n    private removeTooltip(icon: HTMLElement): void {\n        const tooltip = icon.querySelector('.tooltip') as HTMLElement;\n        if (tooltip) {\n            tooltip.remove();\n        }\n    }\n\n    // whenever the component with the tooltip is clicked I destroy the tooltip.\n    // whenever a key is pressed on the component with the tooltip I destroy the tooltip.\n    @HostListener('click')\n    @HostListener('keydown')\n    public onClick(): void {\n        if (\n            this.tooltipEvent === 'hover' &&\n            !navigator.userAgent.match(/Android/i) &&\n            !navigator.userAgent.match(/iPhone/i)\n        ) {\n            this.destroy();\n        }\n    }\n\n    @HostListener('focus')\n    public onFocus(): void {\n        if (this.tooltipEvent === 'focus') {\n            this._createTooltip(true, true);\n        }\n    }\n\n    @HostListener('blur')\n    public onBlur(): void {\n        if (this.tooltipEvent === 'focus') {\n            this.destroy();\n        }\n    }\n\n    @HostListener('mouseenter')\n    public onMouseEnter(): void {\n        if (this.tooltipEvent === 'hover' || this.focusedInputRef) {\n            this.debounceCreateTooltipFunction();\n        }\n    }\n\n    @HostListener('mouseleave')\n    @HostListener('wheel')\n    public onMouseLeave(): void {\n        if ((this.tooltipEvent === 'hover' || this.focusedInputRef) && !this.tooltipCreatedByFocus) {\n            this.destroy();\n        }\n    }\n\n    @HostListener('touchstart')\n    public onTouchStart(): void {\n        if (this.tooltipEvent === 'hover') {\n            window.clearTimeout(this.touchTimeout);\n            const _delay = this.mobileBehavior === 'pressing' ? this.showDelay : 0;\n            this.touchTimeout = window.setTimeout(() => {\n                this._createTooltip(true, true);\n            }, _delay);\n        }\n    }\n\n    @HostListener('touchend')\n    public onTouchEnd(): void {\n        if (this.tooltipEvent === 'hover') {\n            if (this.mobileBehavior === 'pressing') {\n                window.clearTimeout(this.touchTimeout);\n                this.destroy();\n            }\n        }\n    }\n\n    private validatePosition() {\n        const containsPosition = ['top', 'right', 'bottom', 'left'].includes(this.tooltipPosition);\n\n        if (!containsPosition) {\n            this.tooltipPosition = 'top';\n            throw new Error(`Tooltip ${this.tooltipPosition} position is unexpected`);\n        }\n    }\n\n    private getDomElement(): HTMLElement {\n        this.componentRef = this.viewContainerRef.createComponent(TooltipComponent);\n        return this.componentRef.location.nativeElement;\n    }\n\n    private _createTooltip(useShowDelay = false, autoDestroy = true) {\n        const hasText = (this.tooltip || '').length > 0;\n        const hasComponentRef = this.componentRef !== null;\n        if (!hasText || hasComponentRef) {\n            return;\n        }\n        const blockIfIsNotTruncatedContent = this.checkTruncatedText() ? !this.contentIsTruncated : false;\n        if (blockIfIsNotTruncatedContent) {\n            return;\n        }\n        const domElement = this.getDomElement();\n        document.body.appendChild(domElement);\n        this.tooltipDivElement = domElement.querySelector('.tooltip');\n        this.setTooltipComponentProperties();\n        const _showDelay = useShowDelay ? this.showDelay : 0;\n        setTimeout(() => {\n            this.showTooltip();\n        }, _showDelay);\n\n        this.renderer.listen(window, 'scroll', () => {\n            this.setTooltipPositionProperty();\n        });\n\n        if (autoDestroy && this.displayTime) {\n            setTimeout(() => {\n                this.destroy();\n            }, this.displayTime);\n        }\n    }\n\n    private showTooltip() {\n        if (this.componentRef) {\n            this.componentRef.instance.visible = this.visible;\n\n            this.setTooltipPositionProperty();\n        }\n    }\n\n    private setTooltipPositionProperty() {\n        if (!this.componentRef) {\n            return;\n        }\n        const anchorSize = 5;\n        const { left, right } = this.elementRef.nativeElement.getBoundingClientRect();\n\n        const tooltipElementPosition = getElementPositionInfo(this.tooltipDivElement!);\n        const elementLeftover = getElementLeftoverContentAvailable(this.elementRef.nativeElement);\n        const elementTriggerPosition = getElementPositionInfo(this.elementRef.nativeElement);\n        const horizontalPositions: TooltipPositions[] = ['left', 'right'];\n        const isPositionHorizontal = horizontalPositions.includes(this.tooltipPosition);\n        const distanceToBorders = isPositionHorizontal\n            ? tooltipElementPosition.elementWidth\n            : tooltipElementPosition.elementHorizontalCenter;\n        const canSetAtTop = elementLeftover.freeDistanceToTop > tooltipElementPosition.elementHeight;\n        const canSetAtLeft = elementLeftover.freeDistanceToLeft > distanceToBorders;\n        const canSetAtRight = elementLeftover.freeDistanceToRight > distanceToBorders;\n        const canSetAtBottom = elementLeftover.freeDistanceToBottom > tooltipElementPosition.elementHeight;\n        const topVertical = elementTriggerPosition.top + elementTriggerPosition.elementVerticalCenter;\n        const checkTopPosition = () => {\n            return canSetAtTop && canSetAtLeft && canSetAtRight;\n        };\n        const checkBottomPosition = () => {\n            return canSetAtBottom && canSetAtLeft && canSetAtRight;\n        };\n        const checkLeftPosition = () => {\n            return canSetAtLeft && canSetAtTop && canSetAtBottom;\n        };\n\n        const checkRightPosition = () => {\n            return canSetAtRight && canSetAtTop && canSetAtBottom;\n        };\n\n        const setTopPosition = (): { left: number; top: number } => {\n            return {\n                left: left + (right - left) / 2,\n                top: elementTriggerPosition.top - anchorSize,\n            };\n        };\n\n        const setBottomPosition = (): { left: number; top: number } => {\n            return {\n                left: left + (right - left) / 2,\n                top: elementTriggerPosition.top + elementTriggerPosition.elementHeight,\n            };\n        };\n\n        const setLeftPosition = (): { left: number; top: number } => {\n            return {\n                left: left,\n                top: topVertical,\n            };\n        };\n\n        const setRightPosition = (): { left: number; top: number } => {\n            return {\n                left: right,\n                top: topVertical,\n            };\n        };\n\n        type Position = {\n            pos: TooltipPositions;\n            check: () => boolean;\n            set: () => { left: number; top: number };\n        };\n\n        const positions: Position[] = [\n            {\n                pos: 'top',\n                check: checkTopPosition,\n                set: setTopPosition,\n            },\n            {\n                pos: 'bottom',\n                check: checkBottomPosition,\n                set: setBottomPosition,\n            },\n            {\n                pos: 'left',\n                check: checkLeftPosition,\n                set: setLeftPosition,\n            },\n            {\n                pos: 'right',\n                check: checkRightPosition,\n                set: setRightPosition,\n            },\n        ];\n\n        const positionDefined = positions.find((position) => position.pos === this.tooltipPosition);\n        let validPosition: Position | undefined = positionDefined;\n        const canUseDefinedPosition = positionDefined?.check();\n        if (canUseDefinedPosition) {\n            validPosition = positionDefined;\n        } else if (!canUseDefinedPosition) {\n            const validAlternativePosition = positions.find((position) => position.check());\n            if (validAlternativePosition) {\n                validPosition = validAlternativePosition;\n            }\n        }\n        const _position = validPosition!.set();\n        const positionLeft = _position.left;\n        const positionTop = _position.top;\n        this.componentRef.instance.left = Math.round(positionLeft);\n        this.componentRef.instance.top = Math.round(positionTop);\n        this.componentRef.instance.position = validPosition!.pos;\n    }\n\n    private setTooltipComponentProperties() {\n        if (!this.componentRef) return;\n\n        this.componentRef.instance.tooltip = this.tooltip as string;\n        this.componentRef.instance.escape = this.escape;\n    }\n\n    private destroy(): void {\n        if (!isNullOrUndefined(this.componentRef)) {\n            window.clearTimeout(this.showTimeout);\n            this.appRef.detachView(this.componentRef.hostView);\n            this.componentRef.destroy();\n            this.componentRef = null;\n            this.tooltipDivElement = null;\n        }\n    }\n\n    private get contentIsTruncated() {\n        const element = this.elementRef.nativeElement as HTMLElement;\n        const truncatedByVertical = element.scrollHeight > element.clientHeight;\n        const truncatedByHorizontal = element.scrollWidth > element.clientWidth;\n        return truncatedByHorizontal || truncatedByVertical;\n    }\n}\n\n"]}
|
|
277
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip.directive.js","sourceRoot":"","sources":["../../../../../projects/angular-components/tooltip/src/lib/tooltip/tooltip.directive.ts"],"names":[],"mappings":"AAAA,OAAO,EAAwD,OAAO,EAAc,MAAM,sBAAsB,CAAC;AACjH,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAEH,SAAS,EACT,UAAU,EACV,YAAY,EAGZ,SAAS,EACT,MAAM,EACN,KAAK,GACR,MAAM,eAAe,CAAC;AAEvB,OAAO,EAAE,aAAa,EAAE,MAAM,2CAA2C,CAAC;AAI1E,OAAO,EAAE,gBAAgB,EAAE,MAAM,qBAAqB,CAAC;;AAEvD;;;;;;;;;;;GAWG;AAKH,MAAM,OAAO,gBAAgB;IACzB;;OAEG;IACa,OAAO,GAAG,KAAK,CAAqB,SAAS,EAAE,EAAE,KAAK,EAAE,UAAU,EAAE,CAAC,CAAC;IAEtF;;;;;OAKG;IACa,eAAe,GAAG,KAAK,CAAmB,KAAK,CAAC,CAAC;IAEjE;;;;OAIG;IACa,SAAS,GAAG,KAAK,CAAS,GAAG,CAAC,CAAC;IAE/C;;;OAGG;IACa,WAAW,GAAG,KAAK,CAAqB,SAAS,CAAC,CAAC;IAEnE;;;;;;;OAOG;IACa,YAAY,GAAG,KAAK,CAAe,OAAO,CAAC,CAAC;IAE5D;;;;OAIG;IACa,MAAM,GAAG,KAAK,CAAU,KAAK,CAAC,CAAC;IAE/C;;;;;OAKG;IACa,OAAO,GAAG,KAAK,CAAU,IAAI,CAAC,CAAC;IAE/C;;;;;;OAMG;IACa,cAAc,GAAG,KAAK,CAAiB,UAAU,CAAC,CAAC;IAEnE;;OAEG;IACa,eAAe,GAAG,KAAK,CAA0B,IAAI,CAAC,CAAC;IAEvE;;;;;OAKG;IACa,kBAAkB,GAAG,KAAK,CAAU,KAAK,CAAC,CAAC;IAE1C,UAAU,GAAG,MAAM,CAA0B,UAAU,CAAC,CAAC;IACzD,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,CAAC;IAC1B,aAAa,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;IACtC,QAAQ,GAAG,MAAM,CAAC,SAAS,CAAC,CAAC;IAEtC,UAAU,GAAsB,IAAI,CAAC;IACrC,YAAY,GAA0C,IAAI,CAAC;IAC3D,0BAA0B,GAAwB,IAAI,CAAC;IACvD,WAAW,CAAU;IACrB,YAAY,CAAU;IACtB,qBAAqB,GAAG,KAAK,CAAC;IACrB,6BAA6B,GAAG,IAAI,CAAC,aAAa,CAAC,eAAe,CAAC,GAAG,EAAE,CACrF,IAAI,CAAC,cAAc,CAAC,IAAI,EAAE,IAAI,CAAC,CAClC,CAAC;IAEK,QAAQ;QACX,IAAI,CAAC,gBAAgB,EAAE,CAAC;QACxB,IAAI,CAAC,uCAAuC,EAAE,CAAC;IACnD,CAAC;IAEM,WAAW;QACd,IAAI,CAAC,OAAO,EAAE,CAAC;IACnB,CAAC;IAED,4EAA4E;IAC5E,qFAAqF;IAG9E,OAAO;QACV,IACI,IAAI,CAAC,YAAY,EAAE,KAAK,OAAO;YAC/B,CAAC,SAAS,CAAC,SAAS,CAAC,KAAK,CAAC,UAAU,CAAC;YACtC,CAAC,SAAS,CAAC,SAAS,CAAC,KAAK,CAAC,SAAS,CAAC,EACvC,CAAC;YACC,IAAI,CAAC,OAAO,EAAE,CAAC;QACnB,CAAC;IACL,CAAC;IAGM,OAAO;QACV,IAAI,IAAI,CAAC,YAAY,EAAE,KAAK,OAAO,EAAE,CAAC;YAClC,IAAI,CAAC,cAAc,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;QACpC,CAAC;IACL,CAAC;IAGM,MAAM;QACT,IAAI,IAAI,CAAC,YAAY,EAAE,KAAK,OAAO,EAAE,CAAC;YAClC,IAAI,CAAC,OAAO,EAAE,CAAC;QACnB,CAAC;IACL,CAAC;IAGM,YAAY;QACf,IAAI,IAAI,CAAC,YAAY,EAAE,KAAK,OAAO,IAAI,IAAI,CAAC,eAAe,EAAE,EAAE,CAAC;YAC5D,IAAI,CAAC,6BAA6B,EAAE,CAAC;QACzC,CAAC;IACL,CAAC;IAIM,YAAY;QACf,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,KAAK,OAAO,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,qBAAqB,EAAE,CAAC;YAC7F,IAAI,CAAC,OAAO,EAAE,CAAC;QACnB,CAAC;IACL,CAAC;IAGM,YAAY;QACf,IAAI,IAAI,CAAC,YAAY,EAAE,KAAK,OAAO,EAAE,CAAC;YAClC,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YACvC,MAAM,KAAK,GAAG,IAAI,CAAC,cAAc,EAAE,KAAK,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;YAC1E,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,UAAU,CAAC,GAAG,EAAE;gBACvC,IAAI,CAAC,cAAc,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;YACpC,CAAC,EAAE,KAAK,CAAC,CAAC;QACd,CAAC;IACL,CAAC;IAGM,UAAU;QACb,IAAI,IAAI,CAAC,YAAY,EAAE,KAAK,OAAO,IAAI,IAAI,CAAC,cAAc,EAAE,KAAK,UAAU,EAAE,CAAC;YAC1E,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YACvC,IAAI,CAAC,OAAO,EAAE,CAAC;QACnB,CAAC;IACL,CAAC;IAEO,gBAAgB;QACpB,MAAM,gBAAgB,GAAG,CAAC,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC;QAC7F,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACpB,MAAM,IAAI,KAAK,CAAC,qBAAqB,IAAI,CAAC,eAAe,EAAE,iBAAiB,CAAC,CAAC;QAClF,CAAC;IACL,CAAC;IAED;;OAEG;IACK,uCAAuC;QAC3C,MAAM,QAAQ,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;QACxC,IAAI,CAAC,QAAQ;YAAE,OAAO;QAEtB,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,QAAQ,EAAE,OAAO,EAAE,GAAG,EAAE;YACzC,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;YAClC,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC;QACtC,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,QAAQ,EAAE,MAAM,EAAE,GAAG,EAAE;YACxC,IAAI,CAAC,OAAO,EAAE,CAAC;YACf,IAAI,CAAC,qBAAqB,GAAG,KAAK,CAAC;QACvC,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,cAAc,CAAC,YAAY,GAAG,KAAK,EAAE,WAAW,GAAG,IAAI;QAC3D,MAAM,WAAW,GAAG,IAAI,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC;QACzC,IAAI,CAAC,WAAW,CAAC,MAAM,IAAI,IAAI,CAAC,UAAU,KAAK,IAAI;YAAE,OAAO;QAE5D,IAAI,IAAI,CAAC,kBAAkB,EAAE,IAAI,CAAC,IAAI,CAAC,kBAAkB;YAAE,OAAO;QAElE,MAAM,gBAAgB,GAAG,IAAI,CAAC,qBAAqB,EAAE,CAAC;QACtD,MAAM,cAAc,GAAG,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,UAAU,EAAE,CAAC;QAElE,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,gBAAgB,EAAE,cAAc,EAAE,CAAC,CAAC;QAE5E,MAAM,MAAM,GAAG,IAAI,eAAe,CAAC,gBAAgB,CAAC,CAAC;QACrD,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QAEnD,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC;QACnD,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,QAAQ,EAAE,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;QACpD,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,UAAU,EAAE,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC;QAE/D,IAAI,CAAC,0BAA0B,GAAG,gBAAgB,CAAC,eAAe,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE,EAAE;YACpF,MAAM,cAAc,GAAG,IAAI,CAAC,+BAA+B,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC;YACnF,IAAI,CAAC,YAAY,EAAE,QAAQ,CAAC,UAAU,EAAE,cAAc,CAAC,CAAC;QAC5D,CAAC,CAAC,CAAC;QAEH,MAAM,KAAK,GAAG,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QAClD,IAAI,CAAC,WAAW,GAAG,MAAM,CAAC,UAAU,CAAC,GAAG,EAAE;YACtC,IAAI,CAAC,YAAY,EAAE,QAAQ,CAAC,SAAS,EAAE,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;QAC3D,CAAC,EAAE,KAAK,CAAC,CAAC;QAEV,IAAI,WAAW,IAAI,IAAI,CAAC,WAAW,EAAE,EAAE,CAAC;YACpC,MAAM,CAAC,UAAU,CAAC,GAAG,EAAE;gBACnB,IAAI,CAAC,OAAO,EAAE,CAAC;YACnB,CAAC,EAAE,IAAI,CAAC,WAAW,EAAG,CAAC,CAAC;QAC5B,CAAC;IACL,CAAC;IAEO,qBAAqB;QACzB,MAAM,SAAS,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;QACzC,MAAM,YAAY,GAAuB,CAAC,KAAK,EAAE,QAAQ,EAAE,MAAM,EAAE,OAAO,CAAC,CAAC;QAC5E,MAAM,gBAAgB,GAAG,CAAC,SAAS,EAAE,GAAG,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,SAAS,CAAC,CAAC,CAAC;QAErF,OAAO,IAAI,CAAC,OAAO;aACd,QAAQ,EAAE;aACV,mBAAmB,CAAC,IAAI,CAAC,UAAU,CAAC;aACpC,aAAa,CAAC,gBAAgB,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC,CAAC,CAAC,CAAC;aACxE,sBAAsB,CAAC,KAAK,CAAC;aAC7B,QAAQ,CAAC,KAAK,CAAC,CAAC;IACzB,CAAC;IAEO,oBAAoB,CAAC,GAAqB;QAC9C,MAAM,MAAM,GAAG,EAAE,CAAC;QAClB,MAAM,WAAW,GAAgD;YAC7D,GAAG,EAAE,EAAE,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC,MAAM,EAAE;YACpG,MAAM,EAAE,EAAE,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,OAAO,EAAE,MAAM,EAAE;YACtG,IAAI,EAAE,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC,MAAM,EAAE;YACpG,KAAK,EAAE,EAAE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,EAAE;SACvG,CAAC;QACF,OAAO,WAAW,CAAC,GAAG,CAAC,CAAC;IAC5B,CAAC;IAEO,+BAA+B,CAAC,IAA4C;QAChF,IAAI,IAAI,CAAC,QAAQ,KAAK,QAAQ;YAAE,OAAO,KAAK,CAAC;QAC7C,IAAI,IAAI,CAAC,QAAQ,KAAK,KAAK;YAAE,OAAO,QAAQ,CAAC;QAC7C,IAAI,IAAI,CAAC,QAAQ,KAAK,KAAK;YAAE,OAAO,MAAM,CAAC;QAC3C,OAAO,OAAO,CAAC;IACnB,CAAC;IAEO,OAAO;QACX,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACtC,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QACvC,IAAI,CAAC,0BAA0B,EAAE,WAAW,EAAE,CAAC;QAC/C,IAAI,CAAC,0BAA0B,GAAG,IAAI,CAAC;QACvC,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YAClB,IAAI,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;YAC1B,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;YACvB,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;QAC7B,CAAC;IACL,CAAC;IAED,IAAY,kBAAkB;QAC1B,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,CAAC,aAA4B,CAAC;QAC7D,OAAO,OAAO,CAAC,YAAY,GAAG,OAAO,CAAC,YAAY,IAAI,OAAO,CAAC,WAAW,GAAG,OAAO,CAAC,WAAW,CAAC;IACpG,CAAC;wGAzQQ,gBAAgB;4FAAhB,gBAAgB;;4FAAhB,gBAAgB;kBAJ5B,SAAS;mBAAC;oBACP,QAAQ,EAAE,YAAY;oBACtB,UAAU,EAAE,IAAI;iBACnB;8BAuGU,OAAO;sBAFb,YAAY;uBAAC,OAAO;;sBACpB,YAAY;uBAAC,SAAS;gBAYhB,OAAO;sBADb,YAAY;uBAAC,OAAO;gBAQd,MAAM;sBADZ,YAAY;uBAAC,MAAM;gBAQb,YAAY;sBADlB,YAAY;uBAAC,YAAY;gBASnB,YAAY;sBAFlB,YAAY;uBAAC,YAAY;;sBACzB,YAAY;uBAAC,OAAO;gBAQd,YAAY;sBADlB,YAAY;uBAAC,YAAY;gBAYnB,UAAU;sBADhB,YAAY;uBAAC,UAAU","sourcesContent":["import { ConnectedPosition, FlexibleConnectedPositionStrategy, Overlay, OverlayRef } from '@angular/cdk/overlay';\nimport { ComponentPortal } from '@angular/cdk/portal';\nimport {\n    ComponentRef,\n    Directive,\n    ElementRef,\n    HostListener,\n    OnDestroy,\n    OnInit,\n    Renderer2,\n    inject,\n    input,\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { DebounceUtils } from '@seniorsistemas/angular-components/common';\nimport { MobileBehavior } from './models/mobile-behavior';\nimport { TooltipEvent } from './models/tooltip-event';\nimport { TooltipPositions } from './models/tooltip-position';\nimport { TooltipComponent } from './tooltip.component';\n\n/**\n * @description Diretiva que exibe um tooltip (dica de ferramenta) flutuante ao interagir com o elemento hospedeiro.\n * Posiciona-se automaticamente considerando o espaço disponível na tela, utilizando o CDK Overlay.\n * Suporta eventos de hover, click e focus, além de comportamentos customizados para mobile.\n *\n * @example\n * ```html\n * <button sTooltip=\"Clique aqui\" tooltipPosition=\"top\">Ação</button>\n * ```\n *\n * @category Overlay\n */\n@Directive({\n    selector: '[sTooltip]',\n    standalone: true,\n})\nexport class TooltipDirective implements OnInit, OnDestroy {\n    /**\n     * @description Texto exibido no tooltip. Quando vazio ou `undefined`, o tooltip não é renderizado.\n     */\n    public readonly tooltip = input<string | undefined>(undefined, { alias: 'sTooltip' });\n\n    /**\n     * @description Posição preferencial do tooltip em relação ao elemento hospedeiro.\n     * O tooltip se reposiciona automaticamente se não houver espaço suficiente.\n     *\n     * @default 'top'\n     */\n    public readonly tooltipPosition = input<TooltipPositions>('top');\n\n    /**\n     * @description Tempo em milissegundos antes do tooltip aparecer após o evento de ativação.\n     *\n     * @default 500\n     */\n    public readonly showDelay = input<number>(500);\n\n    /**\n     * @description Tempo em milissegundos que o tooltip permanece visível após aparecer.\n     * Quando não informado, o tooltip permanece até o usuário sair do elemento.\n     */\n    public readonly displayTime = input<number | undefined>(undefined);\n\n    /**\n     * @description Evento do elemento hospedeiro que dispara a exibição do tooltip.\n     * - `'hover'`: exibe ao passar o mouse (padrão).\n     * - `'click'`: exibe ao clicar.\n     * - `'focus'`: exibe ao focar no elemento.\n     *\n     * @default 'hover'\n     */\n    public readonly tooltipEvent = input<TooltipEvent>('hover');\n\n    /**\n     * @description Quando `true`, o conteúdo HTML do tooltip é escapado antes de ser renderizado.\n     *\n     * @default false\n     */\n    public readonly escape = input<boolean>(false);\n\n    /**\n     * @description Quando `false`, o tooltip nunca é exibido, mesmo que haja interação do usuário.\n     * Útil para desabilitar tooltips dinamicamente.\n     *\n     * @default true\n     */\n    public readonly visible = input<boolean>(true);\n\n    /**\n     * @description Comportamento do tooltip em dispositivos móveis (touch).\n     * - `'pressing'`: exibe ao pressionar e manter o toque (padrão).\n     * - `'none'`: desabilita o tooltip em dispositivos táteis.\n     *\n     * @default 'pressing'\n     */\n    public readonly mobileBehavior = input<MobileBehavior>('pressing');\n\n    /**\n     * @description Referência a um input HTML para sincronizar a exibição do tooltip com o foco no campo.\n     */\n    public readonly focusedInputRef = input<HTMLInputElement | null>(null);\n\n    /**\n     * @description Quando `true`, o tooltip só é exibido se o texto do elemento estiver truncado (overflow).\n     * Útil para cells de tabela ou elementos com `text-overflow: ellipsis`.\n     *\n     * @default false\n     */\n    public readonly checkTruncatedText = input<boolean>(false);\n\n    private readonly elementRef = inject<ElementRef<HTMLElement>>(ElementRef);\n    private readonly overlay = inject(Overlay);\n    private readonly debounceUtils = inject(DebounceUtils);\n    private readonly renderer = inject(Renderer2);\n\n    private overlayRef: OverlayRef | null = null;\n    private componentRef: ComponentRef<TooltipComponent> | null = null;\n    private positionChangeSubscription: Subscription | null = null;\n    private showTimeout?: number;\n    private touchTimeout?: number;\n    private tooltipCreatedByFocus = false;\n    private readonly debounceCreateTooltipFunction = this.debounceUtils.debounceLeading(() =>\n        this._createTooltip(true, true),\n    );\n\n    public ngOnInit(): void {\n        this.validatePosition();\n        this.updateTooltipVisibilityWhenFocusOnInput();\n    }\n\n    public ngOnDestroy(): void {\n        this.destroy();\n    }\n\n    // whenever the component with the tooltip is clicked I destroy the tooltip.\n    // whenever a key is pressed on the component with the tooltip I destroy the tooltip.\n    @HostListener('click')\n    @HostListener('keydown')\n    public onClick(): void {\n        if (\n            this.tooltipEvent() === 'hover' &&\n            !navigator.userAgent.match(/Android/i) &&\n            !navigator.userAgent.match(/iPhone/i)\n        ) {\n            this.destroy();\n        }\n    }\n\n    @HostListener('focus')\n    public onFocus(): void {\n        if (this.tooltipEvent() === 'focus') {\n            this._createTooltip(true, true);\n        }\n    }\n\n    @HostListener('blur')\n    public onBlur(): void {\n        if (this.tooltipEvent() === 'focus') {\n            this.destroy();\n        }\n    }\n\n    @HostListener('mouseenter')\n    public onMouseEnter(): void {\n        if (this.tooltipEvent() === 'hover' || this.focusedInputRef()) {\n            this.debounceCreateTooltipFunction();\n        }\n    }\n\n    @HostListener('mouseleave')\n    @HostListener('wheel')\n    public onMouseLeave(): void {\n        if ((this.tooltipEvent() === 'hover' || this.focusedInputRef()) && !this.tooltipCreatedByFocus) {\n            this.destroy();\n        }\n    }\n\n    @HostListener('touchstart')\n    public onTouchStart(): void {\n        if (this.tooltipEvent() === 'hover') {\n            window.clearTimeout(this.touchTimeout);\n            const delay = this.mobileBehavior() === 'pressing' ? this.showDelay() : 0;\n            this.touchTimeout = window.setTimeout(() => {\n                this._createTooltip(true, true);\n            }, delay);\n        }\n    }\n\n    @HostListener('touchend')\n    public onTouchEnd(): void {\n        if (this.tooltipEvent() === 'hover' && this.mobileBehavior() === 'pressing') {\n            window.clearTimeout(this.touchTimeout);\n            this.destroy();\n        }\n    }\n\n    private validatePosition(): void {\n        const containsPosition = ['top', 'right', 'bottom', 'left'].includes(this.tooltipPosition());\n        if (!containsPosition) {\n            throw new Error(`Tooltip position \"${this.tooltipPosition()}\" is unexpected`);\n        }\n    }\n\n    /**\n     * Manipula a visibilidade do tooltip quando houver uma referência de input.\n     */\n    private updateTooltipVisibilityWhenFocusOnInput(): void {\n        const inputRef = this.focusedInputRef();\n        if (!inputRef) return;\n\n        this.renderer.listen(inputRef, 'focus', () => {\n            this._createTooltip(false, false);\n            this.tooltipCreatedByFocus = true;\n        });\n        this.renderer.listen(inputRef, 'blur', () => {\n            this.destroy();\n            this.tooltipCreatedByFocus = false;\n        });\n    }\n\n    private _createTooltip(useShowDelay = false, autoDestroy = true): void {\n        const tooltipText = this.tooltip() ?? '';\n        if (!tooltipText.length || this.overlayRef !== null) return;\n\n        if (this.checkTruncatedText() && !this.contentIsTruncated) return;\n\n        const positionStrategy = this.buildPositionStrategy();\n        const scrollStrategy = this.overlay.scrollStrategies.reposition();\n\n        this.overlayRef = this.overlay.create({ positionStrategy, scrollStrategy });\n\n        const portal = new ComponentPortal(TooltipComponent);\n        this.componentRef = this.overlayRef.attach(portal);\n\n        this.componentRef.setInput('tooltip', tooltipText);\n        this.componentRef.setInput('escape', this.escape());\n        this.componentRef.setInput('position', this.tooltipPosition());\n\n        this.positionChangeSubscription = positionStrategy.positionChanges.subscribe((change) => {\n            const mappedPosition = this.mapCdkPositionToTooltipPosition(change.connectionPair);\n            this.componentRef?.setInput('position', mappedPosition);\n        });\n\n        const delay = useShowDelay ? this.showDelay() : 0;\n        this.showTimeout = window.setTimeout(() => {\n            this.componentRef?.setInput('visible', this.visible());\n        }, delay);\n\n        if (autoDestroy && this.displayTime()) {\n            window.setTimeout(() => {\n                this.destroy();\n            }, this.displayTime()!);\n        }\n    }\n\n    private buildPositionStrategy(): FlexibleConnectedPositionStrategy {\n        const preferred = this.tooltipPosition();\n        const allPositions: TooltipPositions[] = ['top', 'bottom', 'left', 'right'];\n        const orderedPositions = [preferred, ...allPositions.filter((p) => p !== preferred)];\n\n        return this.overlay\n            .position()\n            .flexibleConnectedTo(this.elementRef)\n            .withPositions(orderedPositions.map((p) => this.getConnectedPosition(p)))\n            .withFlexibleDimensions(false)\n            .withPush(false);\n    }\n\n    private getConnectedPosition(pos: TooltipPositions): ConnectedPosition {\n        const offset = 10;\n        const positionMap: Record<TooltipPositions, ConnectedPosition> = {\n            top: { originX: 'center', originY: 'top', overlayX: 'center', overlayY: 'bottom', offsetY: -offset },\n            bottom: { originX: 'center', originY: 'bottom', overlayX: 'center', overlayY: 'top', offsetY: offset },\n            left: { originX: 'start', originY: 'center', overlayX: 'end', overlayY: 'center', offsetX: -offset },\n            right: { originX: 'end', originY: 'center', overlayX: 'start', overlayY: 'center', offsetX: offset },\n        };\n        return positionMap[pos];\n    }\n\n    private mapCdkPositionToTooltipPosition(pair: { overlayX: string; overlayY: string }): TooltipPositions {\n        if (pair.overlayY === 'bottom') return 'top';\n        if (pair.overlayY === 'top') return 'bottom';\n        if (pair.overlayX === 'end') return 'left';\n        return 'right';\n    }\n\n    private destroy(): void {\n        window.clearTimeout(this.showTimeout);\n        window.clearTimeout(this.touchTimeout);\n        this.positionChangeSubscription?.unsubscribe();\n        this.positionChangeSubscription = null;\n        if (this.overlayRef) {\n            this.overlayRef.dispose();\n            this.overlayRef = null;\n            this.componentRef = null;\n        }\n    }\n\n    private get contentIsTruncated(): boolean {\n        const element = this.elementRef.nativeElement as HTMLElement;\n        return element.scrollHeight > element.clientHeight || element.scrollWidth > element.clientWidth;\n    }\n}\n\n"]}
|