// Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ import { h } from 'vue' import { renderComponent } from '#tests/support/components/index.ts' import CommonPopover from '#shared/components/CommonPopover/CommonPopover.vue' import { usePopover } from '#shared/components/CommonPopover/usePopover.ts' import CommonPopoverMenu from '../CommonPopoverMenu.vue' import type { MenuItem } from '../types.ts' const html = String.raw const fn = vi.fn() describe('rendering section', () => { it('no output without default slot and items', () => { const view = renderComponent(CommonPopoverMenu, { props: { popover: null, headerLabel: 'Test Header', }, router: true, }) expect(view.queryByText('Test Header')).not.toBeInTheDocument() }) it('if have header prop, renders header', () => { const view = renderComponent(CommonPopoverMenu, { props: { popover: null, headerLabel: 'Test Header', items: [ { label: 'Example', }, ], }, router: true, store: true, }) expect(view.getByText('Test Header')).toBeInTheDocument() }) it('if have header slot, renders header', () => { const view = renderComponent(CommonPopoverMenu, { props: { popover: null, }, slots: { header: '
Test Header
', default: 'Example', }, router: true, }) expect(view.getByText('Test Header')).toBeInTheDocument() }) it('rendering items', () => { const items: MenuItem[] = [ { key: 'login', link: '/login', label: 'Login' }, { key: 'dashboard', link: '/', label: 'Link' }, ] const view = renderComponent(CommonPopoverMenu, { shallow: false, props: { popover: null, items, }, router: true, }) expect(view.getByText('Login')).toBeInTheDocument() expect(view.getByText('Link')).toBeInTheDocument() }) it('rendering only items with permission', () => { const items: MenuItem[] = [ { key: 'login', link: '/login', label: 'Login' }, { key: 'dashboard', link: '/', label: 'Link', permission: ['example'] }, ] const view = renderComponent(CommonPopoverMenu, { shallow: false, props: { popover: null, items, }, router: true, }) expect(view.getByText('Login')).toBeInTheDocument() expect(view.queryByText('Link')).not.toBeInTheDocument() }) it('support click handler on item', async () => { const clickHandler = vi.fn() const items: MenuItem[] = [ { key: 'example', onClick: clickHandler, label: 'Example' }, ] const view = renderComponent(CommonPopoverMenu, { shallow: false, props: { popover: null, items, }, router: true, }) await view.events.click(view.getByText('Example')) expect(clickHandler).toHaveBeenCalledOnce() }) it('close popover on click on item or avoid closing', async () => { const clickHandlerExample = vi.fn() const clickHandlerOther = vi.fn() const view = renderComponent({ components: { CommonPopover, CommonPopoverMenu }, template: html` `, setup() { const { popover, popoverTarget, toggle } = usePopover() const items: MenuItem[] = [ { key: 'example', onClick: clickHandlerExample, label: 'Example' }, { key: 'other', onClick: clickHandlerOther, label: 'Other', noCloseOnClick: true, }, ] return { items, toggle, popover, popoverTarget, } }, }) await view.events.click(view.getByText('Click me')) expect(view.queryByText('Example')).toBeInTheDocument() await view.events.click(view.getByText('Example')) expect(clickHandlerExample).toHaveBeenCalledOnce() expect(view.queryByText('Example')).not.toBeInTheDocument() await view.events.click(view.getByText('Click me')) await view.events.click(view.getByText('Other')) expect(clickHandlerOther).toHaveBeenCalledOnce() expect(view.queryByText('Other')).toBeInTheDocument() }) it('can use an own component for item rendering', async () => { const CustomComponent = (props: any) => { return h('div', `Example ${props.label}`) } CustomComponent.props = ['label'] const items: MenuItem[] = [ { key: 'menu-item', component: CustomComponent, label: 'Menu item' }, ] const view = renderComponent(CommonPopoverMenu, { shallow: false, props: { popover: null, items, }, router: true, }) expect(view.getByText('Example Menu item')).toBeInTheDocument() }) it('yields entity data on show if prop is passed', async () => { renderComponent(CommonPopoverMenu, { props: { popover: null, headerLabel: 'Test Header', entity: { id: 'example', name: 'vitest', }, items: [ { label: 'Example', show: (event: { id: string; name: string }) => { fn(event) return true }, }, ], }, router: true, store: true, }) expect(fn).toBeCalledWith({ id: 'example', name: 'vitest' }) }) it('supports display of groups', () => { const items = [ { key: 'group-test', label: 'group test', groupLabel: 'test group' }, { key: 'group-test-2', label: 'group test 2', groupLabel: 'test group', }, { key: 'group-test-2', label: 'group test 3', groupLabel: 'test group', }, { key: 'single-group-test-2', label: 'single-group test 3', groupLabel: 'single group', }, { key: 'single-test', label: 'single test' }, ] const view = renderComponent(CommonPopoverMenu, { shallow: false, props: { popover: null, items, }, router: true, }) expect(view.getByRole('button', { name: 'group test' })).toBeInTheDocument() expect( view.getByRole('button', { name: 'group test 2' }), ).toBeInTheDocument() expect( view.getByRole('button', { name: 'group test 3' }), ).toBeInTheDocument() expect( view.getByRole('button', { name: 'single-group test 3' }), ).toBeInTheDocument() expect( view.getByRole('button', { name: 'single test' }), ).toBeInTheDocument() expect( view.getByRole('heading', { name: 'test group', level: 3 }), ).toBeInTheDocument() expect( view.getByRole('heading', { name: 'single group', level: 3 }), ).toBeInTheDocument() }) })