123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260 |
- <!-- Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ -->
- <script setup lang="ts">
- import { onClickOutside, onKeyDown, useVModel } from '@vueuse/core'
- import { useTemplateRef, onUnmounted, computed, nextTick, ref } from 'vue'
- import type { SelectOption } from '#shared/components/CommonSelect/types.ts'
- import { useFocusWhenTyping } from '#shared/composables/useFocusWhenTyping.ts'
- import { useTrapTab } from '#shared/composables/useTrapTab.ts'
- import { useTraverseOptions } from '#shared/composables/useTraverseOptions.ts'
- import stopEvent from '#shared/utils/events.ts'
- import testFlags from '#shared/utils/testFlags.ts'
- import CommonSelectItem from './CommonSelectItem.vue'
- import { useCommonSelect } from './useCommonSelect.ts'
- import type { CommonSelectInternalInstance } from './types.ts'
- import type { Ref } from 'vue'
- export interface Props {
- // we cannot move types into separate file, because Vue would not be able to
- // transform these into runtime types
- modelValue?: string | number | boolean | (string | number | boolean)[] | null
- options: SelectOption[]
- /**
- * Do not modify local value
- */
- passive?: boolean
- multiple?: boolean
- noClose?: boolean
- noRefocus?: boolean
- owner?: string
- noOptionsLabelTranslation?: boolean
- }
- const props = defineProps<Props>()
- const emit = defineEmits<{
- 'update:modelValue': [option: string | number | (string | number)[]]
- select: [option: SelectOption]
- close: []
- }>()
- const dialogElement = useTemplateRef('dialog')
- const localValue = useVModel(props, 'modelValue', emit)
- // TODO: do we really want this initial transfomring of the value, when it's null?
- if (localValue.value == null && props.multiple) {
- localValue.value = []
- }
- const getFocusableOptions = () => {
- return Array.from<HTMLElement>(
- dialogElement.value?.querySelectorAll('[tabindex="0"]') || [],
- )
- }
- const showDialog = ref(false)
- let lastFocusableOutsideElement: HTMLElement | null = null
- const getActiveElement = () => {
- if (props.owner) {
- return document.getElementById(props.owner)
- }
- return document.activeElement as HTMLElement
- }
- const openDialog = () => {
- showDialog.value = true
- lastFocusableOutsideElement = getActiveElement()
- requestAnimationFrame(() => {
- // https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/Roles/listbox_role#keyboard_interactions
- // focus selected or first available option
- const focusableElements = getFocusableOptions()
- const selected = focusableElements.find(
- (el) => el.getAttribute('aria-selected') === 'true',
- )
- const focusElement = selected || focusableElements[0]
- focusElement?.focus()
- nextTick(() => {
- testFlags.set('common-select.opened')
- })
- })
- }
- const closeDialog = () => {
- showDialog.value = false
- emit('close')
- if (!props.noRefocus) {
- nextTick(() => lastFocusableOutsideElement?.focus())
- }
- nextTick(() => {
- testFlags.set('common-select.closed')
- })
- }
- const exposedInstance: CommonSelectInternalInstance = {
- isOpen: computed(() => showDialog.value),
- openDialog,
- closeDialog,
- getFocusableOptions,
- }
- const { instances } = useCommonSelect()
- instances.value.add(exposedInstance)
- onUnmounted(() => {
- instances.value.delete(exposedInstance)
- })
- defineExpose(exposedInstance)
- // https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/Roles/listbox_role#keyboard_interactions
- useTraverseOptions(dialogElement, { direction: 'vertical' })
- // - Type-ahead is recommended for all listboxes, especially those with more than seven options
- useFocusWhenTyping(dialogElement)
- onClickOutside(dialogElement, closeDialog)
- useTrapTab(dialogElement)
- onKeyDown(
- 'Escape',
- (event) => {
- stopEvent(event)
- closeDialog()
- },
- { target: dialogElement as Ref<EventTarget> },
- )
- const isCurrentValue = (value: string | number | boolean) => {
- if (props.multiple && Array.isArray(localValue.value)) {
- return localValue.value.includes(value)
- }
- return localValue.value === value
- }
- const select = (option: SelectOption) => {
- if (option.disabled) return
- emit('select', option)
- if (props.passive) {
- if (!props.multiple) {
- closeDialog()
- }
- return
- }
- if (props.multiple && Array.isArray(localValue.value)) {
- if (localValue.value.includes(option.value)) {
- localValue.value = localValue.value.filter((v) => v !== option.value)
- } else {
- localValue.value.push(option.value)
- }
- return
- }
- if (props.modelValue === option.value) {
- localValue.value = undefined
- } else {
- localValue.value = option.value
- }
- if (!props.multiple && !props.noClose) {
- closeDialog()
- }
- }
- const duration = VITE_TEST_MODE ? undefined : { enter: 300, leave: 200 }
- </script>
- <template>
- <slot :state="showDialog" :open="openDialog" :close="closeDialog" />
- <Teleport to="body">
- <Transition :duration="duration">
- <div
- v-if="showDialog"
- id="common-select"
- class="fixed inset-0 z-10 flex overflow-y-auto"
- :aria-label="$t('Dialog window with selections')"
- role="dialog"
- >
- <!-- empty @click is needed for https://stackoverflow.com/a/39712411 -->
- <div
- class="select-overlay fixed inset-0 flex h-full w-full bg-gray-500 opacity-60"
- data-test-id="dialog-overlay"
- role="presentation"
- @click="void 0"
- ></div>
- <div class="select-dialog relative m-auto">
- <div
- class="flex min-w-[294px] max-w-[90vw] flex-col items-start rounded-xl bg-gray-400/80 backdrop-blur-[15px]"
- >
- <div
- ref="dialog"
- :aria-label="$t('Select…')"
- role="listbox"
- :aria-multiselectable="multiple"
- class="max-h-[50vh] w-full divide-y divide-solid divide-white/10 overflow-y-auto"
- >
- <CommonSelectItem
- v-for="option in options"
- :key="String(option.value)"
- :selected="isCurrentValue(option.value)"
- :multiple="multiple"
- :option="option"
- :no-label-translate="noOptionsLabelTranslation"
- @select="select($event)"
- />
- <slot name="footer" />
- </div>
- </div>
- </div>
- </div>
- </Transition>
- </Teleport>
- </template>
- <style scoped>
- .v-enter-active {
- .select-overlay,
- .select-dialog {
- @apply duration-300 ease-out;
- }
- }
- .v-leave-active {
- .select-overlay,
- .select-dialog {
- @apply duration-200 ease-in;
- }
- }
- .v-enter-to,
- .v-leave-from {
- .select-dialog {
- @apply scale-100 opacity-100;
- }
- .select-overlay {
- @apply opacity-60;
- }
- }
- .v-enter-from,
- .v-leave-to {
- .select-dialog {
- @apply scale-95 opacity-0;
- }
- .select-overlay {
- @apply opacity-0;
- }
- }
- </style>
|