123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import { useScroll, useThrottleFn, whenever } from '@vueuse/core'
- import {
- computed,
- onMounted,
- onActivated,
- type Ref,
- ref,
- type ShallowRef,
- watch,
- } from 'vue'
- import type { TicketById } from '#shared/entities/ticket/types.ts'
- import type ArticleList from '#desktop/pages/ticket/components/TicketDetailView/ArticleList.vue'
- import TicketDetailTopBar from '#desktop/pages/ticket/components/TicketDetailView/TicketDetailTopBar/TicketDetailTopBar.vue'
- export const useArticleContainerScroll = (
- ticket: Ref<TicketById>,
- contentContainerElement: Readonly<ShallowRef<HTMLDivElement | null>>,
- articleListInstance: Readonly<
- ShallowRef<InstanceType<typeof ArticleList> | null>
- >,
- topBarInstance: Readonly<
- ShallowRef<InstanceType<typeof TicketDetailTopBar> | null>
- >,
- ) => {
- const THROTTLE_TIME = 250
- let hasMounted = false
- const { arrivedState } = useScroll(contentContainerElement, {
- eventListenerOptions: { passive: true },
- })
- const isHidingTicketDetails = ref(false)
- const isReachingBottom = ref(false)
- const previousPosition = ref(0)
- const reset = () => {
- isReachingBottom.value = false
- isHidingTicketDetails.value = false
- previousPosition.value = 0
- }
- const _isHoveringOnTopBar = ref(false)
- const isHoveringOnTopBar = computed({
- get: () => _isHoveringOnTopBar.value,
- set: (value) => {
- _isHoveringOnTopBar.value = value
- if (value) {
- isHidingTicketDetails.value = false
- }
- },
- })
- const handleScroll = useThrottleFn((event: Event) => {
- // Makes sure we always show the details on initial render and reactivated component.
- if (!hasMounted) {
- hasMounted = true
- return
- }
- const container = event.target! as HTMLDivElement
- const { scrollHeight, clientHeight } = container
- const isScrollable = scrollHeight > clientHeight
- if (!isScrollable) return reset()
- const scrollTop = container.scrollTop ?? 0
- isReachingBottom.value = scrollTop + clientHeight < scrollHeight
- // If we keep the pointer on the top bar we do not want to hide the details if the user starts to scroll on the same time.
- if (!isHoveringOnTopBar.value) {
- isHidingTicketDetails.value =
- scrollTop > (topBarInstance.value?.$el.clientHeight ?? 0)
- }
- previousPosition.value = scrollTop
- }, THROTTLE_TIME)
- whenever(
- () => arrivedState.top,
- () => {
- if (isHoveringOnTopBar.value) return
- isHidingTicketDetails.value = false
- },
- )
- watch(
- () => arrivedState.bottom,
- (value) => {
- isReachingBottom.value = !value
- },
- )
- watch(
- () => ticket.value?.id,
- () => {
- articleListInstance.value?.setDidInitialScroll(false)
- },
- { immediate: true },
- )
- watch(
- () => articleListInstance.value?.rows,
- async () => {
- if (articleListInstance.value?.didScrollInitially) return
- await articleListInstance.value?.scrollToArticle()
- articleListInstance.value?.setDidInitialScroll(true)
- // Normally handleScroll runs after we this, in some edge cases if it is not triggered we reset the states.
- reset()
- },
- { flush: 'post' },
- )
- // Handling scrolling to bottom if new article is added
- watch(
- () => articleListInstance.value?.rows,
- (newRows, oldRows) => {
- if (!newRows || !oldRows) return
- if (newRows.at(-1)?.key === oldRows.at(-1)?.key) return
- // article got removed
- if (newRows.at(-1)?.key === oldRows.at(-2)?.key) return
- // article got added
- articleListInstance.value?.scrollToArticle()
- },
- )
- onMounted(() => {
- hasMounted = false
- isHidingTicketDetails.value = false
- })
- onActivated(() => {
- hasMounted = false
- isHidingTicketDetails.value = false
- })
- return {
- handleScroll,
- isHoveringOnTopBar,
- isHidingTicketDetails,
- isReachingBottom,
- }
- }
|