12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import { useScroll, useThrottleFn } from '@vueuse/core'
- import { whenever } from '@vueuse/shared'
- import { computed, type ComputedRef, isRef, ref, watch } from 'vue'
- import type { MaybeRef } from '@vueuse/shared'
- interface Options {
- scrollStartThreshold?: ComputedRef<number | undefined>
- }
- export const useElementScroll = (
- scrollContainerElement: MaybeRef<HTMLElement>,
- options?: Options,
- ) => {
- const { y, directions } = useScroll(scrollContainerElement, {
- eventListenerOptions: { passive: true },
- })
- const isScrollingDown = ref(false)
- const isScrollingUp = ref(false)
- const resetScrolls = () => {
- isScrollingDown.value = false
- isScrollingUp.value = false
- }
- const reachedTop = computed(() => y.value === 0)
- const scrollNode = computed(() =>
- isRef(scrollContainerElement)
- ? scrollContainerElement.value
- : scrollContainerElement,
- )
- const reachedBottom = computed(
- () =>
- // NB: Check if this is the most optimal calculation.
- // In Webkit based browsers it sometimes results in -0.5 right on the bottom edge,
- // hence the need for the lower bound.
- y.value -
- (scrollNode.value?.scrollHeight ?? 0) +
- (scrollNode.value?.offsetHeight ?? 0) >
- -1,
- )
- const isScrollable = computed(
- () => scrollNode.value?.scrollHeight > scrollNode.value?.clientHeight,
- )
- const hasReachedThreshold = computed(
- () => y.value > (options?.scrollStartThreshold?.value || 0),
- )
- const omitValueChanges = computed(() => {
- return !hasReachedThreshold.value || !isScrollable.value || reachedTop.value
- })
- whenever(reachedTop, resetScrolls, { flush: 'post' })
- const throttledFn = useThrottleFn((newY, oldY) => {
- if (omitValueChanges.value) return
- if (hasReachedThreshold.value) {
- resetScrolls()
- }
- if (newY > oldY) {
- isScrollingDown.value = true
- isScrollingUp.value = false
- }
- if (newY < oldY) {
- isScrollingDown.value = false
- isScrollingUp.value = true
- }
- }, 500) // avoid scrolling glitch
- watch(y, throttledFn, { flush: 'post' })
- return {
- y,
- directions,
- reachedTop,
- reachedBottom,
- isScrollingDown,
- isScrollingUp,
- isScrollable,
- }
- }
|