123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167 |
- <!-- Copyright (C) 2012-2024 Zammad Foundation, https://zammad-foundation.org/ -->
- <script setup lang="ts">
- import { useActiveElement } from '@vueuse/core'
- import { computed, ref, watch } from 'vue'
- import CollapseButton from '#desktop/components/CollapseButton/CollapseButton.vue'
- import { useCollapseHandler } from '#desktop/components/CollapseButton/composables/useCollapseHandler.ts'
- import CommonButton from '#desktop/components/CommonButton/CommonButton.vue'
- import { useResizeWidthHandle } from '#desktop/components/ResizeHandle/composables/useResizeWidthHandle.ts'
- import ResizeHandle from '#desktop/components/ResizeHandle/ResizeHandle.vue'
- import { SidebarPosition } from './types.ts'
- interface Props {
- name: string
- /**
- @property currentWidth
- @property minWidth
- @property maxWidth
- - used for accessibility
- / */
- currentWidth?: number
- minWidth?: number
- maxWidth?: number
- noScroll?: boolean
- collapsible?: boolean
- iconCollapsed?: string
- hideButtonWhenCollapsed?: boolean // Adjust it used as a default for all sidebars
- position?: SidebarPosition
- resizable?: boolean
- id: string
- noPadding?: 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 { toggleCollapse, isCollapsed } = useCollapseHandler(emit, {
- storageKey: `${props.name}-sidebar-collapsed`,
- })
- const showCollapseButtonOnCollapse = computed(() => {
- return props.hideButtonWhenCollapsed ? !isCollapsed.value : true
- })
- // a11y keyboard navigation
- const resizeHandleComponent = ref<InstanceType<typeof ResizeHandle>>()
- const activeElement = useActiveElement()
- const handleKeyStroke = (e: KeyboardEvent, adjustment: number) => {
- if (
- !props.currentWidth ||
- activeElement.value !== resizeHandleComponent.value?.$el
- )
- return
- e.preventDefault()
- const newWidth = props.currentWidth + adjustment
- if (
- props.maxWidth &&
- props.minWidth &&
- (newWidth >= props.maxWidth || newWidth <= props.minWidth)
- )
- return
- emit('resize-horizontal', newWidth)
- }
- const { startResizing, isResizingHorizontal } = useResizeWidthHandle(
- (positionX) => emit('resize-horizontal', positionX),
- resizeHandleComponent,
- handleKeyStroke,
- {
- calculateFromRight: props.position === SidebarPosition.End,
- },
- )
- watch(isResizingHorizontal, (isResizing) => {
- if (isResizing) {
- emit('resize-horizontal-start')
- } else {
- emit('resize-horizontal-end')
- }
- })
- </script>
- <template>
- <aside
- :id="props.id"
- class="group/sidebar -:bg-neutral-950 -:max-h-screen relative flex flex-col border-neutral-100 dark:border-gray-900"
- :class="{
- 'py-3': isCollapsed && !noPadding,
- 'border-e': position === SidebarPosition.Start,
- 'border-s': position === SidebarPosition.End,
- }"
- >
- <CollapseButton
- v-if="collapsible && showCollapseButtonOnCollapse"
- :is-collapsed="isCollapsed"
- :owner-id="id"
- group="sidebar"
- class="absolute top-[49px] z-20"
- :inverse="position === SidebarPosition.End"
- :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,
- }"
- @toggle-collapse="toggleCollapse"
- />
- <ResizeHandle
- v-if="resizable && !isCollapsed"
- ref="resizeHandleComponent"
- class="absolute top-1/2 -translate-y-1/2"
- :class="{
- 'ltr:right-0 rtl:left-0': position === SidebarPosition.Start,
- 'ltr:left-0 rtl:right-0': position === SidebarPosition.End,
- }"
- :aria-label="$t('Resize sidebar')"
- :aria-valuenow="currentWidth"
- :aria-valuemax="maxWidth?.toFixed(2)"
- :aria-valuemin="minWidth"
- role="separator"
- aria-orientation="horizontal"
- tabindex="0"
- @mousedown="startResizing"
- @touchstart="startResizing"
- @dblclick="$emit('reset-width')"
- />
- <CommonButton
- v-if="iconCollapsed && isCollapsed"
- class="mx-auto"
- size="medium"
- data-test-id="action-button"
- variant="neutral"
- :icon="iconCollapsed"
- @click="toggleCollapse"
- />
- <div
- v-else
- class="flex h-full flex-col"
- :class="{
- 'p-3': !isCollapsed && !noPadding,
- 'overflow-y-hidden': noScroll,
- 'overflow-y-auto': !noScroll,
- }"
- >
- <slot v-bind="{ isCollapsed, toggleCollapse }" />
- </div>
- </aside>
- </template>
|