123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import renderComponent from '#tests/support/components/renderComponent.ts'
- import type { ObjectLike } from '#shared/types/utils.ts'
- import CommonActionMenu from '#desktop/components/CommonActionMenu/CommonActionMenu.vue'
- import type { MenuItem } from '#desktop/components/CommonPopoverMenu/types.ts'
- import type { Props } from '../CommonActionMenu.vue'
- const fn = vi.fn()
- describe('CommonActionMenu', () => {
- const actions: MenuItem[] = [
- {
- key: 'delete-foo',
- label: 'Delete Foo',
- icon: 'trash3',
- show: () => true,
- onClick: (entity?: ObjectLike) => {
- fn(entity?.id)
- },
- },
- {
- key: 'change-foo',
- label: 'Change Foo',
- icon: 'person-gear',
- show: () => true,
- onClick: (entity?: ObjectLike) => {
- fn(entity?.id)
- },
- },
- ]
- const renderActionMenu = async (
- actions: MenuItem[],
- props?: Partial<Props>,
- ) => {
- return renderComponent(CommonActionMenu, {
- router: true,
- props: {
- ...props,
- entity: {
- id: 'foo-test-action',
- },
- actions,
- },
- })
- }
- describe('Multiple Actions', () => {
- it('shows action menu button by default', async () => {
- const view = await renderActionMenu(actions)
- expect(view.getByIconName('three-dots-vertical')).toBeInTheDocument()
- })
- it('show not content when no item exists', async () => {
- const view = await renderActionMenu([
- {
- ...actions[0],
- show: () => false,
- },
- {
- ...actions[1],
- show: () => false,
- },
- {
- key: 'example',
- label: 'Example',
- show: () => false,
- },
- ])
- expect(
- view.queryByIconName('three-dots-vertical'),
- ).not.toBeInTheDocument()
- })
- it('calls onClick handler when action is clicked', async () => {
- const view = await renderActionMenu(actions)
- await view.events.click(view.getByIconName('three-dots-vertical'))
- expect(view.getByIconName('trash3')).toBeInTheDocument()
- expect(view.getByIconName('person-gear')).toBeInTheDocument()
- await view.events.click(view.getByText('Change Foo'))
- expect(fn).toHaveBeenCalledWith('foo-test-action')
- })
- it('finds corresponding a11y controls', async () => {
- const view = await renderActionMenu(actions)
- await view.events.click(view.getByIconName('three-dots-vertical'))
- const id = view
- .getByLabelText('Action menu button')
- .getAttribute('aria-controls')
- const popover = document.getElementById(id as string)
- expect(popover?.getAttribute('id')).toEqual(id)
- })
- it('sets a custom aria label on single action button', async () => {
- const view = await renderActionMenu(actions, {
- customMenuButtonLabel: 'Custom Action Menu Label',
- })
- await view.rerender({
- customMenuButtonLabel: 'Custom Action Menu Label',
- })
- expect(
- view.getByLabelText('Custom Action Menu Label'),
- ).toBeInTheDocument()
- })
- })
- describe('single action mode', () => {
- it('adds aria label on single action button', async () => {
- const view = await renderActionMenu([actions[0]])
- expect(view.getByLabelText('Delete Foo')).toBeInTheDocument()
- })
- it('supports single action mode', async () => {
- const view = await renderActionMenu([actions[0]])
- expect(
- view.queryByIconName('three-dots-vertical'),
- ).not.toBeInTheDocument()
- expect(view.getByIconName('trash3')).toBeInTheDocument()
- })
- it('calls onClick handler when action is clicked', async () => {
- const view = await renderActionMenu([actions[0]])
- await view.events.click(view.getByIconName('trash3'))
- expect(fn).toHaveBeenCalledWith('foo-test-action')
- })
- it('renders single action if prop is set', async () => {
- const view = await renderActionMenu([actions[0]], {
- noSingleActionMode: true,
- })
- expect(view.queryByIconName('trash3')).not.toBeInTheDocument()
- await view.events.click(view.getByIconName('three-dots-vertical'))
- expect(view.getByIconName('trash3')).toBeInTheDocument()
- })
- it('sets a custom aria label on single action', async () => {
- const view = await renderActionMenu([
- {
- key: 'delete-foo',
- label: 'Delete Foo',
- ariaLabel: 'Custom Delete Foo',
- icon: 'trash3',
- onClick: (entity?: ObjectLike) => {
- fn(entity?.id)
- },
- },
- ])
- expect(view.getByLabelText('Custom Delete Foo')).toBeInTheDocument()
- await view.rerender({
- actions: [
- {
- key: 'delete-foo',
- label: 'Delete Foo',
- ariaLabel: (entity: ObjectLike) => `label ${entity.id}`,
- icon: 'trash3',
- onClick: (entity?: ObjectLike) => {
- fn(entity?.id)
- },
- },
- ],
- })
- expect(view.getByLabelText('label foo-test-action')).toBeInTheDocument()
- })
- it('supports disabling action menu', async () => {
- const actions = [
- {
- key: 'delete-foo',
- label: 'Delete Foo',
- icon: 'trash3',
- show: () => true,
- onClick: (entity?: ObjectLike) => {
- fn(entity?.id)
- },
- },
- {
- key: 'change-foo',
- label: 'Change Foo',
- icon: 'person-gear',
- show: () => true,
- onClick: (entity?: ObjectLike) => {
- fn(entity?.id)
- },
- },
- ]
- // Testing if single action is presented
- const wrapper = await renderActionMenu(actions, {
- disabled: true,
- })
- expect(
- wrapper.getByRole('button', { name: 'Action menu button' }),
- ).toBeDisabled()
- await wrapper.rerender({
- actions: [actions[0]],
- disabled: true,
- })
- expect(wrapper.getByRole('button', { name: 'Delete Foo' })).toBeDisabled()
- })
- })
- })
|