123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142 |
- <!-- Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ -->
- <script setup lang="ts">
- import { computed, toRefs, useSlots } from 'vue'
- import type { CommonPopoverInstance } from '#shared/components/CommonPopover/types.ts'
- import type { ObjectLike } from '#shared/types/utils.ts'
- import { usePopoverMenu } from '#desktop/components/CommonPopoverMenu/usePopoverMenu.ts'
- import CommonPopoverMenuItem from './CommonPopoverMenuItem.vue'
- import type { MenuItem, Variant } from './types'
- export interface Props {
- popover: CommonPopoverInstance | undefined
- headerLabel?: string
- items?: MenuItem[]
- entity?: ObjectLike
- }
- const props = defineProps<Props>()
- const { items, entity } = toRefs(props)
- const { filteredMenuItems } = usePopoverMenu(items, entity)
- const slots = useSlots()
- const showHeaderLabel = computed(() => {
- if (!filteredMenuItems.value && !slots.default) return false
- return slots.header || props.headerLabel
- })
- const onClickItem = (event: MouseEvent, item: MenuItem) => {
- if (item.onClick) {
- event.preventDefault()
- item.onClick(props.entity)
- }
- if (!item.noCloseOnClick) {
- props.popover?.closePopover()
- }
- }
- const getHoverFocusStyles = (variant?: Variant) => {
- if (variant === 'secondary') {
- return 'focus-within:bg-blue-500 hover:bg-blue-500 hover:focus-within:bg-blue-500 dark:focus-within:bg-blue-950 dark:hover:bg-blue-950 dark:hover:focus-within:bg-blue-950'
- }
- if (variant === 'danger') {
- return 'focus-within:bg-pink-100 hover:bg-pink-100 hover:focus-within:bg-pink-100 dark:focus-within:bg-red-900 dark:hover:bg-red-900 dark:hover:focus-within:bg-red-900'
- }
- return 'focus-within:bg-blue-800 focus-within:text-white hover:bg-blue-600 hover:focus-within:bg-blue-800 dark:hover:bg-blue-900 dark:hover:focus-within:bg-blue-800'
- }
- </script>
- <template>
- <section class="min-w-58 flex max-w-64 flex-col gap-0.5">
- <div
- v-if="showHeaderLabel"
- role="heading"
- aria-level="2"
- class="px-2 py-1.5"
- >
- <slot name="header">
- <CommonLabel
- size="small"
- class="line-clamp-1 text-stone-200 dark:text-neutral-500"
- >{{ i18n.t(headerLabel) }}
- </CommonLabel>
- </slot>
- </div>
- <template v-if="filteredMenuItems || $slots.default">
- <slot>
- <ul role="menu" v-bind="$attrs" class="flex w-full flex-col">
- <template v-for="item in filteredMenuItems" :key="item.key">
- <li
- v-if="'array' in item"
- class="flex flex-col overflow-clip pt-2.5 last:rounded-b-[10px] [&:nth-child(n+2)]:border-t [&:nth-child(n+2)]:border-neutral-100 [&:nth-child(n+2)]:dark:border-gray-900"
- role="menuitem"
- >
- <CommonLabel
- size="small"
- class="line-clamp-1 px-2 text-stone-200 dark:text-neutral-500"
- role="heading"
- aria-level="3"
- >{{ item.groupLabel }}</CommonLabel
- >
- <template v-for="subItem in item.array" :key="subItem.key">
- <slot :name="`item-${subItem.key}`" v-bind="subItem">
- <component
- :is="subItem.component || CommonPopoverMenuItem"
- class="flex grow p-2.5"
- :class="getHoverFocusStyles(subItem.variant)"
- :label="subItem.label"
- :variant="subItem.variant"
- :link="subItem.link"
- :icon="subItem.icon"
- :label-placeholder="subItem.labelPlaceholder"
- @click="onClickItem($event, subItem)"
- />
- <slot :name="`itemRight-${subItem.key}`" v-bind="subItem" />
- </slot>
- </template>
- </li>
- <li
- v-else
- role="menuitem"
- class="group flex items-center justify-between last:rounded-b-[10px]"
- :class="[
- {
- 'first:rounded-t-[10px]': !showHeaderLabel,
- 'border-t border-neutral-100 dark:border-gray-900':
- item.separatorTop,
- },
- getHoverFocusStyles(item.variant),
- ]"
- >
- <slot :name="`item-${item.key}`" v-bind="item">
- <component
- :is="item.component || CommonPopoverMenuItem"
- class="flex grow p-2.5"
- :label="item.label"
- :variant="item.variant"
- :link="item.link"
- :icon="item.icon"
- :label-placeholder="item.labelPlaceholder"
- @click="onClickItem($event, item)"
- />
- <slot :name="`itemRight-${item.key}`" v-bind="item" />
- </slot>
- </li>
- </template>
- </ul>
- </slot>
- </template>
- </section>
- </template>
|