@ethlete/cdk 4.2.1 → 4.3.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 (220) hide show
  1. package/CHANGELOG.md +20 -0
  2. package/esm2022/lib/components/accordion/components/accordion/accordion.component.mjs +3 -3
  3. package/esm2022/lib/components/accordion/components/accordion-group/accordion-group.component.mjs +3 -3
  4. package/esm2022/lib/components/accordion/partials/accordion-hint/accordion-hint.directive.mjs +3 -3
  5. package/esm2022/lib/components/accordion/partials/accordion-hint-wrapper/accordion-hint-wrapper.directive.mjs +3 -3
  6. package/esm2022/lib/components/accordion/partials/accordion-label/accordion-label.directive.mjs +3 -3
  7. package/esm2022/lib/components/accordion/partials/accordion-label-wrapper/accordion-label-wrapper.directive.mjs +3 -3
  8. package/esm2022/lib/components/bracket/components/bracket/bracket.component.mjs +3 -3
  9. package/esm2022/lib/components/bracket/directives/bracket-match/bracket-match.directive.mjs +3 -3
  10. package/esm2022/lib/components/bracket/directives/bracket-round/bracket-round.directive.mjs +3 -3
  11. package/esm2022/lib/components/bracket/partials/bracket-match/bracket-match.component.mjs +3 -3
  12. package/esm2022/lib/components/bracket/partials/bracket-round-header/bracket-round-header.component.mjs +3 -3
  13. package/esm2022/lib/components/button/components/button/button.component.mjs +3 -3
  14. package/esm2022/lib/components/button/components/query-button/query-button.component.mjs +3 -3
  15. package/esm2022/lib/components/button/directives/button/button.directive.mjs +3 -3
  16. package/esm2022/lib/components/button/directives/query-button/query-button.directive.mjs +3 -3
  17. package/esm2022/lib/components/filter/filter-overlay/components/filter-overlay-page-outlet/filter-overlay-page-outlet.component.mjs +3 -3
  18. package/esm2022/lib/components/filter/filter-overlay/directives/filter-overlay-back-or-close/filter-overlay-back-or-close.directive.mjs +3 -3
  19. package/esm2022/lib/components/filter/filter-overlay/directives/filter-overlay-link/filter-overlay-link.directive.mjs +3 -3
  20. package/esm2022/lib/components/filter/filter-overlay/directives/filter-overlay-reset/filter-overlay-reset.directive.mjs +3 -3
  21. package/esm2022/lib/components/filter/filter-overlay/directives/filter-overlay-submit/filter-overlay-submit.directive.mjs +3 -3
  22. package/esm2022/lib/components/filter/filter-overlay/services/filter-overlay.service.mjs +3 -3
  23. package/esm2022/lib/components/filter/rich-filter/components/rich-filter-host/rich-filter-host.component.mjs +3 -3
  24. package/esm2022/lib/components/filter/rich-filter/directives/rich-filter-button/rich-filter-button.directive.mjs +3 -3
  25. package/esm2022/lib/components/filter/rich-filter/directives/rich-filter-button-slot/rich-filter-button-slot.directive.mjs +3 -3
  26. package/esm2022/lib/components/filter/rich-filter/directives/rich-filter-content/rich-filter-content.directive.mjs +3 -3
  27. package/esm2022/lib/components/filter/rich-filter/directives/rich-filter-top/rich-filter-top.directive.mjs +3 -3
  28. package/esm2022/lib/components/forms/components/checkbox/components/checkbox/checkbox.component.mjs +3 -3
  29. package/esm2022/lib/components/forms/components/checkbox/components/checkbox-field/checkbox-field.component.mjs +3 -3
  30. package/esm2022/lib/components/forms/components/checkbox/components/checkbox-group/checkbox-group.component.mjs +3 -3
  31. package/esm2022/lib/components/forms/components/checkbox/directives/checkbox/checkbox.directive.mjs +3 -3
  32. package/esm2022/lib/components/forms/components/checkbox/directives/checkbox-field/checkbox-field.directive.mjs +3 -3
  33. package/esm2022/lib/components/forms/components/checkbox/directives/checkbox-group/checkbox-group.directive.mjs +3 -3
  34. package/esm2022/lib/components/forms/components/checkbox/directives/checkbox-group-control/checkbox-group-control.directive.mjs +3 -3
  35. package/esm2022/lib/components/forms/components/error/components/error/error.component.mjs +3 -3
  36. package/esm2022/lib/components/forms/components/input/components/date-input/date-input.component.mjs +3 -3
  37. package/esm2022/lib/components/forms/components/input/components/email-input/email-input.component.mjs +3 -3
  38. package/esm2022/lib/components/forms/components/input/components/input-field/input-field.component.mjs +3 -3
  39. package/esm2022/lib/components/forms/components/input/components/number-input/number-input.component.mjs +3 -3
  40. package/esm2022/lib/components/forms/components/input/components/password-input/password-input.component.mjs +3 -3
  41. package/esm2022/lib/components/forms/components/input/components/search-input/search-input.component.mjs +3 -3
  42. package/esm2022/lib/components/forms/components/input/components/text-input/text-input.component.mjs +3 -3
  43. package/esm2022/lib/components/forms/components/input/components/textarea-input/textarea-input.component.mjs +3 -3
  44. package/esm2022/lib/components/forms/components/input/directives/autosize-textarea/autosize-textarea.directive.mjs +3 -3
  45. package/esm2022/lib/components/forms/components/input/directives/date-input/date-input.directive.mjs +3 -3
  46. package/esm2022/lib/components/forms/components/input/directives/email-input/email-input.directive.mjs +3 -3
  47. package/esm2022/lib/components/forms/components/input/directives/number-input/number-input.directive.mjs +3 -3
  48. package/esm2022/lib/components/forms/components/input/directives/password-input/password-input.directive.mjs +3 -3
  49. package/esm2022/lib/components/forms/components/input/directives/search-input/search-input.directive.mjs +3 -3
  50. package/esm2022/lib/components/forms/components/input/directives/text-input/text-input.directive.mjs +3 -3
  51. package/esm2022/lib/components/forms/components/input/directives/textarea-input/textarea-input.directive.mjs +3 -3
  52. package/esm2022/lib/components/forms/components/input/partials/password-input-toggle/password-input-toggle.component.mjs +3 -3
  53. package/esm2022/lib/components/forms/components/input/partials/search-input-clear/search-input-clear.component.mjs +3 -3
  54. package/esm2022/lib/components/forms/components/label/components/label/label.component.mjs +3 -3
  55. package/esm2022/lib/components/forms/components/label/directives/label-suffix/label-suffix.directive.mjs +3 -3
  56. package/esm2022/lib/components/forms/components/radio/components/radio/radio.component.mjs +3 -3
  57. package/esm2022/lib/components/forms/components/radio/components/radio-field/radio-field.component.mjs +3 -3
  58. package/esm2022/lib/components/forms/components/radio/components/radio-group/radio-group.component.mjs +3 -3
  59. package/esm2022/lib/components/forms/components/radio/directives/radio/radio.directive.mjs +3 -3
  60. package/esm2022/lib/components/forms/components/radio/directives/radio-field/radio-field.directive.mjs +3 -3
  61. package/esm2022/lib/components/forms/components/radio/directives/radio-group/radio-group.directive.mjs +3 -3
  62. package/esm2022/lib/components/forms/components/segmented-button/components/segmented-button/segmented-button.component.mjs +3 -3
  63. package/esm2022/lib/components/forms/components/segmented-button/components/segmented-button-field/segmented-button-field.component.mjs +3 -3
  64. package/esm2022/lib/components/forms/components/segmented-button/components/segmented-button-group/segmented-button-group.component.mjs +3 -3
  65. package/esm2022/lib/components/forms/components/segmented-button/directives/segmented-button/segmented-button.directive.mjs +3 -3
  66. package/esm2022/lib/components/forms/components/segmented-button/directives/segmented-button-field/segmented-button-field.directive.mjs +3 -3
  67. package/esm2022/lib/components/forms/components/segmented-button/directives/segmented-button-group/segmented-button-group.directive.mjs +3 -3
  68. package/esm2022/lib/components/forms/components/select/components/combobox/components/combobox/combobox.component.mjs +3 -3
  69. package/esm2022/lib/components/forms/components/select/components/combobox/directives/combobox/combobox.directive.mjs +3 -3
  70. package/esm2022/lib/components/forms/components/select/components/combobox/directives/combobox-body-empty-template/combobox-body-empty-template.directive.mjs +3 -3
  71. package/esm2022/lib/components/forms/components/select/components/combobox/directives/combobox-body-error-template/combobox-body-error-template.directive.mjs +3 -3
  72. package/esm2022/lib/components/forms/components/select/components/combobox/directives/combobox-body-loading-template/combobox-body-loading-template.directive.mjs +3 -3
  73. package/esm2022/lib/components/forms/components/select/components/combobox/directives/combobox-body-more-items-hint-template/combobox-body-more-items-hint-template.directive.mjs +3 -3
  74. package/esm2022/lib/components/forms/components/select/components/combobox/directives/combobox-option-template/combobox-option-template.directive.mjs +3 -3
  75. package/esm2022/lib/components/forms/components/select/components/combobox/directives/combobox-selected-option-template/combobox-selected-option-template.directive.mjs +3 -3
  76. package/esm2022/lib/components/forms/components/select/components/combobox/partials/combobox-body/combobox-body.component.mjs +3 -3
  77. package/esm2022/lib/components/forms/components/select/components/combobox/partials/combobox-option/combobox-option.component.mjs +3 -3
  78. package/esm2022/lib/components/forms/components/select/components/native-select/components/native-select/native-select.component.mjs +3 -3
  79. package/esm2022/lib/components/forms/components/select/components/native-select/directives/native-select-input/native-select-input.directive.mjs +3 -3
  80. package/esm2022/lib/components/forms/components/select/components/native-select/directives/native-select-option/native-select-option.directive.mjs +3 -3
  81. package/esm2022/lib/components/forms/components/select/components/native-select/partials/native-select-option/native-select-option.component.mjs +3 -3
  82. package/esm2022/lib/components/forms/components/select/components/select/components/select/select.component.mjs +3 -3
  83. package/esm2022/lib/components/forms/components/select/components/select/directives/select/select.directive.mjs +3 -3
  84. package/esm2022/lib/components/forms/components/select/components/select/directives/select-body/select-body.directive.mjs +3 -3
  85. package/esm2022/lib/components/forms/components/select/components/select/directives/select-option/select-option.directive.mjs +3 -3
  86. package/esm2022/lib/components/forms/components/select/components/select/partials/select-body/select-body.component.mjs +3 -3
  87. package/esm2022/lib/components/forms/components/select/components/select/partials/select-option/select-option.component.mjs +3 -3
  88. package/esm2022/lib/components/forms/components/select/components/select-field/select-field.component.mjs +3 -3
  89. package/esm2022/lib/components/forms/components/select/directives/select-field/select-field.directive.mjs +3 -3
  90. package/esm2022/lib/components/forms/components/selection-list/components/selection-list-field/selection-list-field.component.mjs +3 -3
  91. package/esm2022/lib/components/forms/components/selection-list/components/selection-list-option/selection-list-option.component.mjs +3 -3
  92. package/esm2022/lib/components/forms/components/selection-list/directives/selection-list-field/selection-list-field.directive.mjs +3 -3
  93. package/esm2022/lib/components/forms/components/selection-list/directives/selection-list-option/selection-list-option.directive.mjs +3 -3
  94. package/esm2022/lib/components/forms/components/slide-toggle/components/slide-toggle/slide-toggle.component.mjs +3 -3
  95. package/esm2022/lib/components/forms/components/slide-toggle/components/slide-toggle-field/slide-toggle-field.component.mjs +3 -3
  96. package/esm2022/lib/components/forms/components/slide-toggle/directives/slide-toggle/slide-toggle.directive.mjs +3 -3
  97. package/esm2022/lib/components/forms/components/slider/components/slider/slider.component.mjs +3 -3
  98. package/esm2022/lib/components/forms/components/slider/components/slider-field/slider-field.component.mjs +3 -3
  99. package/esm2022/lib/components/forms/directives/dynamic-form-field/dynamic-form-field.directive.mjs +3 -3
  100. package/esm2022/lib/components/forms/directives/dynamic-form-group/dynamic-form-group.directive.mjs +3 -3
  101. package/esm2022/lib/components/forms/directives/expose-input-vars/expose-input-vars.directive.mjs +3 -3
  102. package/esm2022/lib/components/forms/directives/if-input-empty/if-input-empty.directive.mjs +3 -3
  103. package/esm2022/lib/components/forms/directives/if-input-filled/if-input-filled.directive.mjs +3 -3
  104. package/esm2022/lib/components/forms/directives/input/input.directive.mjs +3 -3
  105. package/esm2022/lib/components/forms/directives/input-prefix/input-prefix.directive.mjs +3 -3
  106. package/esm2022/lib/components/forms/directives/input-suffix/input-suffix.directive.mjs +3 -3
  107. package/esm2022/lib/components/forms/directives/native-input-ref/native-input-ref.directive.mjs +3 -3
  108. package/esm2022/lib/components/forms/directives/static-form-field/static-form-field.directive.mjs +3 -3
  109. package/esm2022/lib/components/forms/directives/static-form-group/static-form-group.directive.mjs +3 -3
  110. package/esm2022/lib/components/forms/directives/writeable-input/writeable-input.directive.mjs +3 -3
  111. package/esm2022/lib/components/forms/services/default-validator-errors.service.mjs +3 -3
  112. package/esm2022/lib/components/forms/services/form-field-state.service.mjs +3 -3
  113. package/esm2022/lib/components/forms/services/form-group-state.service.mjs +3 -3
  114. package/esm2022/lib/components/forms/services/input-state.service.mjs +3 -3
  115. package/esm2022/lib/components/forms/utils/decorated-form-field.base.mjs +3 -3
  116. package/esm2022/lib/components/forms/utils/decorated-input.base.mjs +3 -3
  117. package/esm2022/lib/components/forms/utils/input.base.mjs +3 -3
  118. package/esm2022/lib/components/icons/chevron-icon/chevron-icon.component.mjs +3 -3
  119. package/esm2022/lib/components/masonry/components/masonry/masonry.component.mjs +3 -3
  120. package/esm2022/lib/components/masonry/partials/masonry-item/masonry-item.component.mjs +3 -3
  121. package/esm2022/lib/components/overlay/components/bottom-sheet/components/bottom-sheet-container/bottom-sheet-container.component.mjs +3 -3
  122. package/esm2022/lib/components/overlay/components/bottom-sheet/partials/bottom-sheet-container-base/bottom-sheet-container-base.component.mjs +3 -3
  123. package/esm2022/lib/components/overlay/components/bottom-sheet/partials/bottom-sheet-drag-handle/bottom-sheet-drag-handle.component.mjs +3 -3
  124. package/esm2022/lib/components/overlay/components/bottom-sheet/partials/bottom-sheet-title/bottom-sheet-title.directive.mjs +3 -3
  125. package/esm2022/lib/components/overlay/components/bottom-sheet/services/bottom-sheet-base.service.mjs +3 -3
  126. package/esm2022/lib/components/overlay/components/bottom-sheet/services/bottom-sheet-swipe-handler.service.mjs +3 -3
  127. package/esm2022/lib/components/overlay/components/bottom-sheet/services/bottom-sheet.service.mjs +3 -3
  128. package/esm2022/lib/components/overlay/components/dialog/components/dialog-container/dialog-container.component.mjs +3 -3
  129. package/esm2022/lib/components/overlay/components/dialog/partials/dialog-close/dialog-close.directive.mjs +3 -3
  130. package/esm2022/lib/components/overlay/components/dialog/partials/dialog-container-base/dialog-container-base.component.mjs +3 -3
  131. package/esm2022/lib/components/overlay/components/dialog/partials/dialog-title/dialog-title.directive.mjs +3 -3
  132. package/esm2022/lib/components/overlay/components/dialog/services/dialog-base.service.mjs +3 -3
  133. package/esm2022/lib/components/overlay/components/dialog/services/dialog.service.mjs +3 -3
  134. package/esm2022/lib/components/overlay/components/menu/context-menu-trigger.mjs +6 -6
  135. package/esm2022/lib/components/overlay/components/menu/menu-aim.mjs +6 -6
  136. package/esm2022/lib/components/overlay/components/menu/menu-bar.mjs +3 -3
  137. package/esm2022/lib/components/overlay/components/menu/menu-base.mjs +3 -3
  138. package/esm2022/lib/components/overlay/components/menu/menu-group.mjs +3 -3
  139. package/esm2022/lib/components/overlay/components/menu/menu-item-checkbox.mjs +3 -3
  140. package/esm2022/lib/components/overlay/components/menu/menu-item-radio.mjs +3 -3
  141. package/esm2022/lib/components/overlay/components/menu/menu-item-selectable.mjs +3 -3
  142. package/esm2022/lib/components/overlay/components/menu/menu-item.mjs +3 -3
  143. package/esm2022/lib/components/overlay/components/menu/menu-module.mjs +4 -4
  144. package/esm2022/lib/components/overlay/components/menu/menu-stack.mjs +3 -3
  145. package/esm2022/lib/components/overlay/components/menu/menu-trigger-base.mjs +3 -3
  146. package/esm2022/lib/components/overlay/components/menu/menu-trigger.mjs +3 -3
  147. package/esm2022/lib/components/overlay/components/menu/menu.mjs +3 -3
  148. package/esm2022/lib/components/overlay/components/overlay/components/overlay-container/overlay-container.component.mjs +3 -3
  149. package/esm2022/lib/components/overlay/components/overlay/partials/overlay-close/overlay-close.directive.mjs +3 -3
  150. package/esm2022/lib/components/overlay/components/overlay/partials/overlay-title/overlay-title.directive.mjs +3 -3
  151. package/esm2022/lib/components/overlay/components/overlay/services/overlay.service.mjs +3 -3
  152. package/esm2022/lib/components/overlay/components/toggletip/components/toggletip/toggletip.component.mjs +3 -3
  153. package/esm2022/lib/components/overlay/components/toggletip/directives/toggletip/toggletip.directive.mjs +3 -3
  154. package/esm2022/lib/components/overlay/components/tooltip/components/tooltip/tooltip.component.mjs +3 -3
  155. package/esm2022/lib/components/overlay/components/tooltip/directives/tooltip/tooltip.directive.mjs +3 -3
  156. package/esm2022/lib/components/overlay/directives/dynamic-overlay-title/dynamic-overlay-title.directive.mjs +3 -3
  157. package/esm2022/lib/components/overlay/directives/overlay-close-auto-blocker/overlay-close-auto-blocker.directive.mjs +3 -3
  158. package/esm2022/lib/components/overlay/services/dynamic-overlay.service.mjs +3 -3
  159. package/esm2022/lib/components/pagination/components/pagination/pagination.component.mjs +3 -3
  160. package/esm2022/lib/components/pagination/partials/pagination-link/pagination-link.directive.mjs +3 -3
  161. package/esm2022/lib/components/pagination/services/pagination-head.service.mjs +3 -3
  162. package/esm2022/lib/components/picture/picture-data.directive.mjs +3 -3
  163. package/esm2022/lib/components/picture/picture.component.mjs +3 -3
  164. package/esm2022/lib/components/picture/pipes/normalize-source/normalize-source.pipe.mjs +3 -3
  165. package/esm2022/lib/components/progress-spinner/progress-spinner.component.mjs +3 -3
  166. package/esm2022/lib/components/query-error/components/query-error/query-error.component.mjs +3 -3
  167. package/esm2022/lib/components/query-error/directives/query-error/query-error.directive.mjs +3 -3
  168. package/esm2022/lib/components/scrollable/components/scrollable/scrollable.component.mjs +299 -168
  169. package/esm2022/lib/components/scrollable/directives/index.mjs +2 -0
  170. package/esm2022/lib/components/scrollable/directives/public-api.mjs +2 -0
  171. package/esm2022/lib/components/scrollable/directives/scrollable-ignore-child/public-api.mjs +2 -0
  172. package/esm2022/lib/components/scrollable/directives/scrollable-ignore-child/scrollable-ignore-child.directive.mjs +44 -0
  173. package/esm2022/lib/components/scrollable/public-api.mjs +2 -1
  174. package/esm2022/lib/components/skeleton/components/skeleton/skeleton.component.mjs +3 -3
  175. package/esm2022/lib/components/skeleton/partials/skeleton-item/skeleton-item.component.mjs +3 -3
  176. package/esm2022/lib/components/sort/components/sort-header/sort-header.component.mjs +3 -3
  177. package/esm2022/lib/components/sort/partials/sort/sort.directive.mjs +3 -3
  178. package/esm2022/lib/components/sort/services/sort-header-intl.mjs +3 -3
  179. package/esm2022/lib/components/table/components/table/table.component.mjs +3 -3
  180. package/esm2022/lib/components/table/partials/cells/cell/cell.directive.mjs +3 -3
  181. package/esm2022/lib/components/table/partials/cells/cell-def/cell-def.directive.mjs +3 -3
  182. package/esm2022/lib/components/table/partials/cells/column-def/column-def.directive.mjs +3 -3
  183. package/esm2022/lib/components/table/partials/cells/footer-cell/footer-cell.directive.mjs +3 -3
  184. package/esm2022/lib/components/table/partials/cells/footer-cell-def/footer-cell-def.directive.mjs +3 -3
  185. package/esm2022/lib/components/table/partials/cells/header-cell/header-cell.directive.mjs +3 -3
  186. package/esm2022/lib/components/table/partials/cells/header-cell-def/header-cell-def.directive.mjs +3 -3
  187. package/esm2022/lib/components/table/partials/cells/text-column/text-column.component.mjs +3 -3
  188. package/esm2022/lib/components/table/partials/rows/footer-row/footer-row.component.mjs +3 -3
  189. package/esm2022/lib/components/table/partials/rows/footer-row-def/footer-row-def.directive.mjs +3 -3
  190. package/esm2022/lib/components/table/partials/rows/header-row/header-row.component.mjs +3 -3
  191. package/esm2022/lib/components/table/partials/rows/header-row-def/header-row-def.directive.mjs +3 -3
  192. package/esm2022/lib/components/table/partials/rows/no-data-row/no-data-row.directive.mjs +3 -3
  193. package/esm2022/lib/components/table/partials/rows/recycle-rows/recycle-rows.directive.mjs +3 -3
  194. package/esm2022/lib/components/table/partials/rows/row/row.component.mjs +3 -3
  195. package/esm2022/lib/components/table/partials/rows/row-def/row-def.directive.mjs +3 -3
  196. package/esm2022/lib/components/table/partials/table-busy/table-busy.directive.mjs +3 -3
  197. package/esm2022/lib/components/table/partials/table-busy-outlet/table-busy-outlet.directive.mjs +3 -3
  198. package/esm2022/lib/components/tabs/components/inline-tabs/inline-tabs.component.mjs +3 -3
  199. package/esm2022/lib/components/tabs/components/nav-tabs/nav-tabs.component.mjs +3 -3
  200. package/esm2022/lib/components/tabs/partials/inline-tabs/inline-tab/inline-tab.component.mjs +3 -3
  201. package/esm2022/lib/components/tabs/partials/inline-tabs/inline-tab-body/inline-tab-body.component.mjs +3 -3
  202. package/esm2022/lib/components/tabs/partials/inline-tabs/inline-tab-body-host/inline-tab-body-host.directive.mjs +3 -3
  203. package/esm2022/lib/components/tabs/partials/inline-tabs/inline-tab-content/inline-tab-content.directive.mjs +3 -3
  204. package/esm2022/lib/components/tabs/partials/inline-tabs/inline-tab-header/inline-tab-header.component.mjs +3 -3
  205. package/esm2022/lib/components/tabs/partials/inline-tabs/inline-tab-label/inline-tab-label.directive.mjs +3 -3
  206. package/esm2022/lib/components/tabs/partials/inline-tabs/inline-tab-label-wrapper/inline-tab-label-wrapper.directive.mjs +3 -3
  207. package/esm2022/lib/components/tabs/partials/nav-tabs/nav-tab-link/nav-tab-link.directive.mjs +3 -3
  208. package/esm2022/lib/components/tabs/partials/nav-tabs/nav-tabs-outlet/nav-tabs-outlet.component.mjs +3 -3
  209. package/esm2022/lib/components/tabs/utils/active-tab-underline.util.mjs +3 -3
  210. package/esm2022/lib/components/tabs/utils/paginated-tab-header.directive.mjs +3 -3
  211. package/esm2022/lib/services/swipe-handler.service.mjs +3 -3
  212. package/fesm2022/ethlete-cdk.mjs +957 -787
  213. package/fesm2022/ethlete-cdk.mjs.map +1 -1
  214. package/lib/components/scrollable/components/scrollable/scrollable.component.d.ts +37 -32
  215. package/lib/components/scrollable/directives/index.d.ts +1 -0
  216. package/lib/components/scrollable/directives/public-api.d.ts +1 -0
  217. package/lib/components/scrollable/directives/scrollable-ignore-child/public-api.d.ts +1 -0
  218. package/lib/components/scrollable/directives/scrollable-ignore-child/scrollable-ignore-child.directive.d.ts +16 -0
  219. package/lib/components/scrollable/public-api.d.ts +1 -0
  220. package/package.json +7 -7
@@ -1,9 +1,23 @@
1
1
  import { NgClass } from '@angular/common';
2
2
  import { ChangeDetectionStrategy, Component, ContentChildren, ElementRef, EventEmitter, Input, Output, ViewChild, ViewEncapsulation, booleanAttribute, computed, effect, inject, isDevMode, numberAttribute, signal, } from '@angular/core';
3
- import { CursorDragScrollDirective, IS_ACTIVE_ELEMENT, IS_ELEMENT, LetDirective, ObserveScrollStateDirective, TypedQueryList, createDestroy, getElementVisibleStates, isElementVisible, nextFrame, scrollToElement, signalElementIntersection, signalElementScrollState, signalHostAttributes, signalHostClasses, } from '@ethlete/core';
4
- import { BehaviorSubject, debounceTime, fromEvent, merge, of, startWith, takeUntil, tap } from 'rxjs';
3
+ import { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';
4
+ import { CursorDragScrollDirective, IS_ACTIVE_ELEMENT, LetDirective, ObserveScrollStateDirective, TypedQueryList, getFirstAndLastPartialIntersection, isElementVisible, nextFrame, scrollToElement, signalElementChildren, signalElementIntersection, signalElementScrollState, signalHostAttributes, signalHostClasses, } from '@ethlete/core';
5
+ import { BehaviorSubject, Subject, combineLatest, debounceTime, filter, fromEvent, map, take, takeUntil, tap, } from 'rxjs';
5
6
  import { ChevronIconComponent } from '../../../icons';
7
+ import { ScrollableIgnoreChildDirective, isScrollableChildIgnored } from '../../directives';
6
8
  import * as i0 from "@angular/core";
9
+ // Thresholds for the intersection observer.
10
+ const ELEMENT_INTERSECTION_THRESHOLD = [
11
+ // We use 51 steps to get a 2% step size.
12
+ ...Array.from({ length: 51 }, (_, i) => i / 50),
13
+ // Additional steps needed since display scaling can cause the intersection ratio to be slightly off.
14
+ 0.01,
15
+ 0.005,
16
+ 0.001,
17
+ 0.99,
18
+ 0.995,
19
+ 0.999,
20
+ ];
7
21
  export class ScrollableComponent {
8
22
  set _itemSize(v) {
9
23
  this.itemSize.set(v);
@@ -17,6 +31,9 @@ export class ScrollableComponent {
17
31
  set _scrollableClass(v) {
18
32
  this.scrollableClass.set(v);
19
33
  }
34
+ set _renderNavigation(v) {
35
+ this.renderNavigation.set(v);
36
+ }
20
37
  set _renderMasks(v) {
21
38
  this.renderMasks.set(v);
22
39
  }
@@ -47,6 +64,9 @@ export class ScrollableComponent {
47
64
  set _scrollable(e) {
48
65
  this.scrollable.set(e);
49
66
  }
67
+ set _scrollableContainer(e) {
68
+ this.scrollableContainer.set(e);
69
+ }
50
70
  set _firstElement(e) {
51
71
  this.firstElement.set(e);
52
72
  }
@@ -56,65 +76,14 @@ export class ScrollableComponent {
56
76
  set _activeElementList(e) {
57
77
  this.activeElementList.set(e);
58
78
  }
59
- set _elementList(e) {
60
- this.elementList.set(e);
61
- }
62
- get highestVisibleIntersection() {
63
- const elements = this._latestVisibilityStates$.value;
64
- if (!elements.length) {
65
- return null;
66
- }
67
- return elements.reduce((prev, curr) => {
68
- if (!prev) {
69
- return curr;
70
- }
71
- return curr.intersectionRatio > prev.intersectionRatio ? curr : prev;
72
- });
73
- }
74
- get nextPartialIntersection() {
75
- const elements = this._latestVisibilityStates$.value;
76
- if (!elements.length) {
77
- return null;
78
- }
79
- const highestVisibleIntersection = this.highestVisibleIntersection;
80
- if (!highestVisibleIntersection) {
81
- return null;
82
- }
83
- const nextIndex = elements.slice(highestVisibleIntersection.index).findIndex((e) => !e.isIntersecting);
84
- if (nextIndex === -1) {
85
- return null;
86
- }
87
- const nextElement = elements[highestVisibleIntersection.index + nextIndex];
88
- return nextElement || null;
89
- }
90
- get previousPartialIntersection() {
91
- const elements = this._latestVisibilityStates$.value;
92
- if (!elements.length) {
93
- return null;
94
- }
95
- const highestVisibleIntersection = this.highestVisibleIntersection;
96
- if (!highestVisibleIntersection) {
97
- return null;
98
- }
99
- const previousIndex = elements
100
- .slice(0, highestVisibleIntersection.index)
101
- .reverse()
102
- .findIndex((e) => !e.isIntersecting);
103
- if (previousIndex === -1) {
104
- return null;
105
- }
106
- const previousElement = elements[highestVisibleIntersection.index - previousIndex - 1];
107
- return previousElement || null;
108
- }
109
79
  constructor() {
110
- this._destroy$ = createDestroy();
111
80
  this._elementRef = inject(ElementRef);
112
81
  this._isCursorDragging$ = new BehaviorSubject(false);
113
- this._latestVisibilityStates$ = new BehaviorSubject([]);
114
82
  this.itemSize = signal('auto');
115
83
  this.direction = signal('horizontal');
116
84
  this.scrollableRole = signal(null);
117
85
  this.scrollableClass = signal(null);
86
+ this.renderNavigation = signal(false);
118
87
  this.renderMasks = signal(true);
119
88
  this.renderButtons = signal(true);
120
89
  this.renderScrollbars = signal(false);
@@ -127,15 +96,24 @@ export class ScrollableComponent {
127
96
  this.scrollStateChange = new EventEmitter();
128
97
  this.intersectionChange = new EventEmitter();
129
98
  this.scrollable = signal(null);
99
+ this.scrollableContainer = signal(null);
130
100
  this.firstElement = signal(null);
131
101
  this.lastElement = signal(null);
132
102
  this.activeElementList = signal(null);
133
- this.elementList = signal(null);
134
103
  this.containerScrollState = signalElementScrollState(this.scrollable);
135
- this.firstElementIntersection = signalElementIntersection(this.firstElement);
104
+ this.firstElementIntersection = signalElementIntersection(this.firstElement, { root: this.scrollable });
136
105
  this.firstElementVisibility = signal(null);
137
- this.lastElementIntersection = signalElementIntersection(this.lastElement);
106
+ this.lastElementIntersection = signalElementIntersection(this.lastElement, { root: this.scrollable });
138
107
  this.lastElementVisibility = signal(null);
108
+ this.allScrollableChildren = signalElementChildren(this.scrollableContainer);
109
+ this.scrollableChildren = computed(() => this.allScrollableChildren().filter((c) => !isScrollableChildIgnored(c)));
110
+ this._disableSnapping$ = new Subject();
111
+ this.scrollableContentIntersections = signalElementIntersection(this.scrollableChildren, {
112
+ root: this.scrollable,
113
+ threshold: ELEMENT_INTERSECTION_THRESHOLD,
114
+ });
115
+ this.scrollableContentIntersections$ = toObservable(this.scrollableContentIntersections);
116
+ this.manualActiveNavigationIndex = signal(null);
139
117
  this.canScroll = computed(() => {
140
118
  const dir = this.direction();
141
119
  if (dir === 'horizontal') {
@@ -147,7 +125,7 @@ export class ScrollableComponent {
147
125
  if (!this.canScroll()) {
148
126
  return true;
149
127
  }
150
- const intersection = this.firstElementIntersection();
128
+ const intersection = this.firstElementIntersection()[0];
151
129
  if (!intersection) {
152
130
  return this.firstElementVisibility()?.inline ?? true;
153
131
  }
@@ -157,20 +135,44 @@ export class ScrollableComponent {
157
135
  if (!this.canScroll()) {
158
136
  return true;
159
137
  }
160
- const intersection = this.lastElementIntersection();
138
+ const intersection = this.lastElementIntersection()[0];
161
139
  if (!intersection) {
162
140
  return this.lastElementVisibility()?.inline ?? true;
163
141
  }
164
142
  return intersection.isIntersecting;
165
143
  });
166
144
  this.enableOverlayAnimations = signal(false);
167
- this.hostAttributes = signalHostAttributes({
145
+ this._initialScrollableNavigation = signal([]);
146
+ this.scrollableNavigation = computed(() => {
147
+ const allIntersections = this.scrollableContentIntersections();
148
+ const manualActiveNavigationIndex = this.manualActiveNavigationIndex();
149
+ const initialScrollableNavigation = this._initialScrollableNavigation();
150
+ if (!allIntersections.length) {
151
+ return initialScrollableNavigation;
152
+ }
153
+ const highestIntersection = allIntersections.reduce((prev, curr) => {
154
+ if (prev && prev.intersectionRatio > curr.intersectionRatio) {
155
+ return prev;
156
+ }
157
+ return curr;
158
+ }, allIntersections[0]);
159
+ if (!highestIntersection) {
160
+ return [];
161
+ }
162
+ return allIntersections.map((i, index) => ({
163
+ isActive: manualActiveNavigationIndex !== null
164
+ ? manualActiveNavigationIndex === index
165
+ : i === highestIntersection && highestIntersection.intersectionRatio > 0,
166
+ element: i.target,
167
+ }));
168
+ });
169
+ this.hostAttributeBindings = signalHostAttributes({
168
170
  'item-size': this.itemSize,
169
171
  direction: this.direction,
170
172
  'render-scrollbars': this.renderScrollbars,
171
173
  'sticky-buttons': this.stickyButtons,
172
174
  });
173
- this.hostClasses = signalHostClasses({
175
+ this.hostClassBindings = signalHostClasses({
174
176
  'et-scrollable--can-scroll': this.canScroll,
175
177
  'et-scrollable--is-at-start': this.isAtStart,
176
178
  'et-scrollable--is-at-end': this.isAtEnd,
@@ -188,8 +190,8 @@ export class ScrollableComponent {
188
190
  if (firstActive && !this.disableActiveElementScrolling()) {
189
191
  const offsetTop = firstActive.elementRef.nativeElement.offsetTop - scrollable.offsetTop;
190
192
  const offsetLeft = firstActive.elementRef.nativeElement.offsetLeft - scrollable.offsetLeft;
191
- scrollable.scrollLeft = offsetLeft;
192
- scrollable.scrollTop = offsetTop;
193
+ scrollable.scrollLeft = offsetLeft - this.scrollMargin();
194
+ scrollable.scrollTop = offsetTop - this.scrollMargin();
193
195
  }
194
196
  this.firstElementVisibility.set(isElementVisible({
195
197
  container: scrollable,
@@ -202,6 +204,23 @@ export class ScrollableComponent {
202
204
  // We need to wait one frame before enabling animations to prevent a animation from playing during initial render.
203
205
  nextFrame(() => this.enableOverlayAnimations.set(true));
204
206
  }, { allowSignalWrites: true });
207
+ effect(() => {
208
+ const elementList = this.scrollableChildren();
209
+ const scrollable = this.scrollable()?.nativeElement;
210
+ const renderNavigation = this.renderNavigation();
211
+ if (!elementList || !scrollable || !renderNavigation) {
212
+ return;
213
+ }
214
+ const firstVisibleElement = elementList.find((e) => isElementVisible({ container: scrollable, element: e }));
215
+ if (!firstVisibleElement) {
216
+ return;
217
+ }
218
+ const initialNavigationStuff = elementList.map((e) => ({
219
+ isActive: e === firstVisibleElement,
220
+ element: e,
221
+ }));
222
+ this._initialScrollableNavigation.set(initialNavigationStuff);
223
+ }, { allowSignalWrites: true });
205
224
  effect(() => {
206
225
  const isAtStart = this.isAtStart();
207
226
  const isAtEnd = this.isAtEnd();
@@ -212,9 +231,25 @@ export class ScrollableComponent {
212
231
  isAtStart: !!isAtStart,
213
232
  });
214
233
  });
215
- }
216
- ngAfterContentInit() {
217
- this._setupScrollListening();
234
+ effect(() => {
235
+ const enableSnapping = this.snap();
236
+ if (enableSnapping) {
237
+ this._enableSnapping();
238
+ }
239
+ else {
240
+ this._disableSnapping();
241
+ }
242
+ });
243
+ this.scrollableContentIntersections$
244
+ .pipe(takeUntilDestroyed(), debounceTime(10), tap((entries) => {
245
+ this.intersectionChange.emit(entries.map((i, index) => ({
246
+ index,
247
+ element: i.target,
248
+ intersectionRatio: i.intersectionRatio,
249
+ isIntersecting: i.isIntersecting,
250
+ })));
251
+ }))
252
+ .subscribe();
218
253
  }
219
254
  scrollOneContainerSize(direction) {
220
255
  const scrollElement = this.scrollable()?.nativeElement;
@@ -222,38 +257,129 @@ export class ScrollableComponent {
222
257
  return;
223
258
  }
224
259
  const parent = this._elementRef.nativeElement;
225
- const scrollableSize = this.direction() === 'horizontal' ? parent.clientWidth : parent.clientHeight;
226
- const currentScroll = this.direction() === 'horizontal' ? scrollElement.scrollLeft : scrollElement.scrollTop;
227
- scrollElement.scrollTo({
228
- [this.direction() === 'horizontal' ? 'left' : 'top']: currentScroll + (direction === 'start' ? -scrollableSize : scrollableSize),
229
- behavior: 'smooth',
230
- });
260
+ const isSnappingEnabled = this.snap();
261
+ if (isSnappingEnabled) {
262
+ // If snapping is enabled we want to scroll to a position where no further snapping will happen after the scroll.
263
+ const allIntersections = this.scrollableContentIntersections();
264
+ const intersections = getFirstAndLastPartialIntersection(allIntersections);
265
+ const relevantIntersection = direction === 'start' ? intersections?.first : intersections?.last;
266
+ if (!relevantIntersection)
267
+ return;
268
+ const nextIndex = relevantIntersection.intersection.intersectionRatio !== 1
269
+ ? relevantIntersection.index
270
+ : direction === 'start'
271
+ ? relevantIntersection.index - 1
272
+ : relevantIntersection.index + 1;
273
+ const element = allIntersections[nextIndex]?.target ||
274
+ relevantIntersection.intersection.target;
275
+ this.scrollToElement({
276
+ element: element,
277
+ origin: direction === 'end' ? 'start' : 'end',
278
+ });
279
+ }
280
+ else {
281
+ // Just scroll one size of the scrollable container.
282
+ const scrollableSize = this.direction() === 'horizontal' ? parent.clientWidth : parent.clientHeight;
283
+ const currentScroll = this.direction() === 'horizontal' ? scrollElement.scrollLeft : scrollElement.scrollTop;
284
+ scrollElement.scrollTo({
285
+ [this.direction() === 'horizontal' ? 'left' : 'top']: currentScroll + (direction === 'start' ? -scrollableSize : scrollableSize),
286
+ behavior: 'smooth',
287
+ });
288
+ }
231
289
  }
232
290
  scrollOneItemSize(direction) {
233
- const elements = this._latestVisibilityStates$.value;
234
- if (!elements.length) {
291
+ const allIntersections = this.scrollableContentIntersections();
292
+ const scrollElement = this.scrollable()?.nativeElement;
293
+ if (!allIntersections.length) {
235
294
  if (isDevMode()) {
236
295
  console.warn('No elements found to scroll to. Make sure to apply the isElement directive to the elements you want to scroll to.');
237
296
  }
238
297
  return;
239
298
  }
240
- const el = direction === 'start' ? this.previousPartialIntersection : this.nextPartialIntersection;
241
- if (!el) {
299
+ const intersections = getFirstAndLastPartialIntersection(allIntersections);
300
+ if (!intersections || !scrollElement)
242
301
  return;
302
+ // Means the current element is bigger than the scrollable container.
303
+ // In this case we should scroll to the start of the current element. If we are already there we should scroll to the end of the previous element.
304
+ // This applies to the other direction as well.
305
+ const isFirstAndLastIntersectionEqual = intersections.first.intersection === intersections.last.intersection;
306
+ const scrollableRect = scrollElement.getBoundingClientRect();
307
+ if (isFirstAndLastIntersectionEqual) {
308
+ const intersection = intersections.first.intersection.target.getBoundingClientRect();
309
+ const isStartOfElementVisible = this.direction() === 'horizontal'
310
+ ? intersection.left >= scrollableRect.left
311
+ : intersection.top >= scrollableRect.top;
312
+ const isEndOfElementVisible = this.direction() === 'horizontal'
313
+ ? intersection.right <= scrollableRect.right
314
+ : intersection.bottom <= scrollableRect.bottom;
315
+ if (!isStartOfElementVisible || !isEndOfElementVisible) {
316
+ if (direction === 'start') {
317
+ if (isStartOfElementVisible) {
318
+ // to the end of the previous element
319
+ const previousIndex = intersections.first.index - 1;
320
+ const elementToScrollTo = allIntersections[previousIndex]?.target;
321
+ if (!elementToScrollTo)
322
+ return;
323
+ this.scrollToElement({
324
+ element: elementToScrollTo,
325
+ origin: 'end',
326
+ });
327
+ }
328
+ else {
329
+ // to the start of the current element
330
+ this.scrollToElement({
331
+ element: intersections.first.intersection.target,
332
+ origin: 'start',
333
+ });
334
+ }
335
+ }
336
+ else {
337
+ if (isEndOfElementVisible) {
338
+ // to the start of the next element
339
+ const nextIndex = intersections.last.index + 1;
340
+ const elementToScrollTo = allIntersections[nextIndex]?.target;
341
+ if (!elementToScrollTo)
342
+ return;
343
+ this.scrollToElement({
344
+ element: elementToScrollTo,
345
+ origin: 'start',
346
+ });
347
+ }
348
+ else {
349
+ // to the end of the current element
350
+ this.scrollToElement({
351
+ element: intersections.last.intersection.target,
352
+ origin: 'end',
353
+ });
354
+ }
355
+ }
356
+ return;
357
+ }
358
+ }
359
+ const data = direction === 'start' ? intersections.first : intersections.last;
360
+ let elementToScrollTo = data.intersection.target;
361
+ if (data.intersection.intersectionRatio === 1) {
362
+ if (direction === 'start' && data.index === 0) {
363
+ return;
364
+ }
365
+ if (direction === 'end' && data.index === allIntersections.length - 1) {
366
+ return;
367
+ }
368
+ const nextIndex = direction === 'start' ? data.index - 1 : data.index + 1;
369
+ elementToScrollTo = allIntersections[nextIndex]?.target;
243
370
  }
371
+ if (!elementToScrollTo)
372
+ return;
244
373
  this.scrollToElement({
245
- element: el?.element,
246
- direction: this.direction() === 'horizontal' ? 'inline' : 'block',
374
+ element: elementToScrollTo,
247
375
  origin: direction,
248
- ...(this.direction() === 'horizontal'
249
- ? { scrollInlineMargin: this.scrollMargin() }
250
- : { scrollBlockMargin: this.scrollMargin() }),
251
376
  });
252
377
  }
253
378
  scrollToElement(options) {
254
379
  const scrollElement = this.scrollable()?.nativeElement;
255
380
  scrollToElement({
256
381
  container: scrollElement,
382
+ direction: this.direction() === 'horizontal' ? 'inline' : 'block',
257
383
  ...(this.direction() === 'horizontal'
258
384
  ? { scrollInlineMargin: this.scrollMargin() }
259
385
  : { scrollBlockMargin: this.scrollMargin() }),
@@ -261,7 +387,7 @@ export class ScrollableComponent {
261
387
  });
262
388
  }
263
389
  scrollToElementByIndex(options) {
264
- const elements = this.elementList()?.toArray() ?? [];
390
+ const elements = this.scrollableChildren();
265
391
  if (!elements.length) {
266
392
  if (isDevMode()) {
267
393
  console.warn('No elements found to scroll to. Make sure to apply the isElement directive to the elements you want to scroll to.');
@@ -269,7 +395,7 @@ export class ScrollableComponent {
269
395
  return;
270
396
  }
271
397
  const scrollElement = this.scrollable()?.nativeElement;
272
- const element = elements[options.index]?.elementRef.nativeElement;
398
+ const element = elements[options.index];
273
399
  scrollToElement({
274
400
  container: scrollElement,
275
401
  element,
@@ -279,6 +405,18 @@ export class ScrollableComponent {
279
405
  ...options,
280
406
  });
281
407
  }
408
+ scrollToElementViaNavigation(elementIndex, element) {
409
+ this.manualActiveNavigationIndex.set(elementIndex);
410
+ this.scrollToElement({
411
+ element,
412
+ });
413
+ const scrollElement = this.scrollable()?.nativeElement;
414
+ if (!scrollElement)
415
+ return;
416
+ fromEvent(scrollElement, 'scroll')
417
+ .pipe(debounceTime(50), take(1), tap(() => this.manualActiveNavigationIndex.set(null)))
418
+ .subscribe();
419
+ }
282
420
  setIsCursorDragging(isDragging) {
283
421
  this._isCursorDragging$.next(isDragging);
284
422
  }
@@ -290,7 +428,7 @@ export class ScrollableComponent {
290
428
  this.scrollOneItemSize('start');
291
429
  }
292
430
  }
293
- scrollToStartEnd() {
431
+ scrollToEndDirection() {
294
432
  if (this.scrollMode() === 'container') {
295
433
  this.scrollOneContainerSize('end');
296
434
  }
@@ -298,95 +436,85 @@ export class ScrollableComponent {
298
436
  this.scrollOneItemSize('end');
299
437
  }
300
438
  }
301
- _setupScrollListening() {
302
- const scrollElement = this.scrollable()?.nativeElement;
303
- const elements = this.elementList();
304
- if (!scrollElement || !elements) {
305
- return;
306
- }
307
- let isSnapping = false;
308
- let snapTimeout = 0;
309
- merge(fromEvent(scrollElement, 'wheel'), fromEvent(scrollElement, 'touchstart'))
310
- .pipe(takeUntil(this._destroy$), tap(() => {
311
- isSnapping = false;
312
- }))
313
- .subscribe();
314
- merge(fromEvent(scrollElement, 'scroll'), this._isCursorDragging$, elements.changes.pipe(startWith(elements)) ?? of(null))
315
- .pipe(debounceTime(300), takeUntil(this._destroy$), tap(() => {
316
- const els = elements
317
- ?.toArray()
318
- .map((e) => e?.elementRef.nativeElement)
319
- .filter((e) => !!e) ?? [];
320
- if (!els.length) {
321
- this._latestVisibilityStates$.next([]);
439
+ _enableSnapping() {
440
+ combineLatest([this.scrollableContentIntersections$, this._isCursorDragging$])
441
+ .pipe(filter(([, isDragging]) => !isDragging), map(([intersections]) => intersections), debounceTime(150), tap((allIntersections) => {
442
+ const scrollElement = this.scrollable()?.nativeElement;
443
+ if (!scrollElement)
322
444
  return;
323
- }
324
- const states = getElementVisibleStates({
325
- elements: els,
326
- container: scrollElement,
327
- });
328
- const prop = this.direction() === 'horizontal' ? 'inlineIntersection' : 'blockIntersection';
329
- const stateClass = `et-element--is-intersecting`;
330
- for (const state of states) {
331
- if (state[prop] === 100) {
332
- state.element.classList.add(stateClass);
445
+ const intersections = getFirstAndLastPartialIntersection(allIntersections);
446
+ if (!intersections)
447
+ return;
448
+ const isFirstAndLastIntersectionEqual = intersections.first.intersection === intersections.last.intersection;
449
+ const scrollableRect = scrollElement.getBoundingClientRect();
450
+ if (isFirstAndLastIntersectionEqual) {
451
+ const intersection = intersections.first.intersection.target.getBoundingClientRect();
452
+ const isStartOfElementVisible = this.direction() === 'horizontal'
453
+ ? intersection.left >= scrollableRect.left
454
+ : intersection.top >= scrollableRect.top;
455
+ const isEndOfElementVisible = this.direction() === 'horizontal'
456
+ ? intersection.right <= scrollableRect.right
457
+ : intersection.bottom <= scrollableRect.bottom;
458
+ // Don't snap if neither the start nor the end of the current element is visible.
459
+ // Otherwise this would result in parts of the element being inaccessible.
460
+ if (!isStartOfElementVisible && !isEndOfElementVisible)
461
+ return;
462
+ // If the start of the element is visible we should snap to the start.
463
+ if (isStartOfElementVisible) {
464
+ this.scrollToElement({
465
+ element: intersections.first.intersection.target,
466
+ origin: 'start',
467
+ });
468
+ return;
333
469
  }
334
- else {
335
- state.element.classList.remove(stateClass);
470
+ // If the end of the element is visible we should snap to the end.
471
+ if (isEndOfElementVisible) {
472
+ this.scrollToElement({
473
+ element: intersections.last.intersection.target,
474
+ origin: 'end',
475
+ });
476
+ return;
336
477
  }
337
478
  }
338
- const intersectionChanges = states.map((s, i) => {
339
- const state = {
340
- element: s.element,
341
- intersectionRatio: s[prop] / 100,
342
- isIntersecting: s[prop] === 100,
343
- index: i,
344
- };
345
- return state;
346
- });
347
- this.intersectionChange.emit(intersectionChanges);
348
- this._latestVisibilityStates$.next(intersectionChanges);
349
- if (isSnapping || this._isCursorDragging$.value || !this.snap())
350
- return;
351
- const prev = this.previousPartialIntersection;
352
- const next = this.nextPartialIntersection;
353
- const skipSnap = !prev ||
354
- !next ||
355
- prev.intersectionRatio === 0 ||
356
- next.intersectionRatio === 0 ||
357
- prev.intersectionRatio === next.intersectionRatio;
358
- if (skipSnap)
359
- return;
360
- const highestIntersecting = prev.intersectionRatio > next.intersectionRatio ? prev : next;
361
- const fullIntersectionIndex = this.highestVisibleIntersection?.index;
362
- if (fullIntersectionIndex === undefined)
363
- return;
364
- const highestIntersectingIndex = highestIntersecting.index;
365
- const origin = highestIntersectingIndex > fullIntersectionIndex ? 'end' : 'start';
366
- scrollToElement({
367
- container: scrollElement,
368
- element: highestIntersecting.element,
369
- direction: this.direction() === 'horizontal' ? 'inline' : 'block',
370
- origin,
371
- scrollBlockMargin: this.direction() === 'horizontal' ? 0 : this.scrollMargin(),
372
- scrollInlineMargin: this.direction() === 'horizontal' ? this.scrollMargin() : 0,
373
- });
374
- isSnapping = true;
375
- window.clearTimeout(snapTimeout);
376
- snapTimeout = window.setTimeout(() => {
377
- isSnapping = false;
378
- }, 1000);
379
- }))
479
+ else if ((this.direction() === 'horizontal' &&
480
+ intersections.biggest.intersection.boundingClientRect.width > scrollableRect.width) ||
481
+ (this.direction() === 'vertical' &&
482
+ intersections.biggest.intersection.boundingClientRect.height > scrollableRect.height)) {
483
+ // If the current element is bigger than the scrollable container we should snap to the start of the current element if the scroll direction is forward
484
+ // and to the end of the current element if the scroll direction is backwards.
485
+ const origin = intersections.biggest.index === intersections.first.index ? 'end' : 'start';
486
+ this.scrollToElement({
487
+ element: intersections.biggest.intersection.target,
488
+ origin,
489
+ });
490
+ }
491
+ else {
492
+ // No special case. Just snap to the biggest intersection.
493
+ this.scrollToElement({
494
+ element: intersections.biggest.intersection.target,
495
+ });
496
+ }
497
+ }), takeUntil(this._disableSnapping$))
380
498
  .subscribe();
381
499
  }
382
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.2", ngImport: i0, type: ScrollableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
383
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.0.2", type: ScrollableComponent, isStandalone: true, selector: "et-scrollable", inputs: { _itemSize: ["itemSize", "_itemSize"], _direction: ["direction", "_direction"], _scrollableRole: ["scrollableRole", "_scrollableRole"], _scrollableClass: ["scrollableClass", "_scrollableClass"], _renderMasks: ["renderMasks", "_renderMasks", booleanAttribute], _renderButtons: ["renderButtons", "_renderButtons", booleanAttribute], _renderScrollbars: ["renderScrollbars", "_renderScrollbars", booleanAttribute], _stickyButtons: ["stickyButtons", "_stickyButtons", booleanAttribute], _cursorDragScroll: ["cursorDragScroll", "_cursorDragScroll", booleanAttribute], _disableActiveElementScrolling: ["disableActiveElementScrolling", "_disableActiveElementScrolling", booleanAttribute], _scrollMode: ["scrollMode", "_scrollMode"], _snap: ["snap", "_snap", booleanAttribute], _scrollMargin: ["scrollMargin", "_scrollMargin", numberAttribute] }, outputs: { scrollStateChange: "scrollStateChange", intersectionChange: "intersectionChange" }, host: { classAttribute: "et-scrollable" }, queries: [{ propertyName: "_activeElementList", predicate: IS_ACTIVE_ELEMENT, descendants: true }, { propertyName: "_elementList", predicate: IS_ELEMENT, descendants: true }], viewQueries: [{ propertyName: "_scrollable", first: true, predicate: ["scrollable"], descendants: true, static: true }, { propertyName: "_firstElement", first: true, predicate: ["firstElement"], descendants: true, static: true }, { propertyName: "_lastElement", first: true, predicate: ["lastElement"], descendants: true, static: true }], ngImport: i0, template: "<div\n #scrollable\n [attr.role]=\"scrollableRole() ?? undefined\"\n [etCursorDragScroll]=\"cursorDragScroll()\"\n [allowedDirection]=\"direction()\"\n (cursorDragStart)=\"setIsCursorDragging(true)\"\n (cursorDragEnd)=\"setIsCursorDragging(false)\"\n class=\"et-scrollable-container-outer\"\n>\n <div [ngClass]=\"scrollableClass()\" class=\"et-scrollable-container\">\n <div #firstElement class=\"et-scroll-observer-first-element\"></div>\n <ng-content />\n <div #lastElement class=\"et-scroll-observer-last-element\"></div>\n </div>\n</div>\n\n@if (renderMasks()) {\n <div class=\"et-scrollable-masks\">\n <div class=\"et-scrollable-mask et-scrollable-mask--start\"></div>\n <div class=\"et-scrollable-mask et-scrollable-mask--end\"></div>\n </div>\n}\n\n@if (renderButtons()) {\n <div class=\"et-scrollable-buttons\">\n <button\n (click)=\"scrollToStartDirection()\"\n class=\"et-scrollable-button et-scrollable-button--start\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n <button\n (click)=\"scrollToStartEnd()\"\n class=\"et-scrollable-button et-scrollable-button--end\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n </div>\n}\n", styles: [".et-scrollable{--mask: #121212 0, transparent 100%;--mask-size: 25px;--_auto-size: min-content;--_flow: column;display:grid;position:relative}.et-scrollable[item-size=same]{--_auto-size: 1fr}.et-scrollable[item-size=full]{--_auto-size: 100%}.et-scrollable[render-scrollbars=false] .et-scrollable-container-outer{scrollbar-width:none}.et-scrollable[render-scrollbars=false] .et-scrollable-container-outer::-webkit-scrollbar{display:none}.et-scrollable[direction=horizontal] .et-scrollable-container-outer{overflow-x:auto;overflow-y:hidden;grid-template-columns:auto 1fr auto}.et-scrollable[direction=horizontal] .et-scrollable-container{grid-auto-columns:var(--_auto-size)}.et-scrollable[direction=horizontal] .et-scrollable-mask,.et-scrollable[direction=horizontal] .et-scrollable-button{inline-size:var(--mask-size);block-size:100%}.et-scrollable[direction=horizontal] .et-scrollable-mask--start,.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:0;inset-inline:0 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--end,.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:0;inset-inline:calc(100% - 40px) 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--start{background:linear-gradient(to right,var(--mask))}.et-scrollable[direction=horizontal] .et-scrollable-mask--end{background:linear-gradient(to left,var(--mask));inset-inline:calc(100% - var(--mask-size)) 100%}.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:calc(50% - 20px);transform:rotate(-90deg)}.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:calc(50% - 20px);transform:rotate(90deg)}.et-scrollable[direction=vertical]{--_flow: row}.et-scrollable[direction=vertical] .et-scrollable-container-outer{overflow-x:hidden;overflow-y:auto;grid-template-rows:auto 1fr auto}.et-scrollable[direction=vertical] .et-scrollable-container{grid-auto-rows:var(--_auto-size)}.et-scrollable[direction=vertical] .et-scrollable-mask,.et-scrollable[direction=vertical] .et-scrollable-button{block-size:var(--mask-size);inline-size:100%}.et-scrollable[direction=vertical] .et-scrollable-mask--start,.et-scrollable[direction=vertical] .et-scrollable-button--start{inset-block-start:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--end,.et-scrollable[direction=vertical] .et-scrollable-button--end{inset-block-end:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--start{background:linear-gradient(to bottom,var(--mask))}.et-scrollable[direction=vertical] .et-scrollable-mask--end{background:linear-gradient(to top,var(--mask))}.et-scrollable[direction=vertical] .et-scrollable-button--start{inset-inline-start:calc(50% - 20px);transform:rotate(0)}.et-scrollable[direction=vertical] .et-scrollable-button--end{inset-inline-start:calc(50% - 20px);transform:rotate(180deg)}.et-scrollable:not(.et-scrollable--is-at-start) .et-scrollable-masks .et-scrollable-mask--start,.et-scrollable:not(.et-scrollable--is-at-start) .et-scrollable-buttons .et-scrollable-button--start{opacity:1}.et-scrollable:not(.et-scrollable--is-at-start) .et-scrollable-buttons .et-scrollable-button--start{pointer-events:all}.et-scrollable:not(.et-scrollable--is-at-end) .et-scrollable-masks .et-scrollable-mask--end,.et-scrollable:not(.et-scrollable--is-at-end) .et-scrollable-buttons .et-scrollable-button--end{opacity:1}.et-scrollable:not(.et-scrollable--is-at-end) .et-scrollable-buttons .et-scrollable-button--end{pointer-events:all}.et-scrollable .et-scrollable-container-outer{overflow:auto;display:grid;grid-row:1/1;grid-column:1/1}.et-scrollable .et-scroll-observer-first-element,.et-scrollable .et-scroll-observer-last-element{position:absolute;block-size:100%;inline-size:1px;pointer-events:none}.et-scrollable .et-scroll-observer-first-element{inset-inline-start:1px}.et-scrollable .et-scroll-observer-last-element{inset-inline-end:1px}.et-scrollable .et-scrollable-container{display:grid;grid-auto-flow:var(--_flow);position:relative}.et-scrollable .et-scrollable-masks,.et-scrollable .et-scrollable-buttons{grid-row:1/1;grid-column:1/1;pointer-events:none}.et-scrollable .et-scrollable-masks .et-scrollable-mask,.et-scrollable .et-scrollable-masks .et-scrollable-button,.et-scrollable .et-scrollable-buttons .et-scrollable-mask,.et-scrollable .et-scrollable-buttons .et-scrollable-button{position:absolute;opacity:0}.et-scrollable.et-scrollable--can-scroll.et-scrollable--enable-overlay-animations .et-scrollable-mask,.et-scrollable.et-scrollable--can-scroll.et-scrollable--enable-overlay-animations .et-scrollable-button{transition:opacity .3s var(--ease-5)}.et-scrollable .et-scrollable-buttons .et-scrollable-button{background:transparent;border:none;padding:12px;inline-size:40px;block-size:40px;border-radius:4px;cursor:pointer}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-chevron-icon{pointer-events:none}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--start{inset-inline:0 0}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--end{inset-inline:calc(100% - 40px) 0}.et-scrollable[sticky-buttons=true]{--_sticky-margin: 10%}.et-scrollable[sticky-buttons=true] .et-scrollable-button{position:sticky}.et-scrollable[sticky-buttons=true][direction=horizontal] .et-scrollable-buttons{margin-block:var(--_sticky-margin)}.et-scrollable[sticky-buttons=true][direction=vertical] .et-scrollable-buttons{margin-inline:var(--_sticky-margin)}.et-scrollable[sticky-buttons=true][direction=vertical] .et-scrollable-buttons .et-scrollable-button--end{inset-block-start:100%}\n"], dependencies: [{ kind: "directive", type: CursorDragScrollDirective, selector: "[etCursorDragScroll]", inputs: ["etCursorDragScroll", "allowedDirection"], outputs: ["cursorDragStart", "cursorDragMove", "cursorDragEnd"], exportAs: ["etCursorDragScroll"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: ChevronIconComponent, selector: "et-chevron-icon" }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
500
+ _disableSnapping() {
501
+ this._disableSnapping$.next();
502
+ }
503
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.3", ngImport: i0, type: ScrollableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
504
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.0.3", type: ScrollableComponent, isStandalone: true, selector: "et-scrollable", inputs: { _itemSize: ["itemSize", "_itemSize"], _direction: ["direction", "_direction"], _scrollableRole: ["scrollableRole", "_scrollableRole"], _scrollableClass: ["scrollableClass", "_scrollableClass"], _renderNavigation: ["renderNavigation", "_renderNavigation", booleanAttribute], _renderMasks: ["renderMasks", "_renderMasks", booleanAttribute], _renderButtons: ["renderButtons", "_renderButtons", booleanAttribute], _renderScrollbars: ["renderScrollbars", "_renderScrollbars", booleanAttribute], _stickyButtons: ["stickyButtons", "_stickyButtons", booleanAttribute], _cursorDragScroll: ["cursorDragScroll", "_cursorDragScroll", booleanAttribute], _disableActiveElementScrolling: ["disableActiveElementScrolling", "_disableActiveElementScrolling", booleanAttribute], _scrollMode: ["scrollMode", "_scrollMode"], _snap: ["snap", "_snap", booleanAttribute], _scrollMargin: ["scrollMargin", "_scrollMargin", numberAttribute] }, outputs: { scrollStateChange: "scrollStateChange", intersectionChange: "intersectionChange" }, host: { classAttribute: "et-scrollable" }, queries: [{ propertyName: "_activeElementList", predicate: IS_ACTIVE_ELEMENT, descendants: true }], viewQueries: [{ propertyName: "_scrollable", first: true, predicate: ["scrollable"], descendants: true, static: true }, { propertyName: "_scrollableContainer", first: true, predicate: ["scrollableContainer"], descendants: true, static: true }, { propertyName: "_firstElement", first: true, predicate: ["firstElement"], descendants: true, static: true }, { propertyName: "_lastElement", first: true, predicate: ["lastElement"], descendants: true, static: true }], ngImport: i0, template: "<div class=\"et-scrollable-wrapper\">\n <div\n #scrollable\n [attr.role]=\"scrollableRole() ?? undefined\"\n [etCursorDragScroll]=\"cursorDragScroll()\"\n [allowedDirection]=\"direction()\"\n (cursorDragStart)=\"setIsCursorDragging(true)\"\n (cursorDragEnd)=\"setIsCursorDragging(false)\"\n class=\"et-scrollable-container-outer\"\n >\n <div #scrollableContainer [ngClass]=\"scrollableClass()\" class=\"et-scrollable-container\">\n <div #firstElement class=\"et-scroll-observer-first-element\" etScrollableIgnoreChild></div>\n <ng-content />\n <div #lastElement class=\"et-scroll-observer-last-element\" etScrollableIgnoreChild></div>\n </div>\n </div>\n\n @if (renderMasks()) {\n <div class=\"et-scrollable-masks\">\n <div class=\"et-scrollable-mask et-scrollable-mask--start\"></div>\n <div class=\"et-scrollable-mask et-scrollable-mask--end\"></div>\n </div>\n }\n\n @if (renderButtons()) {\n <div class=\"et-scrollable-buttons\">\n <button\n (click)=\"scrollToStartDirection()\"\n class=\"et-scrollable-button et-scrollable-button--start\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n <button\n (click)=\"scrollToEndDirection()\"\n class=\"et-scrollable-button et-scrollable-button--end\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n </div>\n }\n</div>\n\n@if (renderNavigation()) {\n <div class=\"et-scrollable-navigation\" aria-hidden=\"true\">\n @for (item of scrollableNavigation(); track i; let i = $index) {\n <button\n [class.et-scrollable-navigation-item--active]=\"item.isActive\"\n (click)=\"scrollToElementViaNavigation(i, item.element)\"\n class=\"et-scrollable-navigation-item\"\n ></button>\n }\n </div>\n}\n", styles: [".et-scrollable{--mask: #121212 0, transparent 100%;--mask-size: 25px;--_auto-size: min-content;--_flow: column;display:block}.et-scrollable .et-scrollable-wrapper{display:grid;position:relative}.et-scrollable[item-size=same]{--_auto-size: 1fr}.et-scrollable[item-size=full]{--_auto-size: 100%}.et-scrollable[render-scrollbars=false] .et-scrollable-container-outer{scrollbar-width:none}.et-scrollable[render-scrollbars=false] .et-scrollable-container-outer::-webkit-scrollbar{display:none}.et-scrollable[direction=horizontal] .et-scrollable-container-outer{overflow-x:auto;overflow-y:hidden}.et-scrollable[direction=horizontal] .et-scrollable-container{grid-auto-columns:var(--_auto-size)}.et-scrollable[direction=horizontal] .et-scrollable-mask,.et-scrollable[direction=horizontal] .et-scrollable-button{inline-size:var(--mask-size);block-size:100%}.et-scrollable[direction=horizontal] .et-scrollable-mask--start,.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:0;inset-inline:0 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--end,.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:0;inset-inline:calc(100% - 40px) 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--start{background:linear-gradient(to right,var(--mask))}.et-scrollable[direction=horizontal] .et-scrollable-mask--end{background:linear-gradient(to left,var(--mask));inset-inline:calc(100% - var(--mask-size)) 100%}.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:calc(50% - 20px);transform:rotate(-90deg)}.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:calc(50% - 20px);transform:rotate(90deg)}.et-scrollable[direction=vertical]{--_flow: row}.et-scrollable[direction=vertical] .et-scrollable-container-outer{overflow-x:hidden;overflow-y:auto}.et-scrollable[direction=vertical] .et-scrollable-container{grid-auto-rows:var(--_auto-size)}.et-scrollable[direction=vertical] .et-scrollable-mask,.et-scrollable[direction=vertical] .et-scrollable-button{block-size:var(--mask-size);inline-size:100%}.et-scrollable[direction=vertical] .et-scrollable-mask--start,.et-scrollable[direction=vertical] .et-scrollable-button--start{inset-block-start:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--end,.et-scrollable[direction=vertical] .et-scrollable-button--end{inset-block-end:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--start{background:linear-gradient(to bottom,var(--mask))}.et-scrollable[direction=vertical] .et-scrollable-mask--end{background:linear-gradient(to top,var(--mask))}.et-scrollable[direction=vertical] .et-scrollable-button--start{inset-inline-start:calc(50% - 20px);transform:rotate(0)}.et-scrollable[direction=vertical] .et-scrollable-button--end{inset-inline-start:calc(50% - 20px);transform:rotate(180deg)}.et-scrollable:not(.et-scrollable--is-at-start) .et-scrollable-masks .et-scrollable-mask--start,.et-scrollable:not(.et-scrollable--is-at-start) .et-scrollable-buttons .et-scrollable-button--start{opacity:1}.et-scrollable:not(.et-scrollable--is-at-start) .et-scrollable-buttons .et-scrollable-button--start{pointer-events:all}.et-scrollable:not(.et-scrollable--is-at-end) .et-scrollable-masks .et-scrollable-mask--end,.et-scrollable:not(.et-scrollable--is-at-end) .et-scrollable-buttons .et-scrollable-button--end{opacity:1}.et-scrollable:not(.et-scrollable--is-at-end) .et-scrollable-buttons .et-scrollable-button--end{pointer-events:all}.et-scrollable .et-scrollable-container-outer{overflow:auto;display:grid;position:relative;grid-row:1/1;grid-column:1/1}.et-scrollable .et-scroll-observer-first-element,.et-scrollable .et-scroll-observer-last-element{position:absolute;block-size:100%;inline-size:1px;pointer-events:none}.et-scrollable .et-scroll-observer-first-element{inset-inline-start:1px}.et-scrollable .et-scroll-observer-last-element{inset-inline-end:1px}.et-scrollable .et-scrollable-container{display:grid;grid-auto-flow:var(--_flow);position:relative}.et-scrollable .et-scrollable-masks,.et-scrollable .et-scrollable-buttons{grid-row:1/1;grid-column:1/1;pointer-events:none}.et-scrollable .et-scrollable-masks .et-scrollable-mask,.et-scrollable .et-scrollable-masks .et-scrollable-button,.et-scrollable .et-scrollable-buttons .et-scrollable-mask,.et-scrollable .et-scrollable-buttons .et-scrollable-button{position:absolute;opacity:0}.et-scrollable.et-scrollable--can-scroll.et-scrollable--enable-overlay-animations .et-scrollable-mask,.et-scrollable.et-scrollable--can-scroll.et-scrollable--enable-overlay-animations .et-scrollable-button{transition:opacity .3s var(--ease-5)}.et-scrollable .et-scrollable-buttons .et-scrollable-button{background:transparent;border:none;padding:12px;inline-size:40px;block-size:40px;border-radius:4px;cursor:pointer}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-chevron-icon{pointer-events:none}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--start{inset-inline:0 0}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--end{inset-inline:calc(100% - 40px) 0}.et-scrollable[sticky-buttons=true]{--_sticky-margin: 10%}.et-scrollable[sticky-buttons=true] .et-scrollable-button{position:sticky}.et-scrollable[sticky-buttons=true][direction=horizontal] .et-scrollable-buttons{margin-block:var(--_sticky-margin)}.et-scrollable[sticky-buttons=true][direction=vertical] .et-scrollable-buttons{margin-inline:var(--_sticky-margin)}.et-scrollable[sticky-buttons=true][direction=vertical] .et-scrollable-buttons .et-scrollable-button--end{inset-block-start:100%}.et-scrollable .et-scrollable-navigation{display:flex;justify-content:center;gap:10px;margin-block-start:10px}.et-scrollable .et-scrollable-navigation .et-scrollable-navigation-item{all:unset;inline-size:10px;block-size:10px;display:block;background-color:#4b4b4b;border-radius:50%;cursor:pointer}.et-scrollable .et-scrollable-navigation .et-scrollable-navigation-item.et-scrollable-navigation-item--active{background-color:#c6c6c6}\n"], dependencies: [{ kind: "directive", type: CursorDragScrollDirective, selector: "[etCursorDragScroll]", inputs: ["etCursorDragScroll", "allowedDirection"], outputs: ["cursorDragStart", "cursorDragMove", "cursorDragEnd"], exportAs: ["etCursorDragScroll"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: ChevronIconComponent, selector: "et-chevron-icon" }, { kind: "directive", type: ScrollableIgnoreChildDirective, selector: "[etScrollableIgnoreChild]", inputs: ["etScrollableIgnoreChild"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
384
505
  }
385
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.2", ngImport: i0, type: ScrollableComponent, decorators: [{
506
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.3", ngImport: i0, type: ScrollableComponent, decorators: [{
386
507
  type: Component,
387
- args: [{ selector: 'et-scrollable', standalone: true, encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, imports: [CursorDragScrollDirective, ObserveScrollStateDirective, NgClass, LetDirective, ChevronIconComponent], host: {
508
+ args: [{ selector: 'et-scrollable', standalone: true, encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, imports: [
509
+ CursorDragScrollDirective,
510
+ ObserveScrollStateDirective,
511
+ NgClass,
512
+ LetDirective,
513
+ ChevronIconComponent,
514
+ ScrollableIgnoreChildDirective,
515
+ ], host: {
388
516
  class: 'et-scrollable',
389
- }, template: "<div\n #scrollable\n [attr.role]=\"scrollableRole() ?? undefined\"\n [etCursorDragScroll]=\"cursorDragScroll()\"\n [allowedDirection]=\"direction()\"\n (cursorDragStart)=\"setIsCursorDragging(true)\"\n (cursorDragEnd)=\"setIsCursorDragging(false)\"\n class=\"et-scrollable-container-outer\"\n>\n <div [ngClass]=\"scrollableClass()\" class=\"et-scrollable-container\">\n <div #firstElement class=\"et-scroll-observer-first-element\"></div>\n <ng-content />\n <div #lastElement class=\"et-scroll-observer-last-element\"></div>\n </div>\n</div>\n\n@if (renderMasks()) {\n <div class=\"et-scrollable-masks\">\n <div class=\"et-scrollable-mask et-scrollable-mask--start\"></div>\n <div class=\"et-scrollable-mask et-scrollable-mask--end\"></div>\n </div>\n}\n\n@if (renderButtons()) {\n <div class=\"et-scrollable-buttons\">\n <button\n (click)=\"scrollToStartDirection()\"\n class=\"et-scrollable-button et-scrollable-button--start\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n <button\n (click)=\"scrollToStartEnd()\"\n class=\"et-scrollable-button et-scrollable-button--end\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n </div>\n}\n", styles: [".et-scrollable{--mask: #121212 0, transparent 100%;--mask-size: 25px;--_auto-size: min-content;--_flow: column;display:grid;position:relative}.et-scrollable[item-size=same]{--_auto-size: 1fr}.et-scrollable[item-size=full]{--_auto-size: 100%}.et-scrollable[render-scrollbars=false] .et-scrollable-container-outer{scrollbar-width:none}.et-scrollable[render-scrollbars=false] .et-scrollable-container-outer::-webkit-scrollbar{display:none}.et-scrollable[direction=horizontal] .et-scrollable-container-outer{overflow-x:auto;overflow-y:hidden;grid-template-columns:auto 1fr auto}.et-scrollable[direction=horizontal] .et-scrollable-container{grid-auto-columns:var(--_auto-size)}.et-scrollable[direction=horizontal] .et-scrollable-mask,.et-scrollable[direction=horizontal] .et-scrollable-button{inline-size:var(--mask-size);block-size:100%}.et-scrollable[direction=horizontal] .et-scrollable-mask--start,.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:0;inset-inline:0 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--end,.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:0;inset-inline:calc(100% - 40px) 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--start{background:linear-gradient(to right,var(--mask))}.et-scrollable[direction=horizontal] .et-scrollable-mask--end{background:linear-gradient(to left,var(--mask));inset-inline:calc(100% - var(--mask-size)) 100%}.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:calc(50% - 20px);transform:rotate(-90deg)}.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:calc(50% - 20px);transform:rotate(90deg)}.et-scrollable[direction=vertical]{--_flow: row}.et-scrollable[direction=vertical] .et-scrollable-container-outer{overflow-x:hidden;overflow-y:auto;grid-template-rows:auto 1fr auto}.et-scrollable[direction=vertical] .et-scrollable-container{grid-auto-rows:var(--_auto-size)}.et-scrollable[direction=vertical] .et-scrollable-mask,.et-scrollable[direction=vertical] .et-scrollable-button{block-size:var(--mask-size);inline-size:100%}.et-scrollable[direction=vertical] .et-scrollable-mask--start,.et-scrollable[direction=vertical] .et-scrollable-button--start{inset-block-start:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--end,.et-scrollable[direction=vertical] .et-scrollable-button--end{inset-block-end:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--start{background:linear-gradient(to bottom,var(--mask))}.et-scrollable[direction=vertical] .et-scrollable-mask--end{background:linear-gradient(to top,var(--mask))}.et-scrollable[direction=vertical] .et-scrollable-button--start{inset-inline-start:calc(50% - 20px);transform:rotate(0)}.et-scrollable[direction=vertical] .et-scrollable-button--end{inset-inline-start:calc(50% - 20px);transform:rotate(180deg)}.et-scrollable:not(.et-scrollable--is-at-start) .et-scrollable-masks .et-scrollable-mask--start,.et-scrollable:not(.et-scrollable--is-at-start) .et-scrollable-buttons .et-scrollable-button--start{opacity:1}.et-scrollable:not(.et-scrollable--is-at-start) .et-scrollable-buttons .et-scrollable-button--start{pointer-events:all}.et-scrollable:not(.et-scrollable--is-at-end) .et-scrollable-masks .et-scrollable-mask--end,.et-scrollable:not(.et-scrollable--is-at-end) .et-scrollable-buttons .et-scrollable-button--end{opacity:1}.et-scrollable:not(.et-scrollable--is-at-end) .et-scrollable-buttons .et-scrollable-button--end{pointer-events:all}.et-scrollable .et-scrollable-container-outer{overflow:auto;display:grid;grid-row:1/1;grid-column:1/1}.et-scrollable .et-scroll-observer-first-element,.et-scrollable .et-scroll-observer-last-element{position:absolute;block-size:100%;inline-size:1px;pointer-events:none}.et-scrollable .et-scroll-observer-first-element{inset-inline-start:1px}.et-scrollable .et-scroll-observer-last-element{inset-inline-end:1px}.et-scrollable .et-scrollable-container{display:grid;grid-auto-flow:var(--_flow);position:relative}.et-scrollable .et-scrollable-masks,.et-scrollable .et-scrollable-buttons{grid-row:1/1;grid-column:1/1;pointer-events:none}.et-scrollable .et-scrollable-masks .et-scrollable-mask,.et-scrollable .et-scrollable-masks .et-scrollable-button,.et-scrollable .et-scrollable-buttons .et-scrollable-mask,.et-scrollable .et-scrollable-buttons .et-scrollable-button{position:absolute;opacity:0}.et-scrollable.et-scrollable--can-scroll.et-scrollable--enable-overlay-animations .et-scrollable-mask,.et-scrollable.et-scrollable--can-scroll.et-scrollable--enable-overlay-animations .et-scrollable-button{transition:opacity .3s var(--ease-5)}.et-scrollable .et-scrollable-buttons .et-scrollable-button{background:transparent;border:none;padding:12px;inline-size:40px;block-size:40px;border-radius:4px;cursor:pointer}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-chevron-icon{pointer-events:none}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--start{inset-inline:0 0}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--end{inset-inline:calc(100% - 40px) 0}.et-scrollable[sticky-buttons=true]{--_sticky-margin: 10%}.et-scrollable[sticky-buttons=true] .et-scrollable-button{position:sticky}.et-scrollable[sticky-buttons=true][direction=horizontal] .et-scrollable-buttons{margin-block:var(--_sticky-margin)}.et-scrollable[sticky-buttons=true][direction=vertical] .et-scrollable-buttons{margin-inline:var(--_sticky-margin)}.et-scrollable[sticky-buttons=true][direction=vertical] .et-scrollable-buttons .et-scrollable-button--end{inset-block-start:100%}\n"] }]
517
+ }, template: "<div class=\"et-scrollable-wrapper\">\n <div\n #scrollable\n [attr.role]=\"scrollableRole() ?? undefined\"\n [etCursorDragScroll]=\"cursorDragScroll()\"\n [allowedDirection]=\"direction()\"\n (cursorDragStart)=\"setIsCursorDragging(true)\"\n (cursorDragEnd)=\"setIsCursorDragging(false)\"\n class=\"et-scrollable-container-outer\"\n >\n <div #scrollableContainer [ngClass]=\"scrollableClass()\" class=\"et-scrollable-container\">\n <div #firstElement class=\"et-scroll-observer-first-element\" etScrollableIgnoreChild></div>\n <ng-content />\n <div #lastElement class=\"et-scroll-observer-last-element\" etScrollableIgnoreChild></div>\n </div>\n </div>\n\n @if (renderMasks()) {\n <div class=\"et-scrollable-masks\">\n <div class=\"et-scrollable-mask et-scrollable-mask--start\"></div>\n <div class=\"et-scrollable-mask et-scrollable-mask--end\"></div>\n </div>\n }\n\n @if (renderButtons()) {\n <div class=\"et-scrollable-buttons\">\n <button\n (click)=\"scrollToStartDirection()\"\n class=\"et-scrollable-button et-scrollable-button--start\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n <button\n (click)=\"scrollToEndDirection()\"\n class=\"et-scrollable-button et-scrollable-button--end\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n </div>\n }\n</div>\n\n@if (renderNavigation()) {\n <div class=\"et-scrollable-navigation\" aria-hidden=\"true\">\n @for (item of scrollableNavigation(); track i; let i = $index) {\n <button\n [class.et-scrollable-navigation-item--active]=\"item.isActive\"\n (click)=\"scrollToElementViaNavigation(i, item.element)\"\n class=\"et-scrollable-navigation-item\"\n ></button>\n }\n </div>\n}\n", styles: [".et-scrollable{--mask: #121212 0, transparent 100%;--mask-size: 25px;--_auto-size: min-content;--_flow: column;display:block}.et-scrollable .et-scrollable-wrapper{display:grid;position:relative}.et-scrollable[item-size=same]{--_auto-size: 1fr}.et-scrollable[item-size=full]{--_auto-size: 100%}.et-scrollable[render-scrollbars=false] .et-scrollable-container-outer{scrollbar-width:none}.et-scrollable[render-scrollbars=false] .et-scrollable-container-outer::-webkit-scrollbar{display:none}.et-scrollable[direction=horizontal] .et-scrollable-container-outer{overflow-x:auto;overflow-y:hidden}.et-scrollable[direction=horizontal] .et-scrollable-container{grid-auto-columns:var(--_auto-size)}.et-scrollable[direction=horizontal] .et-scrollable-mask,.et-scrollable[direction=horizontal] .et-scrollable-button{inline-size:var(--mask-size);block-size:100%}.et-scrollable[direction=horizontal] .et-scrollable-mask--start,.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:0;inset-inline:0 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--end,.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:0;inset-inline:calc(100% - 40px) 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--start{background:linear-gradient(to right,var(--mask))}.et-scrollable[direction=horizontal] .et-scrollable-mask--end{background:linear-gradient(to left,var(--mask));inset-inline:calc(100% - var(--mask-size)) 100%}.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:calc(50% - 20px);transform:rotate(-90deg)}.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:calc(50% - 20px);transform:rotate(90deg)}.et-scrollable[direction=vertical]{--_flow: row}.et-scrollable[direction=vertical] .et-scrollable-container-outer{overflow-x:hidden;overflow-y:auto}.et-scrollable[direction=vertical] .et-scrollable-container{grid-auto-rows:var(--_auto-size)}.et-scrollable[direction=vertical] .et-scrollable-mask,.et-scrollable[direction=vertical] .et-scrollable-button{block-size:var(--mask-size);inline-size:100%}.et-scrollable[direction=vertical] .et-scrollable-mask--start,.et-scrollable[direction=vertical] .et-scrollable-button--start{inset-block-start:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--end,.et-scrollable[direction=vertical] .et-scrollable-button--end{inset-block-end:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--start{background:linear-gradient(to bottom,var(--mask))}.et-scrollable[direction=vertical] .et-scrollable-mask--end{background:linear-gradient(to top,var(--mask))}.et-scrollable[direction=vertical] .et-scrollable-button--start{inset-inline-start:calc(50% - 20px);transform:rotate(0)}.et-scrollable[direction=vertical] .et-scrollable-button--end{inset-inline-start:calc(50% - 20px);transform:rotate(180deg)}.et-scrollable:not(.et-scrollable--is-at-start) .et-scrollable-masks .et-scrollable-mask--start,.et-scrollable:not(.et-scrollable--is-at-start) .et-scrollable-buttons .et-scrollable-button--start{opacity:1}.et-scrollable:not(.et-scrollable--is-at-start) .et-scrollable-buttons .et-scrollable-button--start{pointer-events:all}.et-scrollable:not(.et-scrollable--is-at-end) .et-scrollable-masks .et-scrollable-mask--end,.et-scrollable:not(.et-scrollable--is-at-end) .et-scrollable-buttons .et-scrollable-button--end{opacity:1}.et-scrollable:not(.et-scrollable--is-at-end) .et-scrollable-buttons .et-scrollable-button--end{pointer-events:all}.et-scrollable .et-scrollable-container-outer{overflow:auto;display:grid;position:relative;grid-row:1/1;grid-column:1/1}.et-scrollable .et-scroll-observer-first-element,.et-scrollable .et-scroll-observer-last-element{position:absolute;block-size:100%;inline-size:1px;pointer-events:none}.et-scrollable .et-scroll-observer-first-element{inset-inline-start:1px}.et-scrollable .et-scroll-observer-last-element{inset-inline-end:1px}.et-scrollable .et-scrollable-container{display:grid;grid-auto-flow:var(--_flow);position:relative}.et-scrollable .et-scrollable-masks,.et-scrollable .et-scrollable-buttons{grid-row:1/1;grid-column:1/1;pointer-events:none}.et-scrollable .et-scrollable-masks .et-scrollable-mask,.et-scrollable .et-scrollable-masks .et-scrollable-button,.et-scrollable .et-scrollable-buttons .et-scrollable-mask,.et-scrollable .et-scrollable-buttons .et-scrollable-button{position:absolute;opacity:0}.et-scrollable.et-scrollable--can-scroll.et-scrollable--enable-overlay-animations .et-scrollable-mask,.et-scrollable.et-scrollable--can-scroll.et-scrollable--enable-overlay-animations .et-scrollable-button{transition:opacity .3s var(--ease-5)}.et-scrollable .et-scrollable-buttons .et-scrollable-button{background:transparent;border:none;padding:12px;inline-size:40px;block-size:40px;border-radius:4px;cursor:pointer}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-chevron-icon{pointer-events:none}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--start{inset-inline:0 0}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--end{inset-inline:calc(100% - 40px) 0}.et-scrollable[sticky-buttons=true]{--_sticky-margin: 10%}.et-scrollable[sticky-buttons=true] .et-scrollable-button{position:sticky}.et-scrollable[sticky-buttons=true][direction=horizontal] .et-scrollable-buttons{margin-block:var(--_sticky-margin)}.et-scrollable[sticky-buttons=true][direction=vertical] .et-scrollable-buttons{margin-inline:var(--_sticky-margin)}.et-scrollable[sticky-buttons=true][direction=vertical] .et-scrollable-buttons .et-scrollable-button--end{inset-block-start:100%}.et-scrollable .et-scrollable-navigation{display:flex;justify-content:center;gap:10px;margin-block-start:10px}.et-scrollable .et-scrollable-navigation .et-scrollable-navigation-item{all:unset;inline-size:10px;block-size:10px;display:block;background-color:#4b4b4b;border-radius:50%;cursor:pointer}.et-scrollable .et-scrollable-navigation .et-scrollable-navigation-item.et-scrollable-navigation-item--active{background-color:#c6c6c6}\n"] }]
390
518
  }], ctorParameters: () => [], propDecorators: { _itemSize: [{
391
519
  type: Input,
392
520
  args: [{ alias: 'itemSize' }]
@@ -399,6 +527,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.2", ngImpor
399
527
  }], _scrollableClass: [{
400
528
  type: Input,
401
529
  args: [{ alias: 'scrollableClass' }]
530
+ }], _renderNavigation: [{
531
+ type: Input,
532
+ args: [{ transform: booleanAttribute, alias: 'renderNavigation' }]
402
533
  }], _renderMasks: [{
403
534
  type: Input,
404
535
  args: [{ transform: booleanAttribute, alias: 'renderMasks' }]
@@ -433,6 +564,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.2", ngImpor
433
564
  }], _scrollable: [{
434
565
  type: ViewChild,
435
566
  args: ['scrollable', { static: true }]
567
+ }], _scrollableContainer: [{
568
+ type: ViewChild,
569
+ args: ['scrollableContainer', { static: true }]
436
570
  }], _firstElement: [{
437
571
  type: ViewChild,
438
572
  args: ['firstElement', { static: true }]
@@ -442,8 +576,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.2", ngImpor
442
576
  }], _activeElementList: [{
443
577
  type: ContentChildren,
444
578
  args: [IS_ACTIVE_ELEMENT, { descendants: true }]
445
- }], _elementList: [{
446
- type: ContentChildren,
447
- args: [IS_ELEMENT, { descendants: true }]
448
579
  }] } });
449
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable.component.js","sourceRoot":"","sources":["../../../../../../../../../libs/cdk/src/lib/components/scrollable/components/scrollable/scrollable.component.ts","../../../../../../../../../libs/cdk/src/lib/components/scrollable/components/scrollable/scrollable.component.html"],"names":[],"mappings":"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","sourcesContent":["import { NgClass } from '@angular/common';\nimport {\n  AfterContentInit,\n  ChangeDetectionStrategy,\n  Component,\n  ContentChildren,\n  ElementRef,\n  EventEmitter,\n  Input,\n  Output,\n  ViewChild,\n  ViewEncapsulation,\n  booleanAttribute,\n  computed,\n  effect,\n  inject,\n  isDevMode,\n  numberAttribute,\n  signal,\n} from '@angular/core';\nimport {\n  CurrentElementVisibility,\n  CursorDragScrollDirective,\n  IS_ACTIVE_ELEMENT,\n  IS_ELEMENT,\n  IsActiveElementDirective,\n  IsElementDirective,\n  LetDirective,\n  NgClassType,\n  ObserveScrollStateDirective,\n  ScrollObserverScrollState,\n  ScrollToElementOptions,\n  TypedQueryList,\n  createDestroy,\n  getElementVisibleStates,\n  isElementVisible,\n  nextFrame,\n  scrollToElement,\n  signalElementIntersection,\n  signalElementScrollState,\n  signalHostAttributes,\n  signalHostClasses,\n} from '@ethlete/core';\nimport { BehaviorSubject, debounceTime, fromEvent, merge, of, startWith, takeUntil, tap } from 'rxjs';\nimport { ChevronIconComponent } from '../../../icons';\nimport { ScrollableIntersectionChange, ScrollableScrollMode } from '../../types';\n\n@Component({\n  selector: 'et-scrollable',\n  templateUrl: './scrollable.component.html',\n  styleUrls: ['./scrollable.component.scss'],\n  standalone: true,\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [CursorDragScrollDirective, ObserveScrollStateDirective, NgClass, LetDirective, ChevronIconComponent],\n  host: {\n    class: 'et-scrollable',\n  },\n})\nexport class ScrollableComponent implements AfterContentInit {\n  private readonly _destroy$ = createDestroy();\n  private readonly _elementRef = inject<ElementRef<HTMLElement>>(ElementRef);\n\n  private readonly _isCursorDragging$ = new BehaviorSubject<boolean>(false);\n  private readonly _latestVisibilityStates$ = new BehaviorSubject<ScrollableIntersectionChange[]>([]);\n\n  @Input({ alias: 'itemSize' })\n  set _itemSize(v: 'auto' | 'same' | 'full') {\n    this.itemSize.set(v);\n  }\n  readonly itemSize = signal<'auto' | 'same' | 'full'>('auto');\n\n  @Input({ alias: 'direction' })\n  set _direction(v: 'horizontal' | 'vertical') {\n    this.direction.set(v);\n  }\n  readonly direction = signal<'horizontal' | 'vertical'>('horizontal');\n\n  @Input({ alias: 'scrollableRole' })\n  set _scrollableRole(v: string | null) {\n    this.scrollableRole.set(v);\n  }\n  readonly scrollableRole = signal<string | null>(null);\n\n  @Input({ alias: 'scrollableClass' })\n  set _scrollableClass(v: NgClassType | null) {\n    this.scrollableClass.set(v);\n  }\n  readonly scrollableClass = signal<NgClassType | null>(null);\n\n  @Input({ transform: booleanAttribute, alias: 'renderMasks' })\n  set _renderMasks(v: boolean) {\n    this.renderMasks.set(v);\n  }\n  readonly renderMasks = signal(true);\n\n  @Input({ transform: booleanAttribute, alias: 'renderButtons' })\n  set _renderButtons(v: boolean) {\n    this.renderButtons.set(v);\n  }\n  readonly renderButtons = signal(true);\n\n  @Input({ transform: booleanAttribute, alias: 'renderScrollbars' })\n  set _renderScrollbars(v: boolean) {\n    this.renderScrollbars.set(v);\n  }\n  readonly renderScrollbars = signal(false);\n\n  @Input({ transform: booleanAttribute, alias: 'stickyButtons' })\n  set _stickyButtons(v: boolean) {\n    this.stickyButtons.set(v);\n  }\n  readonly stickyButtons = signal(false);\n\n  @Input({ transform: booleanAttribute, alias: 'cursorDragScroll' })\n  set _cursorDragScroll(v: boolean) {\n    this.cursorDragScroll.set(v);\n  }\n  readonly cursorDragScroll = signal(true);\n\n  @Input({ transform: booleanAttribute, alias: 'disableActiveElementScrolling' })\n  set _disableActiveElementScrolling(v: boolean) {\n    this.disableActiveElementScrolling.set(v);\n  }\n  readonly disableActiveElementScrolling = signal(false);\n\n  @Input({ alias: 'scrollMode' })\n  set _scrollMode(v: ScrollableScrollMode) {\n    this.scrollMode.set(v);\n  }\n  readonly scrollMode = signal<ScrollableScrollMode>('container');\n\n  @Input({ transform: booleanAttribute, alias: 'snap' })\n  set _snap(v: boolean) {\n    this.snap.set(v);\n  }\n  readonly snap = signal(false);\n\n  @Input({ transform: numberAttribute, alias: 'scrollMargin' })\n  set _scrollMargin(v: number) {\n    this.scrollMargin.set(v);\n  }\n  readonly scrollMargin = signal(0);\n\n  @Output()\n  readonly scrollStateChange = new EventEmitter<ScrollObserverScrollState>();\n\n  @Output()\n  readonly intersectionChange = new EventEmitter<ScrollableIntersectionChange[]>();\n\n  @ViewChild('scrollable', { static: true })\n  private set _scrollable(e: ElementRef<HTMLElement>) {\n    this.scrollable.set(e);\n  }\n  readonly scrollable = signal<ElementRef<HTMLElement> | null>(null);\n\n  @ViewChild('firstElement', { static: true })\n  private set _firstElement(e: ElementRef<HTMLElement>) {\n    this.firstElement.set(e);\n  }\n  readonly firstElement = signal<ElementRef<HTMLElement> | null>(null);\n\n  @ViewChild('lastElement', { static: true })\n  private set _lastElement(e: ElementRef<HTMLElement>) {\n    this.lastElement.set(e);\n  }\n  readonly lastElement = signal<ElementRef<HTMLElement> | null>(null);\n\n  @ContentChildren(IS_ACTIVE_ELEMENT, { descendants: true })\n  private set _activeElementList(e: TypedQueryList<IsActiveElementDirective>) {\n    this.activeElementList.set(e);\n  }\n  readonly activeElementList = signal<TypedQueryList<IsActiveElementDirective> | null>(null);\n\n  @ContentChildren(IS_ELEMENT, { descendants: true })\n  private set _elementList(e: TypedQueryList<IsElementDirective>) {\n    this.elementList.set(e);\n  }\n  readonly elementList = signal<TypedQueryList<IsElementDirective> | null>(null);\n\n  get highestVisibleIntersection() {\n    const elements = this._latestVisibilityStates$.value;\n\n    if (!elements.length) {\n      return null;\n    }\n\n    return elements.reduce((prev, curr) => {\n      if (!prev) {\n        return curr;\n      }\n\n      return curr.intersectionRatio > prev.intersectionRatio ? curr : prev;\n    });\n  }\n\n  get nextPartialIntersection() {\n    const elements = this._latestVisibilityStates$.value;\n\n    if (!elements.length) {\n      return null;\n    }\n\n    const highestVisibleIntersection = this.highestVisibleIntersection;\n\n    if (!highestVisibleIntersection) {\n      return null;\n    }\n\n    const nextIndex = elements.slice(highestVisibleIntersection.index).findIndex((e) => !e.isIntersecting);\n\n    if (nextIndex === -1) {\n      return null;\n    }\n\n    const nextElement = elements[highestVisibleIntersection.index + nextIndex];\n\n    return nextElement || null;\n  }\n\n  get previousPartialIntersection() {\n    const elements = this._latestVisibilityStates$.value;\n\n    if (!elements.length) {\n      return null;\n    }\n\n    const highestVisibleIntersection = this.highestVisibleIntersection;\n\n    if (!highestVisibleIntersection) {\n      return null;\n    }\n\n    const previousIndex = elements\n      .slice(0, highestVisibleIntersection.index)\n      .reverse()\n      .findIndex((e) => !e.isIntersecting);\n\n    if (previousIndex === -1) {\n      return null;\n    }\n\n    const previousElement = elements[highestVisibleIntersection.index - previousIndex - 1];\n\n    return previousElement || null;\n  }\n\n  protected readonly containerScrollState = signalElementScrollState(this.scrollable);\n  protected readonly firstElementIntersection = signalElementIntersection(this.firstElement);\n  protected readonly firstElementVisibility = signal<CurrentElementVisibility | null>(null);\n  protected readonly lastElementIntersection = signalElementIntersection(this.lastElement);\n  protected readonly lastElementVisibility = signal<CurrentElementVisibility | null>(null);\n\n  protected readonly canScroll = computed(() => {\n    const dir = this.direction();\n\n    if (dir === 'horizontal') {\n      return this.containerScrollState().canScrollHorizontally;\n    }\n\n    return this.containerScrollState().canScrollVertically;\n  });\n\n  protected readonly isAtStart = computed(() => {\n    if (!this.canScroll()) {\n      return true;\n    }\n\n    const intersection = this.firstElementIntersection();\n\n    if (!intersection) {\n      return this.firstElementVisibility()?.inline ?? true;\n    }\n\n    return intersection.isIntersecting;\n  });\n  protected readonly isAtEnd = computed(() => {\n    if (!this.canScroll()) {\n      return true;\n    }\n\n    const intersection = this.lastElementIntersection();\n\n    if (!intersection) {\n      return this.lastElementVisibility()?.inline ?? true;\n    }\n\n    return intersection.isIntersecting;\n  });\n\n  protected readonly enableOverlayAnimations = signal(false);\n\n  protected readonly hostAttributes = signalHostAttributes({\n    'item-size': this.itemSize,\n    direction: this.direction,\n    'render-scrollbars': this.renderScrollbars,\n    'sticky-buttons': this.stickyButtons,\n  });\n\n  protected readonly hostClasses = signalHostClasses({\n    'et-scrollable--can-scroll': this.canScroll,\n    'et-scrollable--is-at-start': this.isAtStart,\n    'et-scrollable--is-at-end': this.isAtEnd,\n    'et-scrollable--enable-overlay-animations': this.enableOverlayAnimations,\n  });\n\n  constructor() {\n    effect(\n      () => {\n        const scrollable = this.scrollable()?.nativeElement;\n        const firstElement = this.firstElement()?.nativeElement;\n        const lastElement = this.lastElement()?.nativeElement;\n        const activeElementList = this.activeElementList()?.toArray();\n\n        if (!scrollable || !firstElement || !lastElement || !activeElementList) {\n          return;\n        }\n\n        const firstActive = activeElementList.find((a) => a.isActiveElement);\n\n        if (firstActive && !this.disableActiveElementScrolling()) {\n          const offsetTop = firstActive.elementRef.nativeElement.offsetTop - scrollable.offsetTop;\n          const offsetLeft = firstActive.elementRef.nativeElement.offsetLeft - scrollable.offsetLeft;\n          scrollable.scrollLeft = offsetLeft;\n          scrollable.scrollTop = offsetTop;\n        }\n\n        this.firstElementVisibility.set(\n          isElementVisible({\n            container: scrollable,\n            element: firstElement,\n          }),\n        );\n\n        this.lastElementVisibility.set(\n          isElementVisible({\n            container: scrollable,\n            element: lastElement,\n          }),\n        );\n\n        // We need to wait one frame before enabling animations to prevent a animation from playing during initial render.\n        nextFrame(() => this.enableOverlayAnimations.set(true));\n      },\n      { allowSignalWrites: true },\n    );\n\n    effect(() => {\n      const isAtStart = this.isAtStart();\n      const isAtEnd = this.isAtEnd();\n      const canScroll = this.canScroll();\n\n      this.scrollStateChange.emit({\n        canScroll,\n        isAtEnd: !!isAtEnd,\n        isAtStart: !!isAtStart,\n      });\n    });\n  }\n\n  ngAfterContentInit(): void {\n    this._setupScrollListening();\n  }\n\n  scrollOneContainerSize(direction: 'start' | 'end') {\n    const scrollElement = this.scrollable()?.nativeElement;\n\n    if (!scrollElement) {\n      return;\n    }\n\n    const parent = this._elementRef.nativeElement;\n\n    const scrollableSize = this.direction() === 'horizontal' ? parent.clientWidth : parent.clientHeight;\n    const currentScroll = this.direction() === 'horizontal' ? scrollElement.scrollLeft : scrollElement.scrollTop;\n\n    scrollElement.scrollTo({\n      [this.direction() === 'horizontal' ? 'left' : 'top']:\n        currentScroll + (direction === 'start' ? -scrollableSize : scrollableSize),\n      behavior: 'smooth',\n    });\n  }\n\n  scrollOneItemSize(direction: 'start' | 'end') {\n    const elements = this._latestVisibilityStates$.value;\n\n    if (!elements.length) {\n      if (isDevMode()) {\n        console.warn(\n          'No elements found to scroll to. Make sure to apply the isElement directive to the elements you want to scroll to.',\n        );\n      }\n      return;\n    }\n\n    const el = direction === 'start' ? this.previousPartialIntersection : this.nextPartialIntersection;\n\n    if (!el) {\n      return;\n    }\n\n    this.scrollToElement({\n      element: el?.element,\n      direction: this.direction() === 'horizontal' ? 'inline' : 'block',\n      origin: direction,\n      ...(this.direction() === 'horizontal'\n        ? { scrollInlineMargin: this.scrollMargin() }\n        : { scrollBlockMargin: this.scrollMargin() }),\n    });\n  }\n\n  scrollToElement(options: Omit<ScrollToElementOptions, 'container'>) {\n    const scrollElement = this.scrollable()?.nativeElement;\n\n    scrollToElement({\n      container: scrollElement,\n      ...(this.direction() === 'horizontal'\n        ? { scrollInlineMargin: this.scrollMargin() }\n        : { scrollBlockMargin: this.scrollMargin() }),\n      ...options,\n    });\n  }\n\n  scrollToElementByIndex(options: Omit<ScrollToElementOptions, 'container'> & { index: number }) {\n    const elements = this.elementList()?.toArray() ?? [];\n\n    if (!elements.length) {\n      if (isDevMode()) {\n        console.warn(\n          'No elements found to scroll to. Make sure to apply the isElement directive to the elements you want to scroll to.',\n        );\n      }\n      return;\n    }\n\n    const scrollElement = this.scrollable()?.nativeElement;\n    const element = elements[options.index]?.elementRef.nativeElement;\n\n    scrollToElement({\n      container: scrollElement,\n      element,\n      ...(this.direction() === 'horizontal'\n        ? { scrollInlineMargin: this.scrollMargin() }\n        : { scrollBlockMargin: this.scrollMargin() }),\n      ...options,\n    });\n  }\n\n  protected setIsCursorDragging(isDragging: boolean) {\n    this._isCursorDragging$.next(isDragging);\n  }\n\n  protected scrollToStartDirection() {\n    if (this.scrollMode() === 'container') {\n      this.scrollOneContainerSize('start');\n    } else {\n      this.scrollOneItemSize('start');\n    }\n  }\n\n  protected scrollToStartEnd() {\n    if (this.scrollMode() === 'container') {\n      this.scrollOneContainerSize('end');\n    } else {\n      this.scrollOneItemSize('end');\n    }\n  }\n\n  private _setupScrollListening() {\n    const scrollElement = this.scrollable()?.nativeElement;\n    const elements = this.elementList();\n\n    if (!scrollElement || !elements) {\n      return;\n    }\n\n    let isSnapping = false;\n    let snapTimeout = 0;\n\n    merge(fromEvent<WheelEvent>(scrollElement, 'wheel'), fromEvent<TouchEvent>(scrollElement, 'touchstart'))\n      .pipe(\n        takeUntil(this._destroy$),\n        tap(() => {\n          isSnapping = false;\n        }),\n      )\n      .subscribe();\n\n    merge(\n      fromEvent(scrollElement, 'scroll'),\n      this._isCursorDragging$,\n      elements.changes.pipe(startWith(elements)) ?? of(null),\n    )\n      .pipe(\n        debounceTime(300),\n        takeUntil(this._destroy$),\n        tap(() => {\n          const els =\n            elements\n              ?.toArray()\n              .map((e) => e?.elementRef.nativeElement)\n              .filter((e): e is HTMLElement => !!e) ?? [];\n\n          if (!els.length) {\n            this._latestVisibilityStates$.next([]);\n\n            return;\n          }\n\n          const states = getElementVisibleStates({\n            elements: els,\n            container: scrollElement,\n          });\n\n          const prop = this.direction() === 'horizontal' ? 'inlineIntersection' : 'blockIntersection';\n          const stateClass = `et-element--is-intersecting`;\n\n          for (const state of states) {\n            if (state[prop] === 100) {\n              state.element.classList.add(stateClass);\n            } else {\n              state.element.classList.remove(stateClass);\n            }\n          }\n\n          const intersectionChanges = states.map((s, i) => {\n            const state: ScrollableIntersectionChange = {\n              element: s.element,\n              intersectionRatio: s[prop] / 100,\n              isIntersecting: s[prop] === 100,\n              index: i,\n            };\n\n            return state;\n          });\n\n          this.intersectionChange.emit(intersectionChanges);\n\n          this._latestVisibilityStates$.next(intersectionChanges);\n\n          if (isSnapping || this._isCursorDragging$.value || !this.snap()) return;\n\n          const prev = this.previousPartialIntersection;\n          const next = this.nextPartialIntersection;\n          const skipSnap =\n            !prev ||\n            !next ||\n            prev.intersectionRatio === 0 ||\n            next.intersectionRatio === 0 ||\n            prev.intersectionRatio === next.intersectionRatio;\n\n          if (skipSnap) return;\n\n          const highestIntersecting = prev.intersectionRatio > next.intersectionRatio ? prev : next;\n          const fullIntersectionIndex = this.highestVisibleIntersection?.index;\n\n          if (fullIntersectionIndex === undefined) return;\n\n          const highestIntersectingIndex = highestIntersecting.index;\n          const origin = highestIntersectingIndex > fullIntersectionIndex ? 'end' : 'start';\n\n          scrollToElement({\n            container: scrollElement,\n            element: highestIntersecting.element,\n            direction: this.direction() === 'horizontal' ? 'inline' : 'block',\n            origin,\n            scrollBlockMargin: this.direction() === 'horizontal' ? 0 : this.scrollMargin(),\n            scrollInlineMargin: this.direction() === 'horizontal' ? this.scrollMargin() : 0,\n          });\n\n          isSnapping = true;\n\n          window.clearTimeout(snapTimeout);\n\n          snapTimeout = window.setTimeout(() => {\n            isSnapping = false;\n          }, 1000);\n        }),\n      )\n      .subscribe();\n  }\n}\n","<div\n  #scrollable\n  [attr.role]=\"scrollableRole() ?? undefined\"\n  [etCursorDragScroll]=\"cursorDragScroll()\"\n  [allowedDirection]=\"direction()\"\n  (cursorDragStart)=\"setIsCursorDragging(true)\"\n  (cursorDragEnd)=\"setIsCursorDragging(false)\"\n  class=\"et-scrollable-container-outer\"\n>\n  <div [ngClass]=\"scrollableClass()\" class=\"et-scrollable-container\">\n    <div #firstElement class=\"et-scroll-observer-first-element\"></div>\n    <ng-content />\n    <div #lastElement class=\"et-scroll-observer-last-element\"></div>\n  </div>\n</div>\n\n@if (renderMasks()) {\n  <div class=\"et-scrollable-masks\">\n    <div class=\"et-scrollable-mask et-scrollable-mask--start\"></div>\n    <div class=\"et-scrollable-mask et-scrollable-mask--end\"></div>\n  </div>\n}\n\n@if (renderButtons()) {\n  <div class=\"et-scrollable-buttons\">\n    <button\n      (click)=\"scrollToStartDirection()\"\n      class=\"et-scrollable-button et-scrollable-button--start\"\n      aria-hidden=\"true\"\n      type=\"button\"\n      tabindex=\"-1\"\n    >\n      <et-chevron-icon />\n    </button>\n    <button\n      (click)=\"scrollToStartEnd()\"\n      class=\"et-scrollable-button et-scrollable-button--end\"\n      aria-hidden=\"true\"\n      type=\"button\"\n      tabindex=\"-1\"\n    >\n      <et-chevron-icon />\n    </button>\n  </div>\n}\n"]}
580
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable.component.js","sourceRoot":"","sources":["../../../../../../../../../libs/cdk/src/lib/components/scrollable/components/scrollable/scrollable.component.ts","../../../../../../../../../libs/cdk/src/lib/components/scrollable/components/scrollable/scrollable.component.html"],"names":[],"mappings":"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","sourcesContent":["import { NgClass } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  ContentChildren,\n  ElementRef,\n  EventEmitter,\n  Input,\n  Output,\n  ViewChild,\n  ViewEncapsulation,\n  booleanAttribute,\n  computed,\n  effect,\n  inject,\n  isDevMode,\n  numberAttribute,\n  signal,\n} from '@angular/core';\nimport { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';\nimport {\n  CurrentElementVisibility,\n  CursorDragScrollDirective,\n  IS_ACTIVE_ELEMENT,\n  IsActiveElementDirective,\n  LetDirective,\n  NgClassType,\n  ObserveScrollStateDirective,\n  ScrollObserverScrollState,\n  ScrollToElementOptions,\n  TypedQueryList,\n  getFirstAndLastPartialIntersection,\n  isElementVisible,\n  nextFrame,\n  scrollToElement,\n  signalElementChildren,\n  signalElementIntersection,\n  signalElementScrollState,\n  signalHostAttributes,\n  signalHostClasses,\n} from '@ethlete/core';\nimport {\n  BehaviorSubject,\n  Subject,\n  combineLatest,\n  debounceTime,\n  filter,\n  fromEvent,\n  map,\n  take,\n  takeUntil,\n  tap,\n} from 'rxjs';\nimport { ChevronIconComponent } from '../../../icons';\nimport { ScrollableIgnoreChildDirective, isScrollableChildIgnored } from '../../directives';\nimport { ScrollableIntersectionChange, ScrollableScrollMode } from '../../types';\n\n// Thresholds for the intersection observer.\nconst ELEMENT_INTERSECTION_THRESHOLD = [\n  // We use 51 steps to get a 2% step size.\n  ...Array.from({ length: 51 }, (_, i) => i / 50),\n\n  // Additional steps needed since display scaling can cause the intersection ratio to be slightly off.\n  0.01,\n  0.005,\n  0.001,\n  0.99,\n  0.995,\n  0.999,\n];\n\ninterface ScrollableNavigationItem {\n  isActive: boolean;\n  element: HTMLElement;\n}\n\n@Component({\n  selector: 'et-scrollable',\n  templateUrl: './scrollable.component.html',\n  styleUrls: ['./scrollable.component.scss'],\n  standalone: true,\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [\n    CursorDragScrollDirective,\n    ObserveScrollStateDirective,\n    NgClass,\n    LetDirective,\n    ChevronIconComponent,\n    ScrollableIgnoreChildDirective,\n  ],\n  host: {\n    class: 'et-scrollable',\n  },\n})\nexport class ScrollableComponent {\n  private readonly _elementRef = inject<ElementRef<HTMLElement>>(ElementRef);\n  private readonly _isCursorDragging$ = new BehaviorSubject<boolean>(false);\n\n  @Input({ alias: 'itemSize' })\n  set _itemSize(v: 'auto' | 'same' | 'full') {\n    this.itemSize.set(v);\n  }\n  readonly itemSize = signal<'auto' | 'same' | 'full'>('auto');\n\n  @Input({ alias: 'direction' })\n  set _direction(v: 'horizontal' | 'vertical') {\n    this.direction.set(v);\n  }\n  readonly direction = signal<'horizontal' | 'vertical'>('horizontal');\n\n  @Input({ alias: 'scrollableRole' })\n  set _scrollableRole(v: string | null) {\n    this.scrollableRole.set(v);\n  }\n  readonly scrollableRole = signal<string | null>(null);\n\n  @Input({ alias: 'scrollableClass' })\n  set _scrollableClass(v: NgClassType | null) {\n    this.scrollableClass.set(v);\n  }\n  readonly scrollableClass = signal<NgClassType | null>(null);\n\n  @Input({ transform: booleanAttribute, alias: 'renderNavigation' })\n  set _renderNavigation(v: boolean) {\n    this.renderNavigation.set(v);\n  }\n  readonly renderNavigation = signal(false);\n\n  @Input({ transform: booleanAttribute, alias: 'renderMasks' })\n  set _renderMasks(v: boolean) {\n    this.renderMasks.set(v);\n  }\n  readonly renderMasks = signal(true);\n\n  @Input({ transform: booleanAttribute, alias: 'renderButtons' })\n  set _renderButtons(v: boolean) {\n    this.renderButtons.set(v);\n  }\n  readonly renderButtons = signal(true);\n\n  @Input({ transform: booleanAttribute, alias: 'renderScrollbars' })\n  set _renderScrollbars(v: boolean) {\n    this.renderScrollbars.set(v);\n  }\n  readonly renderScrollbars = signal(false);\n\n  @Input({ transform: booleanAttribute, alias: 'stickyButtons' })\n  set _stickyButtons(v: boolean) {\n    this.stickyButtons.set(v);\n  }\n  readonly stickyButtons = signal(false);\n\n  @Input({ transform: booleanAttribute, alias: 'cursorDragScroll' })\n  set _cursorDragScroll(v: boolean) {\n    this.cursorDragScroll.set(v);\n  }\n  readonly cursorDragScroll = signal(true);\n\n  @Input({ transform: booleanAttribute, alias: 'disableActiveElementScrolling' })\n  set _disableActiveElementScrolling(v: boolean) {\n    this.disableActiveElementScrolling.set(v);\n  }\n  readonly disableActiveElementScrolling = signal(false);\n\n  @Input({ alias: 'scrollMode' })\n  set _scrollMode(v: ScrollableScrollMode) {\n    this.scrollMode.set(v);\n  }\n  readonly scrollMode = signal<ScrollableScrollMode>('container');\n\n  @Input({ transform: booleanAttribute, alias: 'snap' })\n  set _snap(v: boolean) {\n    this.snap.set(v);\n  }\n  readonly snap = signal(false);\n\n  @Input({ transform: numberAttribute, alias: 'scrollMargin' })\n  set _scrollMargin(v: number) {\n    this.scrollMargin.set(v);\n  }\n  readonly scrollMargin = signal(0);\n\n  @Output()\n  readonly scrollStateChange = new EventEmitter<ScrollObserverScrollState>();\n\n  @Output()\n  readonly intersectionChange = new EventEmitter<ScrollableIntersectionChange[]>();\n\n  @ViewChild('scrollable', { static: true })\n  private set _scrollable(e: ElementRef<HTMLElement>) {\n    this.scrollable.set(e);\n  }\n  readonly scrollable = signal<ElementRef<HTMLElement> | null>(null);\n\n  @ViewChild('scrollableContainer', { static: true })\n  private set _scrollableContainer(e: ElementRef<HTMLElement>) {\n    this.scrollableContainer.set(e);\n  }\n  readonly scrollableContainer = signal<ElementRef<HTMLElement> | null>(null);\n\n  @ViewChild('firstElement', { static: true })\n  private set _firstElement(e: ElementRef<HTMLElement>) {\n    this.firstElement.set(e);\n  }\n  readonly firstElement = signal<ElementRef<HTMLElement> | null>(null);\n\n  @ViewChild('lastElement', { static: true })\n  private set _lastElement(e: ElementRef<HTMLElement>) {\n    this.lastElement.set(e);\n  }\n  readonly lastElement = signal<ElementRef<HTMLElement> | null>(null);\n\n  @ContentChildren(IS_ACTIVE_ELEMENT, { descendants: true })\n  private set _activeElementList(e: TypedQueryList<IsActiveElementDirective>) {\n    this.activeElementList.set(e);\n  }\n  readonly activeElementList = signal<TypedQueryList<IsActiveElementDirective> | null>(null);\n\n  private readonly containerScrollState = signalElementScrollState(this.scrollable);\n  private readonly firstElementIntersection = signalElementIntersection(this.firstElement, { root: this.scrollable });\n  private readonly firstElementVisibility = signal<CurrentElementVisibility | null>(null);\n  private readonly lastElementIntersection = signalElementIntersection(this.lastElement, { root: this.scrollable });\n  private readonly lastElementVisibility = signal<CurrentElementVisibility | null>(null);\n\n  private readonly allScrollableChildren = signalElementChildren(this.scrollableContainer);\n  private readonly scrollableChildren = computed(() =>\n    this.allScrollableChildren().filter((c) => !isScrollableChildIgnored(c)),\n  );\n\n  private readonly _disableSnapping$ = new Subject<void>();\n\n  private readonly scrollableContentIntersections = signalElementIntersection(this.scrollableChildren, {\n    root: this.scrollable,\n    threshold: ELEMENT_INTERSECTION_THRESHOLD,\n  });\n\n  private readonly scrollableContentIntersections$ = toObservable(this.scrollableContentIntersections);\n\n  private readonly manualActiveNavigationIndex = signal<number | null>(null);\n\n  private readonly canScroll = computed(() => {\n    const dir = this.direction();\n\n    if (dir === 'horizontal') {\n      return this.containerScrollState().canScrollHorizontally;\n    }\n\n    return this.containerScrollState().canScrollVertically;\n  });\n\n  private readonly isAtStart = computed(() => {\n    if (!this.canScroll()) {\n      return true;\n    }\n\n    const intersection = this.firstElementIntersection()[0];\n\n    if (!intersection) {\n      return this.firstElementVisibility()?.inline ?? true;\n    }\n\n    return intersection.isIntersecting;\n  });\n  private readonly isAtEnd = computed(() => {\n    if (!this.canScroll()) {\n      return true;\n    }\n\n    const intersection = this.lastElementIntersection()[0];\n\n    if (!intersection) {\n      return this.lastElementVisibility()?.inline ?? true;\n    }\n\n    return intersection.isIntersecting;\n  });\n\n  private readonly enableOverlayAnimations = signal(false);\n\n  private readonly _initialScrollableNavigation = signal<ScrollableNavigationItem[]>([]);\n\n  protected readonly scrollableNavigation = computed<ScrollableNavigationItem[]>(() => {\n    const allIntersections = this.scrollableContentIntersections();\n    const manualActiveNavigationIndex = this.manualActiveNavigationIndex();\n    const initialScrollableNavigation = this._initialScrollableNavigation();\n\n    if (!allIntersections.length) {\n      return initialScrollableNavigation;\n    }\n\n    const highestIntersection = allIntersections.reduce((prev, curr) => {\n      if (prev && prev.intersectionRatio > curr.intersectionRatio) {\n        return prev;\n      }\n\n      return curr;\n    }, allIntersections[0]);\n\n    if (!highestIntersection) {\n      return [];\n    }\n\n    return allIntersections.map((i, index) => ({\n      isActive:\n        manualActiveNavigationIndex !== null\n          ? manualActiveNavigationIndex === index\n          : i === highestIntersection && highestIntersection.intersectionRatio > 0,\n      element: i.target as HTMLElement,\n    }));\n  });\n\n  readonly hostAttributeBindings = signalHostAttributes({\n    'item-size': this.itemSize,\n    direction: this.direction,\n    'render-scrollbars': this.renderScrollbars,\n    'sticky-buttons': this.stickyButtons,\n  });\n\n  readonly hostClassBindings = signalHostClasses({\n    'et-scrollable--can-scroll': this.canScroll,\n    'et-scrollable--is-at-start': this.isAtStart,\n    'et-scrollable--is-at-end': this.isAtEnd,\n    'et-scrollable--enable-overlay-animations': this.enableOverlayAnimations,\n  });\n\n  constructor() {\n    effect(\n      () => {\n        const scrollable = this.scrollable()?.nativeElement;\n        const firstElement = this.firstElement()?.nativeElement;\n        const lastElement = this.lastElement()?.nativeElement;\n        const activeElementList = this.activeElementList()?.toArray();\n\n        if (!scrollable || !firstElement || !lastElement || !activeElementList) {\n          return;\n        }\n\n        const firstActive = activeElementList.find((a) => a.isActiveElement);\n\n        if (firstActive && !this.disableActiveElementScrolling()) {\n          const offsetTop = firstActive.elementRef.nativeElement.offsetTop - scrollable.offsetTop;\n          const offsetLeft = firstActive.elementRef.nativeElement.offsetLeft - scrollable.offsetLeft;\n          scrollable.scrollLeft = offsetLeft - this.scrollMargin();\n          scrollable.scrollTop = offsetTop - this.scrollMargin();\n        }\n\n        this.firstElementVisibility.set(\n          isElementVisible({\n            container: scrollable,\n            element: firstElement,\n          }),\n        );\n\n        this.lastElementVisibility.set(\n          isElementVisible({\n            container: scrollable,\n            element: lastElement,\n          }),\n        );\n\n        // We need to wait one frame before enabling animations to prevent a animation from playing during initial render.\n        nextFrame(() => this.enableOverlayAnimations.set(true));\n      },\n      { allowSignalWrites: true },\n    );\n\n    effect(\n      () => {\n        const elementList = this.scrollableChildren();\n        const scrollable = this.scrollable()?.nativeElement;\n        const renderNavigation = this.renderNavigation();\n\n        if (!elementList || !scrollable || !renderNavigation) {\n          return;\n        }\n\n        const firstVisibleElement = elementList.find((e) => isElementVisible({ container: scrollable, element: e }));\n\n        if (!firstVisibleElement) {\n          return;\n        }\n\n        const initialNavigationStuff: ScrollableNavigationItem[] = elementList.map((e) => ({\n          isActive: e === firstVisibleElement,\n          element: e,\n        }));\n\n        this._initialScrollableNavigation.set(initialNavigationStuff);\n      },\n      { allowSignalWrites: true },\n    );\n\n    effect(() => {\n      const isAtStart = this.isAtStart();\n      const isAtEnd = this.isAtEnd();\n      const canScroll = this.canScroll();\n\n      this.scrollStateChange.emit({\n        canScroll,\n        isAtEnd: !!isAtEnd,\n        isAtStart: !!isAtStart,\n      });\n    });\n\n    effect(() => {\n      const enableSnapping = this.snap();\n\n      if (enableSnapping) {\n        this._enableSnapping();\n      } else {\n        this._disableSnapping();\n      }\n    });\n\n    this.scrollableContentIntersections$\n      .pipe(\n        takeUntilDestroyed(),\n        debounceTime(10),\n        tap((entries) => {\n          this.intersectionChange.emit(\n            entries.map((i, index) => ({\n              index,\n              element: i.target as HTMLElement,\n              intersectionRatio: i.intersectionRatio,\n              isIntersecting: i.isIntersecting,\n            })),\n          );\n        }),\n      )\n      .subscribe();\n  }\n\n  scrollOneContainerSize(direction: 'start' | 'end') {\n    const scrollElement = this.scrollable()?.nativeElement;\n\n    if (!scrollElement) {\n      return;\n    }\n\n    const parent = this._elementRef.nativeElement;\n\n    const isSnappingEnabled = this.snap();\n\n    if (isSnappingEnabled) {\n      // If snapping is enabled we want to scroll to a position where no further snapping will happen after the scroll.\n      const allIntersections = this.scrollableContentIntersections();\n      const intersections = getFirstAndLastPartialIntersection(allIntersections);\n      const relevantIntersection = direction === 'start' ? intersections?.first : intersections?.last;\n\n      if (!relevantIntersection) return;\n\n      const nextIndex =\n        relevantIntersection.intersection.intersectionRatio !== 1\n          ? relevantIntersection.index\n          : direction === 'start'\n            ? relevantIntersection.index - 1\n            : relevantIntersection.index + 1;\n\n      const element =\n        (allIntersections[nextIndex]?.target as HTMLElement) ||\n        (relevantIntersection.intersection.target as HTMLElement);\n\n      this.scrollToElement({\n        element: element,\n        origin: direction === 'end' ? 'start' : 'end',\n      });\n    } else {\n      // Just scroll one size of the scrollable container.\n      const scrollableSize = this.direction() === 'horizontal' ? parent.clientWidth : parent.clientHeight;\n      const currentScroll = this.direction() === 'horizontal' ? scrollElement.scrollLeft : scrollElement.scrollTop;\n\n      scrollElement.scrollTo({\n        [this.direction() === 'horizontal' ? 'left' : 'top']:\n          currentScroll + (direction === 'start' ? -scrollableSize : scrollableSize),\n        behavior: 'smooth',\n      });\n    }\n  }\n\n  scrollOneItemSize(direction: 'start' | 'end') {\n    const allIntersections = this.scrollableContentIntersections();\n    const scrollElement = this.scrollable()?.nativeElement;\n\n    if (!allIntersections.length) {\n      if (isDevMode()) {\n        console.warn(\n          'No elements found to scroll to. Make sure to apply the isElement directive to the elements you want to scroll to.',\n        );\n      }\n      return;\n    }\n\n    const intersections = getFirstAndLastPartialIntersection(allIntersections);\n\n    if (!intersections || !scrollElement) return;\n\n    // Means the current element is bigger than the scrollable container.\n    // In this case we should scroll to the start of the current element. If we are already there we should scroll to the end of the previous element.\n    // This applies to the other direction as well.\n    const isFirstAndLastIntersectionEqual = intersections.first.intersection === intersections.last.intersection;\n    const scrollableRect = scrollElement.getBoundingClientRect();\n\n    if (isFirstAndLastIntersectionEqual) {\n      const intersection = intersections.first.intersection.target.getBoundingClientRect();\n      const isStartOfElementVisible =\n        this.direction() === 'horizontal'\n          ? intersection.left >= scrollableRect.left\n          : intersection.top >= scrollableRect.top;\n\n      const isEndOfElementVisible =\n        this.direction() === 'horizontal'\n          ? intersection.right <= scrollableRect.right\n          : intersection.bottom <= scrollableRect.bottom;\n\n      if (!isStartOfElementVisible || !isEndOfElementVisible) {\n        if (direction === 'start') {\n          if (isStartOfElementVisible) {\n            // to the end of the previous element\n            const previousIndex = intersections.first.index - 1;\n            const elementToScrollTo = allIntersections[previousIndex]?.target as HTMLElement;\n\n            if (!elementToScrollTo) return;\n\n            this.scrollToElement({\n              element: elementToScrollTo,\n              origin: 'end',\n            });\n          } else {\n            // to the start of the current element\n            this.scrollToElement({\n              element: intersections.first.intersection.target as HTMLElement,\n              origin: 'start',\n            });\n          }\n        } else {\n          if (isEndOfElementVisible) {\n            // to the start of the next element\n            const nextIndex = intersections.last.index + 1;\n            const elementToScrollTo = allIntersections[nextIndex]?.target as HTMLElement;\n\n            if (!elementToScrollTo) return;\n\n            this.scrollToElement({\n              element: elementToScrollTo,\n              origin: 'start',\n            });\n          } else {\n            // to the end of the current element\n            this.scrollToElement({\n              element: intersections.last.intersection.target as HTMLElement,\n              origin: 'end',\n            });\n          }\n        }\n\n        return;\n      }\n    }\n\n    const data = direction === 'start' ? intersections.first : intersections.last;\n    let elementToScrollTo = data.intersection.target as HTMLElement;\n\n    if (data.intersection.intersectionRatio === 1) {\n      if (direction === 'start' && data.index === 0) {\n        return;\n      }\n\n      if (direction === 'end' && data.index === allIntersections.length - 1) {\n        return;\n      }\n\n      const nextIndex = direction === 'start' ? data.index - 1 : data.index + 1;\n\n      elementToScrollTo = allIntersections[nextIndex]?.target as HTMLElement;\n    }\n\n    if (!elementToScrollTo) return;\n\n    this.scrollToElement({\n      element: elementToScrollTo,\n      origin: direction,\n    });\n  }\n\n  scrollToElement(options: Omit<ScrollToElementOptions, 'container'>) {\n    const scrollElement = this.scrollable()?.nativeElement;\n\n    scrollToElement({\n      container: scrollElement,\n      direction: this.direction() === 'horizontal' ? 'inline' : 'block',\n      ...(this.direction() === 'horizontal'\n        ? { scrollInlineMargin: this.scrollMargin() }\n        : { scrollBlockMargin: this.scrollMargin() }),\n      ...options,\n    });\n  }\n\n  scrollToElementByIndex(options: Omit<ScrollToElementOptions, 'container'> & { index: number }) {\n    const elements = this.scrollableChildren();\n\n    if (!elements.length) {\n      if (isDevMode()) {\n        console.warn(\n          'No elements found to scroll to. Make sure to apply the isElement directive to the elements you want to scroll to.',\n        );\n      }\n      return;\n    }\n\n    const scrollElement = this.scrollable()?.nativeElement;\n    const element = elements[options.index];\n\n    scrollToElement({\n      container: scrollElement,\n      element,\n      ...(this.direction() === 'horizontal'\n        ? { scrollInlineMargin: this.scrollMargin() }\n        : { scrollBlockMargin: this.scrollMargin() }),\n      ...options,\n    });\n  }\n\n  protected scrollToElementViaNavigation(elementIndex: number, element: HTMLElement) {\n    this.manualActiveNavigationIndex.set(elementIndex);\n\n    this.scrollToElement({\n      element,\n    });\n\n    const scrollElement = this.scrollable()?.nativeElement;\n\n    if (!scrollElement) return;\n\n    fromEvent(scrollElement, 'scroll')\n      .pipe(\n        debounceTime(50),\n        take(1),\n        tap(() => this.manualActiveNavigationIndex.set(null)),\n      )\n      .subscribe();\n  }\n\n  protected setIsCursorDragging(isDragging: boolean) {\n    this._isCursorDragging$.next(isDragging);\n  }\n\n  protected scrollToStartDirection() {\n    if (this.scrollMode() === 'container') {\n      this.scrollOneContainerSize('start');\n    } else {\n      this.scrollOneItemSize('start');\n    }\n  }\n\n  protected scrollToEndDirection() {\n    if (this.scrollMode() === 'container') {\n      this.scrollOneContainerSize('end');\n    } else {\n      this.scrollOneItemSize('end');\n    }\n  }\n\n  private _enableSnapping() {\n    combineLatest([this.scrollableContentIntersections$, this._isCursorDragging$])\n      .pipe(\n        filter(([, isDragging]) => !isDragging),\n        map(([intersections]) => intersections),\n        debounceTime(150),\n        tap((allIntersections) => {\n          const scrollElement = this.scrollable()?.nativeElement;\n\n          if (!scrollElement) return;\n\n          const intersections = getFirstAndLastPartialIntersection(allIntersections);\n\n          if (!intersections) return;\n\n          const isFirstAndLastIntersectionEqual = intersections.first.intersection === intersections.last.intersection;\n          const scrollableRect = scrollElement.getBoundingClientRect();\n\n          if (isFirstAndLastIntersectionEqual) {\n            const intersection = intersections.first.intersection.target.getBoundingClientRect();\n            const isStartOfElementVisible =\n              this.direction() === 'horizontal'\n                ? intersection.left >= scrollableRect.left\n                : intersection.top >= scrollableRect.top;\n\n            const isEndOfElementVisible =\n              this.direction() === 'horizontal'\n                ? intersection.right <= scrollableRect.right\n                : intersection.bottom <= scrollableRect.bottom;\n\n            // Don't snap if neither the start nor the end of the current element is visible.\n            // Otherwise this would result in parts of the element being inaccessible.\n            if (!isStartOfElementVisible && !isEndOfElementVisible) return;\n\n            // If the start of the element is visible we should snap to the start.\n            if (isStartOfElementVisible) {\n              this.scrollToElement({\n                element: intersections.first.intersection.target as HTMLElement,\n                origin: 'start',\n              });\n              return;\n            }\n\n            // If the end of the element is visible we should snap to the end.\n            if (isEndOfElementVisible) {\n              this.scrollToElement({\n                element: intersections.last.intersection.target as HTMLElement,\n                origin: 'end',\n              });\n              return;\n            }\n          } else if (\n            (this.direction() === 'horizontal' &&\n              intersections.biggest.intersection.boundingClientRect.width > scrollableRect.width) ||\n            (this.direction() === 'vertical' &&\n              intersections.biggest.intersection.boundingClientRect.height > scrollableRect.height)\n          ) {\n            // If the current element is bigger than the scrollable container we should snap to the start of the current element if the scroll direction is forward\n            // and to the end of the current element if the scroll direction is backwards.\n            const origin = intersections.biggest.index === intersections.first.index ? 'end' : 'start';\n\n            this.scrollToElement({\n              element: intersections.biggest.intersection.target as HTMLElement,\n              origin,\n            });\n          } else {\n            // No special case. Just snap to the biggest intersection.\n            this.scrollToElement({\n              element: intersections.biggest.intersection.target as HTMLElement,\n            });\n          }\n        }),\n        takeUntil(this._disableSnapping$),\n      )\n      .subscribe();\n  }\n\n  private _disableSnapping() {\n    this._disableSnapping$.next();\n  }\n}\n","<div class=\"et-scrollable-wrapper\">\n  <div\n    #scrollable\n    [attr.role]=\"scrollableRole() ?? undefined\"\n    [etCursorDragScroll]=\"cursorDragScroll()\"\n    [allowedDirection]=\"direction()\"\n    (cursorDragStart)=\"setIsCursorDragging(true)\"\n    (cursorDragEnd)=\"setIsCursorDragging(false)\"\n    class=\"et-scrollable-container-outer\"\n  >\n    <div #scrollableContainer [ngClass]=\"scrollableClass()\" class=\"et-scrollable-container\">\n      <div #firstElement class=\"et-scroll-observer-first-element\" etScrollableIgnoreChild></div>\n      <ng-content />\n      <div #lastElement class=\"et-scroll-observer-last-element\" etScrollableIgnoreChild></div>\n    </div>\n  </div>\n\n  @if (renderMasks()) {\n    <div class=\"et-scrollable-masks\">\n      <div class=\"et-scrollable-mask et-scrollable-mask--start\"></div>\n      <div class=\"et-scrollable-mask et-scrollable-mask--end\"></div>\n    </div>\n  }\n\n  @if (renderButtons()) {\n    <div class=\"et-scrollable-buttons\">\n      <button\n        (click)=\"scrollToStartDirection()\"\n        class=\"et-scrollable-button et-scrollable-button--start\"\n        aria-hidden=\"true\"\n        type=\"button\"\n        tabindex=\"-1\"\n      >\n        <et-chevron-icon />\n      </button>\n      <button\n        (click)=\"scrollToEndDirection()\"\n        class=\"et-scrollable-button et-scrollable-button--end\"\n        aria-hidden=\"true\"\n        type=\"button\"\n        tabindex=\"-1\"\n      >\n        <et-chevron-icon />\n      </button>\n    </div>\n  }\n</div>\n\n@if (renderNavigation()) {\n  <div class=\"et-scrollable-navigation\" aria-hidden=\"true\">\n    @for (item of scrollableNavigation(); track i; let i = $index) {\n      <button\n        [class.et-scrollable-navigation-item--active]=\"item.isActive\"\n        (click)=\"scrollToElementViaNavigation(i, item.element)\"\n        class=\"et-scrollable-navigation-item\"\n      ></button>\n    }\n  </div>\n}\n"]}