123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198 |
- <!-- Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ -->
- <script setup lang="ts">
- import { useActiveElement } from '@vueuse/core'
- import { computed, useTemplateRef, watch } from 'vue'
- import CollapseButton from '#desktop/components/CollapseButton/CollapseButton.vue'
- import type { CollapseOptions } from '#desktop/components/CollapseButton/types.ts'
- import { useCollapseHandler } from '#desktop/components/CollapseButton/useCollapseHandler.ts'
- import CommonButton from '#desktop/components/CommonButton/CommonButton.vue'
- import ResizeLine from '#desktop/components/ResizeLine/ResizeLine.vue'
- import { useResizeLine } from '#desktop/components/ResizeLine/useResizeLine.ts'
- import { SidebarPosition } from './types.ts'
- interface Props {
- name: string
- /**
- @property currentWidth
- @property minWidth
- @property maxWidth
- - used for accessibility
- / */
- currentWidth?: number | string
- minWidth?: number | string
- maxWidth?: number | string
- noScroll?: boolean
- collapsible?: boolean
- iconCollapsed?: string
- position?: SidebarPosition
- resizable?: boolean
- id: string
- noPadding?: boolean
- classes?: {
- resizeLine?: string
- collapseButton?: string
- }
- rememberCollapse?: boolean
- }
- const props = withDefaults(defineProps<Props>(), {
- position: SidebarPosition.Start,
- hideButtonWhenCollapsed: false,
- })
- const emit = defineEmits<{
- 'resize-horizontal': [number]
- 'resize-horizontal-start': []
- 'resize-horizontal-end': []
- 'reset-width': []
- collapse: [boolean]
- expand: [boolean]
- }>()
- const collapseOptions: CollapseOptions = {
- name: props.name,
- }
- if (props.rememberCollapse)
- collapseOptions.storageKey = `${props.name}-sidebar-collapsed`
- const { toggleCollapse, isCollapsed } = useCollapseHandler(
- emit,
- collapseOptions,
- )
- // a11y keyboard navigation // TS: Does not infer type for some reason?
- const resizeLineInstance =
- useTemplateRef<InstanceType<typeof ResizeLine>>('resize-line')
- const activeElement = useActiveElement()
- const handleKeyStroke = (e: KeyboardEvent, adjustment: number) => {
- if (
- !props.currentWidth ||
- activeElement.value !== resizeLineInstance.value?.resizeLine
- )
- return
- e.preventDefault()
- const newWidth = Number(props.currentWidth) + adjustment
- emit('resize-horizontal', newWidth)
- }
- const { startResizing, isResizing } = useResizeLine(
- (positionX) => emit('resize-horizontal', positionX),
- resizeLineInstance.value?.resizeLine,
- handleKeyStroke,
- {
- calculateFromRight: props.position === SidebarPosition.End,
- orientation: 'vertical',
- },
- )
- watch(isResizing, (isResizing) => {
- if (isResizing) {
- emit('resize-horizontal-start')
- } else {
- emit('resize-horizontal-end')
- }
- })
- const collapseButtonClass = computed(() => {
- if (props.position === SidebarPosition.Start)
- return 'ltr:rounded-l-none rtl:rounded-r-none'
- if (props.position === SidebarPosition.End)
- return 'ltr:rounded-r-none rtl:rounded-l-none'
- return ''
- })
- </script>
- <template>
- <aside
- :id="id"
- class="-:bg-neutral-950 -:max-h-screen relative flex flex-col overflow-y-clip border-neutral-100 dark:border-gray-900"
- :class="{
- 'py-3': isCollapsed && !noPadding,
- 'border-s': position === SidebarPosition.End,
- }"
- >
- <CommonButton
- v-if="iconCollapsed && isCollapsed"
- class="mx-auto"
- size="large"
- data-test-id="action-button"
- variant="neutral"
- :icon="iconCollapsed"
- @click="toggleCollapse"
- />
- <div
- v-else
- class="flex h-full max-w-full flex-col overflow-x-hidden"
- :class="{
- 'px-3 py-2.5': !isCollapsed && !noPadding,
- 'overflow-y-hidden': noScroll,
- 'overflow-y-auto': !noScroll,
- }"
- >
- <slot v-bind="{ isCollapsed, toggleCollapse }" />
- </div>
- <ResizeLine
- v-if="resizable"
- ref="resize-line"
- :label="$t('Resize sidebar')"
- class="absolute z-20 has-[+div:hover]:opacity-100"
- :class="[
- {
- 'ltr:right-0 ltr:translate-x-1/2 rtl:left-0 rtl:-translate-x-1/2':
- position === SidebarPosition.Start,
- 'ltr:left-0 ltr:-translate-x-1/2 rtl:right-0 rtl:translate-x-1/2':
- position === SidebarPosition.End,
- peer: !resizeLineInstance?.resizing,
- },
- classes?.resizeLine || '',
- ]"
- :values="{
- max: Number(maxWidth)?.toFixed(2),
- min: minWidth,
- current: currentWidth,
- }"
- :disabled="isCollapsed"
- @mousedown-event="startResizing"
- @touchstart-event="startResizing"
- @dblclick-event="$emit('reset-width')"
- />
- <CollapseButton
- v-if="collapsible"
- :collapsed="isCollapsed"
- :owner-id="id"
- class="absolute top-[49px] z-20 peer-hover:opacity-100"
- :inverse="position === SidebarPosition.End"
- variant="tertiary-gray"
- :collapse-label="$t('Collapse sidebar')"
- :expand-label="$t('Expand sidebar')"
- :class="[
- {
- 'ltr:right-0 ltr:translate-x-[calc(100%-10px)] rtl:left-0 rtl:-translate-x-[calc(100%-10px)]':
- position === SidebarPosition.Start,
- 'ltr:left-0 ltr:-translate-x-[calc(100%-10px)] rtl:right-0 rtl:translate-x-[calc(100%-10px)]':
- position === SidebarPosition.End,
- 'ltr:translate-x-[calc(100%-7.5px)] rtl:-translate-x-[calc(100%-7.5px)]':
- isCollapsed && position === SidebarPosition.Start,
- 'ltr:-translate-x-[calc(100%-7.5px)] rtl:translate-x-[calc(100%-7.5px)]':
- isCollapsed && position === SidebarPosition.End,
- },
- classes?.collapseButton || '',
- ]"
- :button-class="collapseButtonClass"
- @click="(node: MouseEvent) => (node.target as HTMLButtonElement)?.blur()"
- @toggle-collapse="toggleCollapse"
- />
- </aside>
- </template>
|