123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154 |
- <!-- Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ -->
- <script lang="ts" setup>
- import { useEventListener } from '@vueuse/core'
- import { computed, readonly, ref } from 'vue'
- import getUuid from '#shared/utils/getUuid.ts'
- interface Props {
- label: string
- /**
- * horizontal line or vertical line
- * */
- orientation?: 'horizontal' | 'vertical'
- values?: {
- /**
- * Maximum width/height in px value of what the container can be resized to
- * */
- max?: number | string
- /**
- * Minimum width/height in px value of what the container can be resized to
- * */
- min?: number | string
- /**
- * Current width/height in px value of the container
- * */
- current?: number | string
- }
- disabled?: boolean
- buttonClass?: string
- }
- const props = withDefaults(defineProps<Props>(), {
- orientation: 'vertical',
- })
- const resizeLine = ref<HTMLDivElement>()
- const resizing = ref(false)
- const emit = defineEmits<{
- 'mousedown-event': [MouseEvent]
- 'touchstart-event': [TouchEvent]
- 'dblclick-event': [MouseEvent]
- }>()
- const resizeOrientation = computed(() =>
- // Vertical resize line should have horizontal aria-orientation -> container width
- // Horizontal resize line should have vertical aria-orientation -> container height
- props.orientation === 'horizontal' ? 'vertical' : 'horizontal',
- )
- const addRemoveResizingListener = (event: 'mouseup' | 'touchend') => {
- useEventListener(
- event,
- () => {
- resizing.value = false
- },
- { once: true },
- )
- }
- const handleMousedown = (event: MouseEvent) => {
- if (props.disabled) return
- emit('mousedown-event', event)
- resizing.value = true
- addRemoveResizingListener('mouseup')
- }
- const handleTouchstart = (event: TouchEvent) => {
- if (props.disabled) return
- emit('touchstart-event', event)
- resizing.value = true
- addRemoveResizingListener('touchend')
- }
- const handleDoubleClick = (event: MouseEvent) => {
- if (props.disabled) return
- emit('dblclick-event', event)
- resizeLine.value?.blur()
- }
- const id = getUuid()
- defineExpose({
- resizeLine,
- resizing: readonly(resizing),
- })
- </script>
- <template>
- <div class="hover-area" :class="[`hover-area--${props.orientation}`]">
- <button
- ref="resizeLine"
- v-tooltip="!disabled ? label : undefined"
- :aria-describedby="id"
- :disabled="disabled"
- tabindex="0"
- class="line"
- :class="[{ '!bg-blue-800': resizing }, buttonClass]"
- @mousedown="handleMousedown"
- @blur="resizing = false"
- @touchstart="handleTouchstart"
- @dblclick="handleDoubleClick"
- />
- <span
- v-if="!disabled"
- :id="id"
- role="separator"
- class="invisible absolute -z-20"
- :aria-orientation="resizeOrientation"
- :aria-valuenow="values?.current ?? undefined"
- :aria-valuemax="values?.max ?? undefined"
- :aria-valuemin="values?.min ?? undefined"
- />
- </div>
- </template>
- <style scoped>
- .line {
- @apply focus:outline-none;
- &:focus-within {
- background-color: theme('colors.blue.800') !important;
- }
- }
- .hover-area {
- @apply flex justify-center opacity-0 focus-within:opacity-100 hover:opacity-100;
- .line:not(:disabled) {
- @apply bg-neutral-500 hover:bg-blue-600 dark:bg-gray-200 dark:hover:bg-blue-900;
- }
- &--horizontal {
- @apply -:w-full h-[12px];
- .line {
- @apply h-1 w-full enabled:cursor-row-resize;
- }
- }
- &--vertical {
- @apply -:h-full w-[12px];
- .line {
- @apply h-full w-1 enabled:cursor-col-resize;
- }
- }
- }
- </style>
|