1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import { useTimeout } from '@vueuse/core'
- import { ref } from 'vue'
- export const useBubbleHeader = () => {
- const showMetaInformation = ref(false)
- const isInteractiveTarget = (target: HTMLElement) => {
- if (!target) return false
- const interactiveElements = new Set(['A', 'BUTTON'])
- // Parent interactive or traversed nodes
- const hasInteractiveElements = target.closest(
- Array.from(interactiveElements).join(','),
- )
- return interactiveElements.has(target.tagName) || hasInteractiveElements
- }
- const hasSelectionRange = (target: HTMLElement) => {
- if (!target) return false
- const selection = window.getSelection()
- if (!selection || selection.type !== 'Range') return false
- return true
- }
- const { start, stop } = useTimeout(200, {
- controls: true,
- callback: () => {
- showMetaInformation.value = !showMetaInformation.value
- },
- immediate: false,
- })
- const toggleHeader = async (event: MouseEvent) => {
- stop()
- if (
- event.detail === 2 || // Double-click
- isInteractiveTarget(event.target as HTMLElement) ||
- hasSelectionRange(event.target as HTMLElement)
- )
- return
- start()
- }
- return {
- showMetaInformation,
- toggleHeader,
- }
- }
|