@tacdaed/fragments 1.0.0-beta.0 → 1.0.0-beta.2

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 (248) hide show
  1. package/README.md +2 -18
  2. package/ng-package.json +25 -0
  3. package/package.json +22 -29
  4. package/src/lib/components/accordion/accordion.component.html +103 -0
  5. package/src/lib/components/accordion/accordion.component.scss +382 -0
  6. package/src/lib/components/accordion/accordion.component.spec.ts +147 -0
  7. package/src/lib/components/accordion/accordion.component.ts +211 -0
  8. package/src/lib/components/accordion/accordion.type.ts +82 -0
  9. package/src/lib/components/breadcrumb/breadcrumb.component.html +43 -0
  10. package/src/lib/components/breadcrumb/breadcrumb.component.scss +112 -0
  11. package/src/lib/components/breadcrumb/breadcrumb.component.spec.ts +33 -0
  12. package/src/lib/components/breadcrumb/breadcrumb.component.ts +103 -0
  13. package/src/lib/components/breadcrumb/breadcrumb.interface.ts +7 -0
  14. package/src/lib/components/button/button.component.html +57 -0
  15. package/src/lib/components/button/button.component.scss +445 -0
  16. package/src/lib/components/button/button.component.spec.ts +99 -0
  17. package/src/lib/components/button/button.component.ts +143 -0
  18. package/src/lib/components/button/button.type.ts +7 -0
  19. package/src/lib/components/card/card.component.html +44 -0
  20. package/src/lib/components/card/card.component.scss +114 -0
  21. package/src/lib/components/card/card.component.spec.ts +65 -0
  22. package/src/lib/components/card/card.component.ts +21 -0
  23. package/src/lib/components/card/card.type.ts +3 -0
  24. package/src/lib/components/code-block/code-block.component.html +55 -0
  25. package/src/lib/components/code-block/code-block.component.scss +122 -0
  26. package/src/lib/components/code-block/code-block.component.spec.ts +81 -0
  27. package/src/lib/components/code-block/code-block.component.ts +302 -0
  28. package/src/lib/components/code-block/code-block.interface.ts +28 -0
  29. package/src/lib/components/code-block/code-block.type.ts +73 -0
  30. package/src/lib/components/decorative/sparkle-field/sparkle-field.component.html +14 -0
  31. package/src/lib/components/decorative/sparkle-field/sparkle-field.component.scss +20 -0
  32. package/src/lib/components/decorative/sparkle-field/sparkle-field.component.spec.ts +38 -0
  33. package/src/lib/components/decorative/sparkle-field/sparkle-field.component.ts +181 -0
  34. package/src/lib/components/input/input-base.ts +187 -0
  35. package/src/lib/components/input/input-calendar/input-calendar.component.html +76 -0
  36. package/src/lib/components/input/input-calendar/input-calendar.component.scss +179 -0
  37. package/src/lib/components/input/input-calendar/input-calendar.component.spec.ts +44 -0
  38. package/src/lib/components/input/input-calendar/input-calendar.component.ts +299 -0
  39. package/src/lib/components/input/input-checkbox/input-checkbox.component.html +37 -0
  40. package/src/lib/components/input/input-checkbox/input-checkbox.component.scss +128 -0
  41. package/src/lib/components/input/input-checkbox/input-checkbox.component.spec.ts +43 -0
  42. package/src/lib/components/input/input-checkbox/input-checkbox.component.ts +112 -0
  43. package/src/lib/components/input/input-checkbox-group/input-checkbox-group.component.html +43 -0
  44. package/src/lib/components/input/input-checkbox-group/input-checkbox-group.component.scss +140 -0
  45. package/src/lib/components/input/input-checkbox-group/input-checkbox-group.component.spec.ts +62 -0
  46. package/src/lib/components/input/input-checkbox-group/input-checkbox-group.component.ts +136 -0
  47. package/src/lib/components/input/input-clock-picker/input-clock-picker.component.html +81 -0
  48. package/src/lib/components/input/input-clock-picker/input-clock-picker.component.scss +228 -0
  49. package/src/lib/components/input/input-clock-picker/input-clock-picker.component.spec.ts +62 -0
  50. package/src/lib/components/input/input-clock-picker/input-clock-picker.component.ts +178 -0
  51. package/src/lib/components/input/input-consts.ts +132 -0
  52. package/src/lib/components/input/input-date/input-date-validators.ts +41 -0
  53. package/src/lib/components/input/input-date/input-date.component.html +41 -0
  54. package/src/lib/components/input/input-date/input-date.component.scss +95 -0
  55. package/src/lib/components/input/input-date/input-date.component.spec.ts +43 -0
  56. package/src/lib/components/input/input-date/input-date.component.ts +359 -0
  57. package/src/lib/components/input/input-date-time/input-date-time.component.html +70 -0
  58. package/src/lib/components/input/input-date-time/input-date-time.component.scss +133 -0
  59. package/src/lib/components/input/input-date-time/input-date-time.component.spec.ts +36 -0
  60. package/src/lib/components/input/input-date-time/input-date-time.component.ts +387 -0
  61. package/src/lib/components/input/input-file-upload/input-file-upload.component.html +89 -0
  62. package/src/lib/components/input/input-file-upload/input-file-upload.component.scss +171 -0
  63. package/src/lib/components/input/input-file-upload/input-file-upload.component.spec.ts +43 -0
  64. package/src/lib/components/input/input-file-upload/input-file-upload.component.ts +351 -0
  65. package/src/lib/components/input/input-interface.ts +8 -0
  66. package/src/lib/components/input/input-number/input-number-validators.ts +0 -0
  67. package/src/lib/components/input/input-number/input-number.component.html +51 -0
  68. package/src/lib/components/input/input-number/input-number.component.scss +140 -0
  69. package/src/lib/components/input/input-number/input-number.component.spec.ts +44 -0
  70. package/src/lib/components/input/input-number/input-number.component.ts +343 -0
  71. package/src/lib/components/input/input-radio-group/input-radio-group.component.html +44 -0
  72. package/src/lib/components/input/input-radio-group/input-radio-group.component.scss +139 -0
  73. package/src/lib/components/input/input-radio-group/input-radio-group.component.spec.ts +58 -0
  74. package/src/lib/components/input/input-radio-group/input-radio-group.component.ts +132 -0
  75. package/src/lib/components/input/input-slider/input-slider.component.html +111 -0
  76. package/src/lib/components/input/input-slider/input-slider.component.scss +203 -0
  77. package/src/lib/components/input/input-slider/input-slider.component.spec.ts +46 -0
  78. package/src/lib/components/input/input-slider/input-slider.component.ts +410 -0
  79. package/src/lib/components/input/input-text/input-text-validators.ts +67 -0
  80. package/src/lib/components/input/input-text/input-text.component.html +71 -0
  81. package/src/lib/components/input/input-text/input-text.component.scss +118 -0
  82. package/src/lib/components/input/input-text/input-text.component.spec.ts +55 -0
  83. package/src/lib/components/input/input-text/input-text.component.ts +215 -0
  84. package/src/lib/components/input/input-time/input-time-validators.ts +42 -0
  85. package/src/lib/components/input/input-time/input-time.component.html +92 -0
  86. package/src/lib/components/input/input-time/input-time.component.scss +191 -0
  87. package/src/lib/components/input/input-time/input-time.component.spec.ts +39 -0
  88. package/src/lib/components/input/input-time/input-time.component.ts +691 -0
  89. package/src/lib/components/input/input-toggle-switch/input-toggle-switch.component.html +36 -0
  90. package/src/lib/components/input/input-toggle-switch/input-toggle-switch.component.scss +121 -0
  91. package/src/lib/components/input/input-toggle-switch/input-toggle-switch.component.spec.ts +54 -0
  92. package/src/lib/components/input/input-toggle-switch/input-toggle-switch.component.ts +117 -0
  93. package/src/lib/components/input/input-type.ts +18 -0
  94. package/src/lib/components/input/input-validation/input-validation.component.html +19 -0
  95. package/src/lib/components/input/input-validation/input-validation.component.scss +39 -0
  96. package/src/lib/components/input/input-validation/input-validation.component.spec.ts +45 -0
  97. package/src/lib/components/input/input-validation/input-validation.component.ts +13 -0
  98. package/src/lib/components/input/input.pipe.ts +14 -0
  99. package/src/lib/components/layout/container/container.component.html +1 -0
  100. package/src/lib/components/layout/container/container.component.scss +33 -0
  101. package/src/lib/components/layout/container/container.component.ts +32 -0
  102. package/src/lib/components/layout/container/container.type.ts +1 -0
  103. package/src/lib/components/layout/divider/divider.component.html +1 -0
  104. package/src/lib/components/layout/divider/divider.component.scss +60 -0
  105. package/src/lib/components/layout/divider/divider.component.ts +38 -0
  106. package/src/lib/components/layout/divider/divider.type.ts +2 -0
  107. package/src/lib/components/layout/section/section.component.html +21 -0
  108. package/src/lib/components/layout/section/section.component.scss +43 -0
  109. package/src/lib/components/layout/section/section.component.ts +33 -0
  110. package/src/lib/components/layout/section/section.type.ts +2 -0
  111. package/src/lib/components/layout/separator/separator.component.html +9 -0
  112. package/src/lib/components/layout/separator/separator.component.scss +52 -0
  113. package/src/lib/components/layout/separator/separator.component.ts +25 -0
  114. package/src/lib/components/layout/separator/separator.type.ts +1 -0
  115. package/src/lib/components/loader/content-blur/content-blur.component.html +13 -0
  116. package/src/lib/components/loader/content-blur/content-blur.component.scss +43 -0
  117. package/src/lib/components/loader/content-blur/content-blur.component.spec.ts +42 -0
  118. package/src/lib/components/loader/content-blur/content-blur.component.ts +34 -0
  119. package/src/lib/components/loader/loader.type.ts +2 -0
  120. package/src/lib/components/loader/progress-bar/progress-bar.component.html +26 -0
  121. package/src/lib/components/loader/progress-bar/progress-bar.component.scss +151 -0
  122. package/src/lib/components/loader/progress-bar/progress-bar.component.spec.ts +47 -0
  123. package/src/lib/components/loader/progress-bar/progress-bar.component.ts +28 -0
  124. package/src/lib/components/loader/progress-bar/progress-bar.type.ts +8 -0
  125. package/src/lib/components/loader/pulse-loader/pulse-loader.component.html +12 -0
  126. package/src/lib/components/loader/pulse-loader/pulse-loader.component.scss +202 -0
  127. package/src/lib/components/loader/pulse-loader/pulse-loader.component.spec.ts +55 -0
  128. package/src/lib/components/loader/pulse-loader/pulse-loader.component.ts +73 -0
  129. package/src/lib/components/loader/pulse-loader/pulse-loader.type.ts +6 -0
  130. package/src/lib/components/loader/skeleton-loader/skeleton-loader.component.html +13 -0
  131. package/src/lib/components/loader/skeleton-loader/skeleton-loader.component.scss +113 -0
  132. package/src/lib/components/loader/skeleton-loader/skeleton-loader.component.spec.ts +37 -0
  133. package/src/lib/components/loader/skeleton-loader/skeleton-loader.component.ts +51 -0
  134. package/src/lib/components/loader/skeleton-loader/skeleton-loader.type.ts +6 -0
  135. package/src/lib/components/loader/spinner/spinner.component.html +20 -0
  136. package/src/lib/components/loader/spinner/spinner.component.scss +137 -0
  137. package/src/lib/components/loader/spinner/spinner.component.spec.ts +43 -0
  138. package/src/lib/components/loader/spinner/spinner.component.ts +32 -0
  139. package/src/lib/components/loader/spinner/spinner.type.ts +6 -0
  140. package/src/lib/components/modal/modal.component.html +47 -0
  141. package/src/lib/components/modal/modal.component.scss +139 -0
  142. package/src/lib/components/modal/modal.component.spec.ts +60 -0
  143. package/src/lib/components/modal/modal.component.ts +83 -0
  144. package/src/lib/components/modal/modal.type.ts +9 -0
  145. package/src/lib/components/morph/blob-moph/blob-moprh.component.spec.ts +79 -0
  146. package/src/lib/components/morph/blob-moph/blob-moprh.component.ts +96 -0
  147. package/src/lib/components/morph/blob-moph/blob-morph.component.html +34 -0
  148. package/src/lib/components/morph/blob-moph/blob-morph.component.scss +7 -0
  149. package/src/lib/components/morph/morph.abstract.ts +13 -0
  150. package/src/lib/components/pagination/pagination.interface.ts +4 -0
  151. package/src/lib/components/pagination/small-pagination/small-pagination.component.html +61 -0
  152. package/src/lib/components/pagination/small-pagination/small-pagination.component.scss +187 -0
  153. package/src/lib/components/pagination/small-pagination/small-pagination.component.spec.ts +88 -0
  154. package/src/lib/components/pagination/small-pagination/small-pagination.component.ts +177 -0
  155. package/src/lib/components/selection-lists/multi-select/multi-select.component.html +170 -0
  156. package/src/lib/components/selection-lists/multi-select/multi-select.component.scss +312 -0
  157. package/src/lib/components/selection-lists/multi-select/multi-select.component.spec.ts +61 -0
  158. package/src/lib/components/selection-lists/multi-select/multi-select.component.ts +372 -0
  159. package/src/lib/components/selection-lists/selection-list/selection-list.component.html +125 -0
  160. package/src/lib/components/selection-lists/selection-list/selection-list.component.scss +267 -0
  161. package/src/lib/components/selection-lists/selection-list/selection-list.component.spec.ts +66 -0
  162. package/src/lib/components/selection-lists/selection-list/selection-list.component.ts +315 -0
  163. package/src/lib/components/selection-lists/selection-lists-base.ts +35 -0
  164. package/src/lib/components/selection-lists/selection-lists-const.ts +17 -0
  165. package/src/lib/components/selection-lists/selection-lists-interface.ts +7 -0
  166. package/src/lib/components/selection-lists/selection-lists.type.ts +1 -0
  167. package/src/lib/components/side-nav/side-nav.component.html +101 -0
  168. package/src/lib/components/side-nav/side-nav.component.scss +295 -0
  169. package/src/lib/components/side-nav/side-nav.component.spec.ts +0 -0
  170. package/src/lib/components/side-nav/side-nav.component.ts +18 -0
  171. package/src/lib/components/side-nav/side-nav.type.ts +28 -0
  172. package/src/lib/components/snackbar/snackbar.component.html +33 -0
  173. package/src/lib/components/snackbar/snackbar.component.scss +195 -0
  174. package/src/lib/components/snackbar/snackbar.component.ts +112 -0
  175. package/src/lib/components/snackbar/snackbar.type.ts +27 -0
  176. package/src/lib/components/status/chip/chip.component.html +51 -0
  177. package/src/lib/components/status/chip/chip.component.scss +149 -0
  178. package/src/lib/components/status/chip/chip.component.spec.ts +62 -0
  179. package/src/lib/components/status/chip/chip.component.ts +83 -0
  180. package/src/lib/components/status/chip/chip.type.ts +42 -0
  181. package/src/lib/components/status/directives/badge/badge.directive.spec.ts +60 -0
  182. package/src/lib/components/status/directives/badge/badge.directive.ts +190 -0
  183. package/src/lib/components/status/directives/badge/badge.interface.ts +19 -0
  184. package/src/lib/components/status/pill/pill.component.html +40 -0
  185. package/src/lib/components/status/pill/pill.component.scss +113 -0
  186. package/src/lib/components/status/pill/pill.component.spec.ts +47 -0
  187. package/src/lib/components/status/pill/pill.component.ts +83 -0
  188. package/src/lib/components/status/pill/pill.type.ts +42 -0
  189. package/src/lib/components/status/status.interface.ts +57 -0
  190. package/src/lib/components/status/status.type.ts +62 -0
  191. package/src/lib/components/status/tag/tag.component.html +39 -0
  192. package/src/lib/components/status/tag/tag.component.scss +140 -0
  193. package/src/lib/components/status/tag/tag.component.spec.ts +47 -0
  194. package/src/lib/components/status/tag/tag.component.ts +83 -0
  195. package/src/lib/components/status/tag/tag.type.ts +42 -0
  196. package/src/lib/components/stepper/stepper.component.html +83 -0
  197. package/src/lib/components/stepper/stepper.component.scss +196 -0
  198. package/src/lib/components/stepper/stepper.component.ts +482 -0
  199. package/src/lib/components/stepper/stepper.type.ts +60 -0
  200. package/src/lib/components/table/table.component.html +438 -0
  201. package/src/lib/components/table/table.component.scss +259 -0
  202. package/src/lib/components/table/table.component.spec.ts +117 -0
  203. package/src/lib/components/table/table.component.ts +215 -0
  204. package/src/lib/components/table/table.enum.ts +4 -0
  205. package/src/lib/components/table/table.function.ts +47 -0
  206. package/src/lib/components/table/table.interface.ts +143 -0
  207. package/src/lib/components/table/table.pipe.ts +62 -0
  208. package/src/lib/components/table/table.type.ts +15 -0
  209. package/src/lib/components/tabs/tabs.component.html +88 -0
  210. package/src/lib/components/tabs/tabs.component.scss +305 -0
  211. package/src/lib/components/tabs/tabs.component.spec.ts +94 -0
  212. package/src/lib/components/tabs/tabs.component.ts +282 -0
  213. package/src/lib/components/tabs/tabs.type.ts +81 -0
  214. package/src/lib/components/title-bar/title-bar.component.html +21 -0
  215. package/src/lib/components/title-bar/title-bar.component.scss +139 -0
  216. package/src/lib/components/title-bar/title-bar.component.spec.ts +44 -0
  217. package/src/lib/components/title-bar/title-bar.component.ts +13 -0
  218. package/src/lib/components/toast/toast.component.html +36 -0
  219. package/src/lib/components/toast/toast.component.scss +241 -0
  220. package/src/lib/components/toast/toast.component.ts +165 -0
  221. package/src/lib/components/toast/toast.type.ts +37 -0
  222. package/src/lib/components/toast-stack/toast-stack.component.html +30 -0
  223. package/src/lib/components/toast-stack/toast-stack.component.scss +35 -0
  224. package/src/lib/components/toast-stack/toast-stack.component.ts +51 -0
  225. package/src/lib/consts/country-prefix.ts +244 -0
  226. package/src/lib/directives/tooltip/popover.directive.ts +274 -0
  227. package/src/lib/directives/tooltip/tooltip.directive.spec.ts +86 -0
  228. package/src/lib/directives/tooltip/tooltip.directive.ts +234 -0
  229. package/src/lib/directives/tooltip/tooltip.interface.ts +29 -0
  230. package/src/lib/directives/tooltip/tooltip.type.ts +9 -0
  231. package/src/lib/interfaces/common.interfaces.ts +4 -0
  232. package/src/lib/pipes/chunk.pipe.ts +16 -0
  233. package/src/lib/pipes/safe-html.pipe.ts +14 -0
  234. package/src/lib/pipes/sanitize-html.pipe.ts +23 -0
  235. package/src/lib/types/base.types.ts +23 -0
  236. package/src/lib/types/common.types.ts +98 -0
  237. package/src/lib/types/form.types.ts +5 -0
  238. package/src/lib/utils/common.utils.ts +53 -0
  239. package/src/lib/utils/date.utils.ts +474 -0
  240. package/src/lib/utils/number.utils.ts +16 -0
  241. package/src/lib/utils/uuid.utils.ts +39 -0
  242. package/src/public-api.ts +114 -0
  243. package/tsconfig.lib.json +17 -0
  244. package/tsconfig.lib.prod.json +10 -0
  245. package/tsconfig.spec.json +9 -0
  246. package/fesm2022/fragments.mjs +0 -8928
  247. package/fesm2022/fragments.mjs.map +0 -1
  248. package/index.d.ts +0 -3929
@@ -0,0 +1,274 @@
1
+ import {
2
+ Directive,
3
+ ElementRef,
4
+ HostListener,
5
+ inject,
6
+ Input,
7
+ OnDestroy,
8
+ Renderer2,
9
+ } from '@angular/core';
10
+ import { IConfigTooltipPopover, ITooltipEventAction, ITooltipLinkAction } from './tooltip.interface';
11
+
12
+ @Directive({
13
+ selector: '[frgPopover]',
14
+ standalone: true,
15
+ })
16
+ export class FragmentsPopoverDirective implements OnDestroy {
17
+ private readonly elemRef: ElementRef = inject(ElementRef);
18
+ private readonly renderer: Renderer2 = inject(Renderer2);
19
+ @Input() popoverConfig: IConfigTooltipPopover = {} as IConfigTooltipPopover;
20
+ popover: HTMLElement | undefined;
21
+
22
+ @HostListener('click', ['$event']) onClick(event: MouseEvent) {
23
+ event.stopPropagation();
24
+ if (!this.popover) {
25
+ this.show();
26
+ return;
27
+ }
28
+ this.hide();
29
+ }
30
+
31
+ @HostListener('document:click') onDocumentClick() {
32
+ this.close();
33
+ }
34
+
35
+ @HostListener('keydown.escape') onEscape() {
36
+ this.close();
37
+ }
38
+
39
+ ngOnDestroy(): void {
40
+ this.close();
41
+ }
42
+
43
+ private close() {
44
+ if (this.popover) {
45
+ this.hide();
46
+ }
47
+ }
48
+
49
+ private show() {
50
+ if (!this.popoverConfig.apply) {
51
+ return;
52
+ }
53
+
54
+ this.initPopover();
55
+
56
+ if (this.popoverConfig?.text) {
57
+ this.popover = this.renderer.createElement('div');
58
+ this.renderer.addClass(this.popover, 'frg-popover');
59
+ this.renderer.setStyle(this.popover, 'text-align', 'left');
60
+ this.renderer.setAttribute(this.popover, 'role', 'dialog');
61
+ this.renderer.setAttribute(this.popover, 'tabindex', '-1');
62
+ this.renderer.setStyle(this.popover, 'position', 'absolute');
63
+ this.renderer.setStyle(this.popover, 'will-change', 'transform');
64
+ this.renderer.setStyle(this.popover, 'left', '0');
65
+ this.renderer.setStyle(this.popover, 'top', '0');
66
+ this.renderer.setStyle(this.popover, 'z-index', '9999');
67
+
68
+ const innerPopover: HTMLElement = this.renderer.createElement('div');
69
+ this.renderer.addClass(innerPopover, 'frg-popover-inner');
70
+
71
+ const textParagraph: HTMLElement = this.renderer.createElement('p');
72
+ const strongTextParagraph: HTMLElement = this.renderer.createElement('strong');
73
+
74
+ this.renderer.appendChild(
75
+ strongTextParagraph,
76
+ this.renderer.createText(this.popoverConfig.text)
77
+ );
78
+
79
+ this.renderer.appendChild(textParagraph, strongTextParagraph);
80
+ this.renderer.appendChild(innerPopover, textParagraph);
81
+
82
+ if (this.popoverConfig.textContent) {
83
+ const contentParagraph: HTMLElement = this.renderer.createElement('p');
84
+ this.renderer.appendChild(
85
+ contentParagraph,
86
+ this.renderer.createText(this.popoverConfig.textContent)
87
+ );
88
+ this.renderer.appendChild(innerPopover, contentParagraph);
89
+ }
90
+
91
+ if (this.popoverConfig.buttonConfig?.text) {
92
+ const actions: HTMLElement = this.renderer.createElement('div');
93
+ this.renderer.addClass(actions, 'frg-popover-actions');
94
+
95
+ const actionButton: HTMLElement = this.renderer.createElement('button');
96
+ this.renderer.addClass(actionButton, 'frg-popover-button');
97
+ this.renderer.setAttribute(actionButton, 'type', 'button');
98
+ this.renderer.appendChild(
99
+ actionButton,
100
+ this.renderer.createText(this.popoverConfig.buttonConfig.text)
101
+ );
102
+
103
+ this.renderer.listen(actionButton, 'click', (event: Event) => {
104
+ event.stopPropagation();
105
+ this.handleAction();
106
+ });
107
+
108
+ this.renderer.appendChild(actions, actionButton);
109
+ this.renderer.appendChild(innerPopover, actions);
110
+ }
111
+
112
+ this.renderer.appendChild(this.popover, innerPopover);
113
+
114
+ if (!this.popoverConfig.hideIndicator) {
115
+ const indicator: HTMLElement = this.renderer.createElement('span');
116
+ this.renderer.addClass(indicator, 'frg-popover-indicator');
117
+ this.renderer.appendChild(this.popover, indicator);
118
+ }
119
+
120
+ this.renderer.appendChild(document.body, this.popover);
121
+ this.setPosition();
122
+ }
123
+ }
124
+
125
+ private hide() {
126
+ this.renderer.removeChild(document.body, this.popover);
127
+ this.popover = undefined;
128
+ }
129
+
130
+ private initPopover() {
131
+ this.popoverConfig.text = this.popoverConfig.text
132
+ ? this.popoverConfig.text
133
+ : 'No popover text';
134
+ this.popoverConfig.position = this.popoverConfig.position
135
+ ? this.popoverConfig.position
136
+ : 'top';
137
+ this.popoverConfig.gap = this.popoverConfig.gap ?? 0;
138
+ this.popoverConfig.hideIndicator = this.popoverConfig.hideIndicator ?? false;
139
+ this.popoverConfig.apply = this.popoverConfig.apply ?? true;
140
+ }
141
+
142
+ private handleAction(): void {
143
+ const action = this.popoverConfig.buttonConfig?.action;
144
+ if (!action) {
145
+ return;
146
+ }
147
+
148
+ if ((action as ITooltipLinkAction).link) {
149
+ const linkAction = action as ITooltipLinkAction;
150
+ window.open(linkAction.link, linkAction.target || '_self');
151
+ return;
152
+ }
153
+
154
+ if ((action as ITooltipEventAction).clickEvent) {
155
+ const eventAction = action as ITooltipEventAction;
156
+ eventAction.clickEvent(undefined);
157
+ }
158
+ }
159
+
160
+ private setPosition() {
161
+ const element = this.elemRef.nativeElement;
162
+ const rect = element.getBoundingClientRect();
163
+ const bodyRect = document.body.getBoundingClientRect();
164
+ const viewportWidth =
165
+ document.documentElement?.clientWidth || window.innerWidth;
166
+ const viewportPadding = 8;
167
+
168
+ const xOffset = rect.left - bodyRect.left;
169
+ const yOffset = rect.top - bodyRect.top;
170
+ const widthOffset = rect.width;
171
+ const heightOffset = rect.height;
172
+
173
+ const p_widthOffset: number = this.popover?.offsetWidth ?? 0;
174
+ const p_heightOffset: number = this.popover?.offsetHeight ?? 0;
175
+
176
+ const clampX = (x: number) =>
177
+ Math.min(
178
+ Math.max(x, viewportPadding),
179
+ viewportWidth - p_widthOffset - viewportPadding
180
+ );
181
+
182
+ const position = (this.popoverConfig.position || 'top')
183
+ .toLowerCase()
184
+ .trim()
185
+ .replaceAll(' ', '-');
186
+
187
+ this.renderer.setAttribute(this.popover, 'data-position', position);
188
+
189
+ switch (position) {
190
+ case 'top':
191
+ this.renderer.setStyle(
192
+ this.popover,
193
+ 'transform',
194
+ `translate(${clampX(xOffset + widthOffset / 2 - p_widthOffset / 2)}px, ${
195
+ yOffset - p_heightOffset - (this.popoverConfig?.gap ?? 0)
196
+ }px)`
197
+ );
198
+ break;
199
+
200
+ case 'top-left':
201
+ this.renderer.setStyle(
202
+ this.popover,
203
+ 'transform',
204
+ `translate(${clampX(xOffset + (widthOffset - p_widthOffset))}px, ${
205
+ yOffset - p_heightOffset - (this.popoverConfig?.gap ?? 0)
206
+ }px)`
207
+ );
208
+ break;
209
+
210
+ case 'top-right':
211
+ this.renderer.setStyle(
212
+ this.popover,
213
+ 'transform',
214
+ `translate(${clampX(xOffset)}px, ${
215
+ yOffset - p_heightOffset - (this.popoverConfig?.gap ?? 0)
216
+ }px)`
217
+ );
218
+ break;
219
+
220
+ case 'left':
221
+ this.renderer.setStyle(
222
+ this.popover,
223
+ 'transform',
224
+ `translate(${clampX(
225
+ xOffset - p_widthOffset - (this.popoverConfig?.gap ?? 0)
226
+ )}px, ${yOffset + heightOffset / 2 - p_heightOffset / 2}px)`
227
+ );
228
+ break;
229
+
230
+ case 'right':
231
+ this.renderer.setStyle(
232
+ this.popover,
233
+ 'transform',
234
+ `translate(${clampX(
235
+ xOffset + widthOffset + (this.popoverConfig?.gap ?? 0)
236
+ )}px, ${yOffset + heightOffset / 2 - p_heightOffset / 2}px)`
237
+ );
238
+ break;
239
+
240
+ case 'bottom':
241
+ this.renderer.setStyle(
242
+ this.popover,
243
+ 'transform',
244
+ `translate(${clampX(xOffset + widthOffset / 2 - p_widthOffset / 2)}px, ${
245
+ yOffset + heightOffset + (this.popoverConfig?.gap ?? 0)
246
+ }px)`
247
+ );
248
+ break;
249
+
250
+ case 'bottom-left':
251
+ this.renderer.setStyle(
252
+ this.popover,
253
+ 'transform',
254
+ `translate(${clampX(xOffset + (widthOffset - p_widthOffset))}px, ${
255
+ yOffset + heightOffset + (this.popoverConfig?.gap ?? 0)
256
+ }px)`
257
+ );
258
+ break;
259
+
260
+ case 'bottom-right':
261
+ this.renderer.setStyle(
262
+ this.popover,
263
+ 'transform',
264
+ `translate(${clampX(xOffset)}px, ${
265
+ yOffset + heightOffset + (this.popoverConfig?.gap ?? 0)
266
+ }px)`
267
+ );
268
+ break;
269
+
270
+ default:
271
+ break;
272
+ }
273
+ }
274
+ }
@@ -0,0 +1,86 @@
1
+ import { Component } from '@angular/core';
2
+ import { ComponentFixture, TestBed } from '@angular/core/testing';
3
+ import { By } from '@angular/platform-browser';
4
+ import { FragmentsTooltipDirective } from './tooltip.directive';
5
+ import { IConfigTooltip } from './tooltip.interface';
6
+
7
+ @Component({
8
+ standalone: true,
9
+ imports: [FragmentsTooltipDirective],
10
+ template: `
11
+ <span frgTooltip [tooltipConfig]="tooltipConfig">Hover me</span>
12
+ `
13
+ })
14
+ class TooltipHostComponent {
15
+ tooltipConfig: IConfigTooltip = {
16
+ apply: true,
17
+ text: 'Tooltip text',
18
+ position: 'top',
19
+ gap: 8
20
+ };
21
+ }
22
+
23
+ describe('FragmentsTooltipDirective', () => {
24
+ let fixture: ComponentFixture<TooltipHostComponent>;
25
+
26
+ beforeEach(async () => {
27
+ await TestBed.configureTestingModule({
28
+ imports: [TooltipHostComponent]
29
+ }).compileComponents();
30
+
31
+ fixture = TestBed.createComponent(TooltipHostComponent);
32
+ fixture.detectChanges();
33
+ });
34
+
35
+ afterEach(() => {
36
+ const tooltip = document.body.querySelector('.frg-tooltip');
37
+ if (tooltip?.parentElement) {
38
+ tooltip.parentElement.removeChild(tooltip);
39
+ }
40
+ fixture.destroy();
41
+ });
42
+
43
+ it('renders tooltip on mouseenter with indicator by default', () => {
44
+ const target = fixture.debugElement.query(By.css('span')).nativeElement as HTMLElement;
45
+ target.dispatchEvent(new Event('mouseenter'));
46
+ fixture.detectChanges();
47
+
48
+ const tooltip = document.body.querySelector('.frg-tooltip') as HTMLElement;
49
+ expect(tooltip).toBeTruthy();
50
+ expect(tooltip.querySelector('.frg-tooltip-indicator')).toBeTruthy();
51
+ });
52
+
53
+ it('hides indicator when hideIndicator is true', () => {
54
+ fixture.componentInstance.tooltipConfig = {
55
+ apply: true,
56
+ text: 'Tooltip text',
57
+ position: 'bottom-right',
58
+ hideIndicator: true
59
+ };
60
+ fixture.detectChanges();
61
+
62
+ const target = fixture.debugElement.query(By.css('span')).nativeElement as HTMLElement;
63
+ target.dispatchEvent(new Event('mouseenter'));
64
+ fixture.detectChanges();
65
+
66
+ const tooltip = document.body.querySelector('.frg-tooltip') as HTMLElement;
67
+ expect(tooltip).toBeTruthy();
68
+ expect(tooltip.querySelector('.frg-tooltip-indicator')).toBeFalsy();
69
+ });
70
+
71
+ it('sets data-position based on config', () => {
72
+ fixture.componentInstance.tooltipConfig = {
73
+ apply: true,
74
+ text: 'Tooltip text',
75
+ position: 'bottom-right'
76
+ };
77
+ fixture.detectChanges();
78
+
79
+ const target = fixture.debugElement.query(By.css('span')).nativeElement as HTMLElement;
80
+ target.dispatchEvent(new Event('mouseenter'));
81
+ fixture.detectChanges();
82
+
83
+ const tooltip = document.body.querySelector('.frg-tooltip') as HTMLElement;
84
+ expect(tooltip.getAttribute('data-position')).toBe('bottom-right');
85
+ });
86
+ });
@@ -0,0 +1,234 @@
1
+ import {
2
+ Directive,
3
+ ElementRef,
4
+ HostListener,
5
+ inject,
6
+ Input,
7
+ OnDestroy,
8
+ Renderer2,
9
+ } from '@angular/core';
10
+ import { IConfigTooltip } from './tooltip.interface';
11
+
12
+ @Directive({
13
+ selector: '[frgTooltip]',
14
+ standalone: true,
15
+ })
16
+ export class FragmentsTooltipDirective implements OnDestroy {
17
+ private readonly elemRef: ElementRef = inject(ElementRef);
18
+ private readonly renderer: Renderer2 = inject(Renderer2);
19
+ @Input() tooltipConfig: IConfigTooltip = {} as IConfigTooltip;
20
+ tooltip: HTMLElement | undefined;
21
+
22
+ @HostListener('mouseenter') onMouseEnter() {
23
+ if (!this.tooltip && this.tooltipConfig.text) this.show();
24
+ }
25
+
26
+ @HostListener('mouseleave') onMouseLeave() {
27
+ this.ngOnDestroy();
28
+ }
29
+
30
+ ngOnDestroy(): void {
31
+ if (this.tooltip) {
32
+ this.hide();
33
+ }
34
+ }
35
+
36
+ private show() {
37
+ if (this.tooltipConfig.apply === false)
38
+ return;
39
+
40
+ this.initTooltip();
41
+
42
+ if (this.tooltipConfig?.text) {
43
+ this.tooltip = this.renderer.createElement('div');
44
+ this.renderer.addClass(this.tooltip, 'frg-tooltip');
45
+ this.renderer.setStyle(this.tooltip, 'text-align', 'left');
46
+ this.renderer.setAttribute(this.tooltip, 'role', 'tooltip');
47
+ this.renderer.setAttribute(this.tooltip, 'tabindex', '-1');
48
+ this.renderer.setStyle(this.tooltip, 'position', 'absolute');
49
+ this.renderer.setStyle(this.tooltip, 'will-change', 'transform');
50
+ this.renderer.setStyle(this.tooltip, 'left', '0');
51
+ this.renderer.setStyle(this.tooltip, 'top', '0');
52
+ this.renderer.setStyle(this.tooltip, 'z-index', '9999');
53
+
54
+ if (this.tooltipConfig.textContent) {
55
+ const innerTooltip: HTMLElement = this.renderer.createElement('div');
56
+ this.renderer.addClass(innerTooltip, 'frg-tooltip-inner');
57
+
58
+ const textParagraph: HTMLElement = this.renderer.createElement('p');
59
+ const contentParagraph: HTMLElement = this.renderer.createElement('p');
60
+ const strongTextParagraph: HTMLElement =
61
+ this.renderer.createElement('strong');
62
+
63
+ this.renderer.appendChild(
64
+ strongTextParagraph,
65
+ this.renderer.createText(this.tooltipConfig.text)
66
+ );
67
+
68
+ this.renderer.appendChild(textParagraph, strongTextParagraph);
69
+
70
+ this.renderer.appendChild(
71
+ contentParagraph,
72
+ this.renderer.createText(this.tooltipConfig.textContent)
73
+ );
74
+
75
+ this.renderer.appendChild(innerTooltip, textParagraph);
76
+
77
+ this.renderer.appendChild(innerTooltip, contentParagraph);
78
+
79
+ this.renderer.appendChild(this.tooltip, innerTooltip);
80
+ } else {
81
+ const innerTooltip: HTMLElement = this.renderer.createElement('strong');
82
+
83
+ this.renderer.appendChild(
84
+ innerTooltip,
85
+ this.renderer.createText(this.tooltipConfig.text)
86
+ );
87
+
88
+ this.renderer.appendChild(this.tooltip, innerTooltip);
89
+ }
90
+
91
+ if (!this.tooltipConfig.hideIndicator) {
92
+ const indicator: HTMLElement = this.renderer.createElement('span');
93
+ this.renderer.addClass(indicator, 'frg-tooltip-indicator');
94
+ this.renderer.appendChild(this.tooltip, indicator);
95
+ }
96
+
97
+ this.renderer.appendChild(document.body, this.tooltip);
98
+ this.setPosition();
99
+ }
100
+ }
101
+
102
+ private hide() {
103
+ this.renderer.removeChild(this.elemRef, this.tooltip);
104
+ this.tooltip = undefined;
105
+ }
106
+
107
+ private initTooltip() {
108
+ this.tooltipConfig.text = this.tooltipConfig.text
109
+ ? this.tooltipConfig.text
110
+ : 'No tooltip text';
111
+ this.tooltipConfig.position = this.tooltipConfig.position
112
+ ? this.tooltipConfig.position
113
+ : 'top';
114
+ this.tooltipConfig.gap = this.tooltipConfig.gap ?? 0;
115
+ this.tooltipConfig.hideIndicator = this.tooltipConfig.hideIndicator ?? false;
116
+ this.tooltipConfig.apply = this.tooltipConfig.apply ?? true;
117
+ }
118
+
119
+ private setPosition() {
120
+ const element = this.elemRef.nativeElement;
121
+ const rect = element.getBoundingClientRect();
122
+ const bodyRect = document.body.getBoundingClientRect();
123
+ const viewportWidth =
124
+ document.documentElement?.clientWidth || window.innerWidth;
125
+ const viewportPadding = 8;
126
+
127
+ const xOffset = rect.left - bodyRect.left;
128
+ const yOffset = rect.top - bodyRect.top;
129
+ const widthOffset = rect.width;
130
+ const heightOffset = rect.height;
131
+
132
+ const t_widthOffset: number = this.tooltip?.offsetWidth ?? 0;
133
+ const t_heightOffset: number = this.tooltip?.offsetHeight ?? 0;
134
+
135
+ const clampX = (x: number) =>
136
+ Math.min(
137
+ Math.max(x, viewportPadding),
138
+ viewportWidth - t_widthOffset - viewportPadding
139
+ );
140
+
141
+ const position = (this.tooltipConfig.position || 'top')
142
+ .toLowerCase()
143
+ .trim()
144
+ .replaceAll(' ', '-');
145
+
146
+ this.renderer.setAttribute(this.tooltip, 'data-position', position);
147
+
148
+ switch (position) {
149
+ case 'top':
150
+ this.renderer.setStyle(
151
+ this.tooltip,
152
+ 'transform',
153
+ `translate(${clampX(xOffset + widthOffset / 2 - t_widthOffset / 2)}px, ${
154
+ yOffset - t_heightOffset - (this.tooltipConfig?.gap ?? 0)
155
+ }px)`
156
+ );
157
+
158
+ break;
159
+
160
+ case 'top-left':
161
+ this.renderer.setStyle(
162
+ this.tooltip,
163
+ 'transform',
164
+ `translate(${clampX(xOffset + (widthOffset - t_widthOffset))}px, ${
165
+ yOffset - t_heightOffset - (this.tooltipConfig?.gap ?? 0)
166
+ }px)`
167
+ );
168
+ break;
169
+
170
+ case 'top-right':
171
+ this.renderer.setStyle(
172
+ this.tooltip,
173
+ 'transform',
174
+ `translate(${clampX(xOffset)}px, ${
175
+ yOffset - t_heightOffset - (this.tooltipConfig?.gap ?? 0)
176
+ }px)`
177
+ );
178
+ break;
179
+
180
+ case 'left':
181
+ this.renderer.setStyle(
182
+ this.tooltip,
183
+ 'transform',
184
+ `translate(${clampX(
185
+ xOffset - t_widthOffset - (this.tooltipConfig?.gap ?? 0)
186
+ )}px, ${yOffset + heightOffset / 2 - t_heightOffset / 2}px)`
187
+ );
188
+ break;
189
+
190
+ case 'right':
191
+ this.renderer.setStyle(
192
+ this.tooltip,
193
+ 'transform',
194
+ `translate(${clampX(
195
+ xOffset + widthOffset + (this.tooltipConfig?.gap ?? 0)
196
+ )}px, ${yOffset + heightOffset / 2 - t_heightOffset / 2}px)`
197
+ );
198
+ break;
199
+
200
+ case 'bottom':
201
+ this.renderer.setStyle(
202
+ this.tooltip,
203
+ 'transform',
204
+ `translate(${clampX(xOffset + widthOffset / 2 - t_widthOffset / 2)}px, ${
205
+ yOffset + heightOffset + (this.tooltipConfig?.gap ?? 0)
206
+ }px)`
207
+ );
208
+ break;
209
+
210
+ case 'bottom-left':
211
+ this.renderer.setStyle(
212
+ this.tooltip,
213
+ 'transform',
214
+ `translate(${clampX(xOffset + (widthOffset - t_widthOffset))}px, ${
215
+ yOffset + heightOffset + (this.tooltipConfig?.gap ?? 0)
216
+ }px)`
217
+ );
218
+ break;
219
+
220
+ case 'bottom-right':
221
+ this.renderer.setStyle(
222
+ this.tooltip,
223
+ 'transform',
224
+ `translate(${clampX(xOffset)}px, ${
225
+ yOffset + heightOffset + (this.tooltipConfig?.gap ?? 0)
226
+ }px)`
227
+ );
228
+ break;
229
+
230
+ default:
231
+ break;
232
+ }
233
+ }
234
+ }
@@ -0,0 +1,29 @@
1
+ import { GenericFunction, TBehaviourAction } from '../../types/common.types';
2
+ import { TooltipPosition } from './tooltip.type';
3
+
4
+ export interface ITooltipLinkAction {
5
+ link: string;
6
+ target: '_self' | '_blank' | '_parent' | '_top';
7
+ }
8
+
9
+ export interface ITooltipEventAction<T = any> {
10
+ clickEvent: GenericFunction<T>;
11
+ }
12
+
13
+ export interface IConfigTooltip {
14
+ text: string;
15
+ textContent?: string;
16
+ apply?: boolean;
17
+ hideIndicator?: boolean;
18
+ position?: TooltipPosition;
19
+ gap?: number;
20
+ }
21
+
22
+ export interface IConfigTooltipPopover extends IConfigTooltip {
23
+ buttonConfig: IConfigTooltipButtton;
24
+ }
25
+
26
+ export interface IConfigTooltipButtton {
27
+ text: string;
28
+ action: TBehaviourAction<ITooltipLinkAction, ITooltipEventAction>;
29
+ }
@@ -0,0 +1,9 @@
1
+ export type TooltipPosition =
2
+ | 'top'
3
+ | 'top-left'
4
+ | 'top-right'
5
+ | 'left'
6
+ | 'right'
7
+ | 'bottom'
8
+ | 'bottom-left'
9
+ | 'bottom-right';
@@ -0,0 +1,4 @@
1
+ export interface FormatRegex {
2
+ regex: RegExp;
3
+ tokens: string[];
4
+ }
@@ -0,0 +1,16 @@
1
+ import { Pipe, PipeTransform } from '@angular/core';
2
+
3
+ @Pipe({
4
+ name: 'chunk',
5
+ pure: true
6
+ })
7
+ export class ChunkPipe implements PipeTransform {
8
+ transform<T>(array: T[], size: number): T[][] {
9
+ if (!array || size <= 0) return [];
10
+ const chunks: T[][] = [];
11
+ for (let i = 0; i < array.length; i += size) {
12
+ chunks.push(array.slice(i, i + size));
13
+ }
14
+ return chunks;
15
+ }
16
+ }
@@ -0,0 +1,14 @@
1
+ import { Pipe, inject, type PipeTransform } from '@angular/core';
2
+ import { DomSanitizer, SafeHtml } from '@angular/platform-browser';
3
+
4
+ @Pipe({
5
+ name: 'safeHtml',
6
+ standalone: true,
7
+ })
8
+ export class SafeHtmlPipe implements PipeTransform {
9
+ private readonly sanitizer = inject(DomSanitizer);
10
+
11
+ transform(value: string): SafeHtml {
12
+ return this.sanitizer.bypassSecurityTrustHtml(value);
13
+ }
14
+ }
@@ -0,0 +1,23 @@
1
+ import { Pipe, PipeTransform, inject } from '@angular/core';
2
+ import { DomSanitizer, SafeHtml } from '@angular/platform-browser';
3
+
4
+ @Pipe({
5
+ name: 'sanitizeHtml',
6
+ standalone: true,
7
+ })
8
+ export class SanitizeHtmlPipe implements PipeTransform {
9
+ _sanitizer = inject(DomSanitizer);
10
+
11
+ transform(v: string | number | null | undefined): SafeHtml {
12
+ let text: string = '';
13
+ if (typeof v === 'number') {
14
+ text = String(v);
15
+ } else if (v === null || v === undefined) {
16
+ text = '';
17
+ } else {
18
+ text = v.replace(/(?:\r\n|\r|\\n)/g, '<br>');
19
+ }
20
+
21
+ return this._sanitizer.bypassSecurityTrustHtml(text);
22
+ }
23
+ }