@seniorsistemas/angular-components 19.0.9 → 19.1.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.
Files changed (123) hide show
  1. package/breadcrumb/lib/breadcrumb/breadcrumb-models.d.ts +2 -0
  2. package/button/lib/button/button.component.d.ts +8 -2
  3. package/checkbox/lib/checkbox/checkbox.component.d.ts +5 -1
  4. package/dialog/lib/src/dialog/dialog.component.d.ts +7 -1
  5. package/esm2022/accordion/lib/accordion/components/accordion-panel/accordion-panel.component.mjs +2 -2
  6. package/esm2022/breadcrumb/lib/breadcrumb/breadcrumb-models.mjs +1 -1
  7. package/esm2022/breadcrumb/lib/breadcrumb/breadcrumb.component.mjs +3 -3
  8. package/esm2022/breadcrumb/lib/dynamic-breadcrumb/dynamic-breadcrumb.component.mjs +8 -6
  9. package/esm2022/button/lib/button/button.component.mjs +20 -8
  10. package/esm2022/checkbox/lib/checkbox/checkbox.component.mjs +8 -5
  11. package/esm2022/confirm-dialog/lib/popup-confirm-dialog/popup-confirm-dialog.component.mjs +2 -2
  12. package/esm2022/content-generator/lib/content-generator/content-generator.component.mjs +1 -1
  13. package/esm2022/dialog/lib/src/dialog/dialog.component.mjs +9 -3
  14. package/esm2022/dynamic-form/dynamic-form/components/grid/row/row.component.mjs +1 -1
  15. package/esm2022/dynamic-form/dynamic-form/components/lookup/lookup.component.mjs +2 -2
  16. package/esm2022/dynamic-form/dynamic-form/form-field/fields/bignumber/bignumber-field.component.mjs +1 -1
  17. package/esm2022/dynamic-form/dynamic-form/form-field/fields/boolean/boolean-radio-field/boolean-field.component.mjs +1 -1
  18. package/esm2022/dynamic-form/dynamic-form/form-field/fields/boolean/boolean-switch-field/boolean-switch-field.component.mjs +2 -2
  19. package/esm2022/dynamic-form/dynamic-form/form-field/fields/button-field/button-field.component.mjs +1 -1
  20. package/esm2022/dynamic-form/dynamic-form/form-field/fields/number/number-field.component.mjs +1 -1
  21. package/esm2022/dynamic-form/dynamic-form/form-field/fields/password/password-field.component.mjs +1 -1
  22. package/esm2022/dynamic-form/dynamic-form/form-field/fields/radio-button/radio-button-field.component.mjs +1 -1
  23. package/esm2022/dynamic-form/dynamic-form/form-field/fields/select/select-field.component.mjs +2 -2
  24. package/esm2022/dynamic-form/dynamic-form/form-field/fields/text/text-field.component.mjs +1 -1
  25. package/esm2022/empty-state/lib/empty-state/empty-state.component.mjs +1 -1
  26. package/esm2022/file-picker/lib/file-picker/file-picker.component.mjs +1 -1
  27. package/esm2022/global-search/lib/global-search/components/dropdown/global-search-dropdown-item.component.mjs +1 -1
  28. package/esm2022/global-search/lib/global-search/global-search.component.mjs +1 -1
  29. package/esm2022/help-popover/lib/help-popover/help-popover/help-popover.component.mjs +1 -1
  30. package/esm2022/image-cropper/lib/image-cropper/image-cropper.component.mjs +1 -1
  31. package/esm2022/inline-edit/lib/inline-edit/components/fields/inline-edit-number/inline-edit-number.component.mjs +1 -1
  32. package/esm2022/inline-edit/lib/inline-edit/components/inline-edit-item/inline-edit-item.component.mjs +1 -1
  33. package/esm2022/kanban/lib/kanban/components/kanban-column/kanban-column.component.mjs +1 -1
  34. package/esm2022/kanban/lib/kanban/components/kanban-item/kanban-item.component.mjs +1 -1
  35. package/esm2022/lib/locale/fallback.mjs +13 -2
  36. package/esm2022/navigation-button/lib/navigation-button/navigation-button.component.mjs +1 -1
  37. package/esm2022/object-card/lib/object-card/elements/field/object-card-field.component.mjs +1 -1
  38. package/esm2022/object-card/lib/object-card/elements/main/object-card-main.component.mjs +1 -1
  39. package/esm2022/paginator/lib/paginator/paginator.component.mjs +2 -2
  40. package/esm2022/picklist/lib/picklist/picklist.component.mjs +1 -1
  41. package/esm2022/profile-picture-picker/lib/profile-picture-picker/profile-picture-picker.component.mjs +1 -1
  42. package/esm2022/select/lib/select/components/select-option/select-option.component.mjs +3 -3
  43. package/esm2022/select/lib/select/select.component.mjs +132 -20
  44. package/esm2022/shared/lib/accessibility/paginator-accessibility.directive.mjs +182 -0
  45. package/esm2022/shared/lib/accessibility/table-checkbox-accessibility.directive.mjs +174 -0
  46. package/esm2022/shared/public-api.mjs +6 -4
  47. package/esm2022/switch/lib/switch/switch.component.mjs +19 -8
  48. package/esm2022/table/lib/table/table-paging/table-paging.component.mjs +1 -1
  49. package/esm2022/tiered-menu/lib/tiered-menu/components/tiered-menu/tiered-menu.component.mjs +18 -8
  50. package/esm2022/tiered-menu/lib/tiered-menu/components/tiered-menu-item/tiered-menu-item.component.mjs +1 -1
  51. package/esm2022/tiered-menu/lib/tiered-menu/services/tiered-menu.service.mjs +1 -1
  52. package/esm2022/tiered-menu/lib/tiered-menu/tiered-menu.directive.mjs +11 -3
  53. package/esm2022/tooltip/lib/tooltip/tooltip.component.mjs +17 -11
  54. package/esm2022/tooltip/lib/tooltip/tooltip.directive.mjs +108 -241
  55. package/esm2022/tooltip/lib/tooltip/tooltip.module.mjs +4 -6
  56. package/esm2022/topbar/lib/topbar/topbar.component.mjs +2 -2
  57. package/fesm2022/seniorsistemas-angular-components-accordion.mjs +1 -1
  58. package/fesm2022/seniorsistemas-angular-components-accordion.mjs.map +1 -1
  59. package/fesm2022/seniorsistemas-angular-components-breadcrumb.mjs +9 -7
  60. package/fesm2022/seniorsistemas-angular-components-breadcrumb.mjs.map +1 -1
  61. package/fesm2022/seniorsistemas-angular-components-button.mjs +19 -7
  62. package/fesm2022/seniorsistemas-angular-components-button.mjs.map +1 -1
  63. package/fesm2022/seniorsistemas-angular-components-checkbox.mjs +7 -4
  64. package/fesm2022/seniorsistemas-angular-components-checkbox.mjs.map +1 -1
  65. package/fesm2022/seniorsistemas-angular-components-confirm-dialog.mjs +1 -1
  66. package/fesm2022/seniorsistemas-angular-components-confirm-dialog.mjs.map +1 -1
  67. package/fesm2022/seniorsistemas-angular-components-content-generator.mjs +1 -1
  68. package/fesm2022/seniorsistemas-angular-components-content-generator.mjs.map +1 -1
  69. package/fesm2022/seniorsistemas-angular-components-dialog.mjs +8 -2
  70. package/fesm2022/seniorsistemas-angular-components-dialog.mjs.map +1 -1
  71. package/fesm2022/seniorsistemas-angular-components-dynamic-form.mjs +11 -11
  72. package/fesm2022/seniorsistemas-angular-components-dynamic-form.mjs.map +1 -1
  73. package/fesm2022/seniorsistemas-angular-components-empty-state.mjs +1 -1
  74. package/fesm2022/seniorsistemas-angular-components-empty-state.mjs.map +1 -1
  75. package/fesm2022/seniorsistemas-angular-components-file-picker.mjs +1 -1
  76. package/fesm2022/seniorsistemas-angular-components-file-picker.mjs.map +1 -1
  77. package/fesm2022/seniorsistemas-angular-components-global-search.mjs +2 -2
  78. package/fesm2022/seniorsistemas-angular-components-global-search.mjs.map +1 -1
  79. package/fesm2022/seniorsistemas-angular-components-help-popover.mjs +1 -1
  80. package/fesm2022/seniorsistemas-angular-components-help-popover.mjs.map +1 -1
  81. package/fesm2022/seniorsistemas-angular-components-image-cropper.mjs +1 -1
  82. package/fesm2022/seniorsistemas-angular-components-image-cropper.mjs.map +1 -1
  83. package/fesm2022/seniorsistemas-angular-components-inline-edit.mjs +2 -2
  84. package/fesm2022/seniorsistemas-angular-components-inline-edit.mjs.map +1 -1
  85. package/fesm2022/seniorsistemas-angular-components-kanban.mjs +2 -2
  86. package/fesm2022/seniorsistemas-angular-components-kanban.mjs.map +1 -1
  87. package/fesm2022/seniorsistemas-angular-components-navigation-button.mjs +1 -1
  88. package/fesm2022/seniorsistemas-angular-components-navigation-button.mjs.map +1 -1
  89. package/fesm2022/seniorsistemas-angular-components-object-card.mjs +2 -2
  90. package/fesm2022/seniorsistemas-angular-components-object-card.mjs.map +1 -1
  91. package/fesm2022/seniorsistemas-angular-components-paginator.mjs +1 -1
  92. package/fesm2022/seniorsistemas-angular-components-paginator.mjs.map +1 -1
  93. package/fesm2022/seniorsistemas-angular-components-picklist.mjs +1 -1
  94. package/fesm2022/seniorsistemas-angular-components-picklist.mjs.map +1 -1
  95. package/fesm2022/seniorsistemas-angular-components-profile-picture-picker.mjs +1 -1
  96. package/fesm2022/seniorsistemas-angular-components-profile-picture-picker.mjs.map +1 -1
  97. package/fesm2022/seniorsistemas-angular-components-select.mjs +133 -21
  98. package/fesm2022/seniorsistemas-angular-components-select.mjs.map +1 -1
  99. package/fesm2022/seniorsistemas-angular-components-shared.mjs +354 -2
  100. package/fesm2022/seniorsistemas-angular-components-shared.mjs.map +1 -1
  101. package/fesm2022/seniorsistemas-angular-components-switch.mjs +18 -7
  102. package/fesm2022/seniorsistemas-angular-components-switch.mjs.map +1 -1
  103. package/fesm2022/seniorsistemas-angular-components-table.mjs +1 -1
  104. package/fesm2022/seniorsistemas-angular-components-table.mjs.map +1 -1
  105. package/fesm2022/seniorsistemas-angular-components-tiered-menu.mjs +26 -8
  106. package/fesm2022/seniorsistemas-angular-components-tiered-menu.mjs.map +1 -1
  107. package/fesm2022/seniorsistemas-angular-components-tooltip.mjs +125 -254
  108. package/fesm2022/seniorsistemas-angular-components-tooltip.mjs.map +1 -1
  109. package/fesm2022/seniorsistemas-angular-components-topbar.mjs +2 -2
  110. package/fesm2022/seniorsistemas-angular-components-topbar.mjs.map +1 -1
  111. package/fesm2022/seniorsistemas-angular-components.mjs +12 -1
  112. package/fesm2022/seniorsistemas-angular-components.mjs.map +1 -1
  113. package/package.json +1 -1
  114. package/select/lib/select/select.component.d.ts +23 -2
  115. package/shared/lib/accessibility/paginator-accessibility.directive.d.ts +30 -0
  116. package/shared/lib/accessibility/table-checkbox-accessibility.directive.d.ts +46 -0
  117. package/shared/public-api.d.ts +7 -5
  118. package/switch/lib/switch/switch.component.d.ts +9 -5
  119. package/tiered-menu/lib/tiered-menu/components/tiered-menu/tiered-menu.component.d.ts +1 -0
  120. package/tiered-menu/lib/tiered-menu/tiered-menu.directive.d.ts +4 -2
  121. package/tooltip/lib/tooltip/tooltip.component.d.ts +13 -7
  122. package/tooltip/lib/tooltip/tooltip.directive.d.ts +27 -40
  123. package/tooltip/lib/tooltip/tooltip.module.d.ts +3 -4
@@ -1,11 +1,12 @@
1
- import { Directive, HostListener, input, Input, } from '@angular/core';
2
- import { getElementLeftoverContentAvailable, getElementPositionInfo, isNullOrUndefined, } from '@seniorsistemas/angular-components/utils';
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
- constructor(elementRef, appRef, debounceUtils, renderer, viewContainerRef) {
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 _delay = this.mobileBehavior === 'pressing' ? this.showDelay : 0;
133
+ const delay = this.mobileBehavior() === 'pressing' ? this.showDelay() : 0;
180
134
  this.touchTimeout = window.setTimeout(() => {
181
135
  this._createTooltip(true, true);
182
- }, _delay);
136
+ }, delay);
183
137
  }
184
138
  }
185
139
  onTouchEnd() {
186
- if (this.tooltipEvent === 'hover') {
187
- if (this.mobileBehavior === 'pressing') {
188
- window.clearTimeout(this.touchTimeout);
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 = 'top';
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
- getDomElement() {
201
- this.componentRef = this.viewContainerRef.createComponent(TooltipComponent);
202
- return this.componentRef.location.nativeElement;
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 hasText = (this.tooltip || '').length > 0;
206
- const hasComponentRef = this.componentRef !== null;
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 domElement = this.getDomElement();
215
- document.body.appendChild(domElement);
216
- this.tooltipDivElement = domElement.querySelector('.tooltip');
217
- this.setTooltipComponentProperties();
218
- const _showDelay = useShowDelay ? this.showDelay : 0;
219
- setTimeout(() => {
220
- this.showTooltip();
221
- }, _showDelay);
222
- this.renderer.listen(window, 'scroll', () => {
223
- this.setTooltipPositionProperty();
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
- if (autoDestroy && this.displayTime) {
226
- setTimeout(() => {
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
- showTooltip() {
232
- if (this.componentRef) {
233
- this.componentRef.instance.visible = this.visible;
234
- this.setTooltipPositionProperty();
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
- setTooltipPositionProperty() {
238
- if (!this.componentRef) {
239
- return;
240
- }
241
- const anchorSize = 5;
242
- const { left, right } = this.elementRef.nativeElement.getBoundingClientRect();
243
- const tooltipElementPosition = getElementPositionInfo(this.tooltipDivElement);
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
- const checkBottomPosition = () => {
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
- setTooltipComponentProperties() {
334
- if (!this.componentRef)
335
- return;
336
- this.componentRef.instance.tooltip = this.tooltip;
337
- this.componentRef.instance.escape = this.escape;
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
- if (!isNullOrUndefined(this.componentRef)) {
341
- window.clearTimeout(this.showTimeout);
342
- this.appRef.detachView(this.componentRef.hostView);
343
- this.componentRef.destroy();
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
- const truncatedByVertical = element.scrollHeight > element.clientHeight;
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: [{ token: i0.ElementRef }, { token: i0.ApplicationRef }, { token: i1.DebounceUtils }, { token: i0.Renderer2 }, { token: i0.ViewContainerRef }], target: i0.ɵɵFactoryTarget.Directive });
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: false, isRequired: false, transformFunction: null }, tooltipPosition: { classPropertyName: "tooltipPosition", publicName: "tooltipPosition", isSignal: false, isRequired: false, transformFunction: null }, showDelay: { classPropertyName: "showDelay", publicName: "showDelay", isSignal: false, isRequired: false, transformFunction: null }, displayTime: { classPropertyName: "displayTime", publicName: "displayTime", isSignal: false, isRequired: false, transformFunction: null }, tooltipEvent: { classPropertyName: "tooltipEvent", publicName: "tooltipEvent", isSignal: false, isRequired: false, transformFunction: null }, escape: { classPropertyName: "escape", publicName: "escape", isSignal: false, isRequired: false, transformFunction: null }, visible: { classPropertyName: "visible", publicName: "visible", isSignal: false, isRequired: false, transformFunction: null }, mobileBehavior: { classPropertyName: "mobileBehavior", publicName: "mobileBehavior", isSignal: false, isRequired: false, transformFunction: null }, focusedInputRef: { classPropertyName: "focusedInputRef", publicName: "focusedInputRef", isSignal: false, 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 });
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
- }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i0.ApplicationRef }, { type: i1.DebounceUtils }, { type: i0.Renderer2 }, { type: i0.ViewContainerRef }], propDecorators: { tooltip: [{
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":"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","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"]}