123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313 |
- <!-- Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ -->
- <script setup lang="ts">
- import {
- useWindowSize,
- useLocalStorage,
- useScroll,
- onKeyUp,
- useActiveElement,
- } from '@vueuse/core'
- import {
- computed,
- nextTick,
- useTemplateRef,
- onMounted,
- type Ref,
- ref,
- shallowRef,
- watch,
- } from 'vue'
- import stopEvent from '#shared/utils/events.ts'
- import { getFirstFocusableElement } from '#shared/utils/getFocusableElements.ts'
- import CommonButton from '#desktop/components/CommonButton/CommonButton.vue'
- import CommonOverlayContainer from '#desktop/components/CommonOverlayContainer/CommonOverlayContainer.vue'
- import ResizeLine from '#desktop/components/ResizeLine/ResizeLine.vue'
- import { useResizeLine } from '#desktop/components/ResizeLine/useResizeLine.ts'
- import CommonFlyoutActionFooter from './CommonFlyoutActionFooter.vue'
- import { closeFlyout } from './useFlyout.ts'
- import type { ActionFooterOptions, FlyoutSizes } from './types.ts'
- export interface Props {
- /**
- * Unique name which gets used to identify the flyout
- * @example 'crop-avatar'
- */
- name: string
- /**
- * If true, the given flyout resizable width will be stored in local storage
- * Stored under the key `flyout-${name}-width`
- * @example 'crop-avatar' => 'flyout-crop-avatar-width'
- */
- persistResizeWidth?: boolean
- headerTitle?: string
- size?: FlyoutSizes
- headerIcon?: string
- resizable?: boolean
- showBackdrop?: boolean
- noCloseOnBackdropClick?: boolean
- noCloseOnEscape?: boolean
- hideFooter?: boolean
- footerActionOptions?: ActionFooterOptions
- noCloseOnAction?: boolean
- /**
- * Don't focus the first element inside a Flyout after being mounted
- * if nothing is focusable, will focus "Close" button when dismissible is active.
- */
- noAutofocus?: boolean
- }
- const props = withDefaults(defineProps<Props>(), {
- resizable: true,
- showBackdrop: true,
- })
- defineOptions({
- inheritAttrs: false,
- })
- const emit = defineEmits<{
- action: []
- close: [boolean?]
- }>()
- const close = async (isCancel?: boolean) => {
- emit('close', isCancel)
- await closeFlyout(props.name)
- }
- // TODO: maybe we could add a better handling in combination with a form....
- const action = async () => {
- emit('action')
- if (props.noCloseOnAction) return
- await closeFlyout(props.name)
- }
- const flyoutId = `flyout-${props.name}`
- const flyoutSize = { medium: 500 }
- // Width control over flyout
- let flyoutContainerWidth: Ref<number>
- const gap = 16 // Gap between sidebar and flyout
- const storageKeys = Object.keys(localStorage).filter((key) =>
- key.includes('sidebar-width'),
- )
- const leftSideBarKey = storageKeys.find((key) => key.includes('left'))
- const leftSidebarWidth = leftSideBarKey
- ? useLocalStorage(leftSideBarKey, 0)
- : shallowRef(0)
- const { width: screenWidth } = useWindowSize()
- // Calculate the viewport width minus the left sidebar width and a threshold gap
- const flyoutMaxWidth = computed(
- () => screenWidth.value - leftSidebarWidth.value - gap,
- )
- if (props.persistResizeWidth) {
- flyoutContainerWidth = useLocalStorage(
- `${flyoutId}-width`,
- flyoutSize[props.size || 'medium'],
- )
- } else {
- flyoutContainerWidth = ref(flyoutSize[props.size || 'medium'])
- }
- const resizeHandleInstance = useTemplateRef('resize-handle')
- const resizeCallback = (valueX: number) => {
- if (valueX >= flyoutMaxWidth.value) return
- flyoutContainerWidth.value = valueX
- }
- // a11y keyboard navigation
- const activeElement = useActiveElement()
- const handleKeyStroke = (e: KeyboardEvent, adjustment: number) => {
- if (
- !flyoutContainerWidth.value ||
- activeElement.value !== resizeHandleInstance.value?.resizeLine
- )
- return
- e.preventDefault()
- const newWidth = flyoutContainerWidth.value + adjustment
- if (newWidth >= flyoutMaxWidth.value) return
- resizeCallback(newWidth)
- }
- const { startResizing, isResizing } = useResizeLine(
- resizeCallback,
- resizeHandleInstance.value?.resizeLine,
- handleKeyStroke,
- {
- calculateFromRight: true,
- orientation: 'vertical',
- },
- )
- const resetWidth = () => {
- flyoutContainerWidth.value = flyoutSize[props.size || 'medium']
- }
- onMounted(async () => {
- // Prevent left sidebar to collapse with flyout
- await nextTick()
- if (!leftSideBarKey) return
- const leftSidebarWidth = useLocalStorage(leftSideBarKey, 500)
- watch(leftSidebarWidth, (newWidth, oldValue) => {
- if (newWidth + gap < screenWidth.value - flyoutContainerWidth.value) return
- resizeCallback(flyoutContainerWidth.value - (newWidth - oldValue))
- })
- })
- // Keyboard
- onKeyUp('Escape', (e) => {
- if (props.noCloseOnEscape) return
- stopEvent(e)
- close()
- })
- // Style
- const contentElement = useTemplateRef('content')
- const headerElement = useTemplateRef('header')
- const footerElement = useTemplateRef('footer')
- const { arrivedState } = useScroll(contentElement)
- const isContentOverflowing = ref(false)
- watch(
- flyoutContainerWidth,
- async () => {
- // Watch if panel gets resized to show and hide styling based on content overflow
- await nextTick()
- if (
- contentElement.value?.scrollHeight &&
- contentElement.value?.clientHeight
- ) {
- isContentOverflowing.value =
- contentElement.value.scrollHeight > contentElement.value.clientHeight
- }
- },
- { immediate: true },
- )
- // Focus
- onMounted(() => {
- if (props.noAutofocus) return
- const firstFocusableNode =
- getFirstFocusableElement(contentElement.value) ||
- getFirstFocusableElement(footerElement.value) ||
- getFirstFocusableElement(headerElement.value)
- nextTick(() => {
- firstFocusableNode?.focus()
- firstFocusableNode?.scrollIntoView({ block: 'nearest' })
- })
- })
- </script>
- <template>
- <CommonOverlayContainer
- :id="flyoutId"
- tag="aside"
- tabindex="-1"
- class="overflow-clip-x fixed bottom-0 top-0 z-40 flex max-h-dvh min-w-min flex-col border-y border-neutral-100 bg-neutral-50 ltr:right-0 ltr:rounded-l-xl ltr:border-l rtl:left-0 rtl:rounded-r-xl rtl:border-r dark:border-gray-900 dark:bg-gray-500"
- :no-close-on-backdrop-click="noCloseOnBackdropClick"
- :show-backdrop="showBackdrop"
- :style="{ width: `${flyoutContainerWidth}px` }"
- :class="{ 'transition-all': !isResizing }"
- :aria-label="$t('Side panel')"
- :aria-labelledby="`${flyoutId}-title`"
- @click-background="close()"
- >
- <header
- ref="header"
- class="sticky top-0 flex items-center border-b border-neutral-100 border-b-transparent bg-neutral-50 p-3 ltr:rounded-tl-xl rtl:rounded-tr-xl dark:bg-gray-500"
- :class="{
- 'border-b-neutral-100 dark:border-b-gray-900':
- !arrivedState.top && isContentOverflowing,
- }"
- >
- <slot name="header">
- <CommonLabel
- v-if="headerTitle"
- :id="`${flyoutId}-title`"
- tag="h2"
- class="min-h-7 grow"
- size="large"
- :prefix-icon="headerIcon"
- icon-color="text-stone-200 dark:text-neutral-500"
- >
- {{ $t(headerTitle) }}
- </CommonLabel>
- </slot>
- <CommonButton
- class="ltr:ml-auto rtl:mr-auto"
- variant="neutral"
- size="medium"
- :aria-label="$t('Close side panel')"
- icon="x-lg"
- @click="close()"
- />
- </header>
- <div ref="content" class="h-full overflow-y-scroll px-3" v-bind="$attrs">
- <slot />
- </div>
- <footer
- v-if="$slots.footer || !hideFooter"
- ref="footer"
- :aria-label="$t('Side panel footer')"
- class="sticky bottom-0 border-t border-t-transparent bg-neutral-50 p-3 ltr:rounded-bl-xl rtl:rounded-br-xl dark:bg-gray-500"
- :class="{
- 'border-t-neutral-100 dark:border-t-gray-900':
- !arrivedState.bottom && isContentOverflowing,
- }"
- >
- <slot name="footer" v-bind="{ action, close }">
- <CommonFlyoutActionFooter
- v-bind="footerActionOptions"
- @cancel="close(true)"
- @action="action()"
- />
- </slot>
- </footer>
- <ResizeLine
- v-if="resizable"
- ref="resize-handle"
- :label="$t('Resize side panel')"
- class="absolute top-2 h-[calc(100%-16px)] overflow-clip ltr:left-px ltr:-translate-x-1/2 rtl:right-px rtl:translate-x-1/2"
- orientation="vertical"
- :values="{
- current: flyoutContainerWidth,
- max: flyoutMaxWidth,
- }"
- @mousedown-event="startResizing"
- @touchstart-event="startResizing"
- @dblclick-event="resetWidth()"
- />
- </CommonOverlayContainer>
- </template>
|