// Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
import { fireEvent, waitFor } from '@testing-library/vue'
import { beforeEach, describe, vi } from 'vitest'
import renderComponent from '#tests/support/components/renderComponent.ts'
import { mockLocale } from '#tests/support/mock-locale.ts'
describe('TooltipDirective', () => {
describe('on non-touch device', () => {
it('should show/hide tooltip on hover', async () => {
const wrapper = renderComponent({
template: `
Foo Test World
`,
})
await wrapper.events.hover(wrapper.getByText('Foo Test World'))
await waitFor(() =>
expect(wrapper.queryByText('Hello, Tooltip')).toBeInTheDocument(),
)
await wrapper.events.unhover(wrapper.getByText('Foo Test World'))
await waitFor(() => {
expect(wrapper.queryByText('Hello, Tooltip')).not.toBeInTheDocument()
})
})
it('has accessibility attribute', async () => {
const wrapper = renderComponent({
template: `
Foo Test World
`,
})
await waitFor(() =>
expect(wrapper.queryByLabelText('Hello, Tooltip')).toBeInTheDocument(),
)
})
it('should hide tooltip on scroll', async () => {
const wrapper = renderComponent({
template: `
Foo Test World
`,
})
await wrapper.events.hover(wrapper.getByText('Foo Test World'))
await waitFor(() =>
expect(wrapper.queryByText('Hello, Tooltip')).toBeInTheDocument(),
)
window.dispatchEvent(new Event('scroll'))
await waitFor(() =>
expect(wrapper.queryByText('Hello, Tooltip')).not.toBeInTheDocument(),
)
})
})
describe('on touch device', () => {
beforeEach(() => {
vi.mock('#shared/composables/useTouchDevice.ts', () => ({
useTouchDevice: vi
.fn()
.mockReturnValue({ isTouchDevice: { value: true } }),
}))
})
it('should hide tooltip on first touch', async () => {
const wrapper = renderComponent({
template: `
Foo Test World
`,
})
await fireEvent.touchStart(wrapper.getByText('Foo Test World'))
await waitFor(() => {
expect(wrapper.queryByText('Hello, Tooltip')).toBeInTheDocument()
})
await fireEvent.touchStart(wrapper.getByText('Foo Test World'))
await fireEvent.touchEnd(wrapper.getByText('Foo Test World'))
await waitFor(() =>
expect(wrapper.queryByText('Hello, Tooltip')).not.toBeInTheDocument(),
)
})
it('updated tooltip locale', async () => {
const translationSpy = mockLocale('Hello, Tooltip', 'Hola, Tooltip')
const wrapper = renderComponent({
template: `
Foo Test World
`,
})
await wrapper.events.hover(wrapper.getByText('Foo Test World'))
await waitFor(() => {
expect(wrapper.queryByText('Hola, Tooltip')).toBeInTheDocument()
expect(wrapper.getByLabelText('Hola, Tooltip')).toBeInTheDocument()
})
expect(translationSpy).toHaveBeenCalledOnce()
})
})
describe('modifiers', () => {
it.todo('detects truncation if modifier is set', async () => {
// :TODO - Move this to a real browser env -> Cypress
let wrapper = renderComponent({
template: `
`,
})
await wrapper.events.hover(wrapper.getByText('Short Text'))
await waitFor(() => {
expect(wrapper.queryByText('Foo Test world')).not.toBeInTheDocument()
})
wrapper = renderComponent({
template: `
This is a very long text that will be truncated
`,
})
await wrapper.events.hover(
wrapper.getByText('This is a very long text that will be truncated'),
)
})
})
})