123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342 |
- <!-- Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ -->
- <script setup lang="ts">
- import { useActiveElement, useLocalStorage, useWindowSize } from '@vueuse/core'
- import { computed, nextTick, ref, watch, type MaybeRef } from 'vue'
- import type { TicketById } from '#shared/entities/ticket/types'
- import type { AppSpecificTicketArticleType } from '#shared/entities/ticket-article/action/plugins/types.ts'
- import { useSessionStore } from '#shared/stores/session.ts'
- import type { ButtonVariant } from '#shared/types/button.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 { useElementScroll } from '#desktop/composables/useElementScroll.ts'
- interface Props {
- ticket: TicketById
- newArticlePresent?: boolean
- createArticleType?: string | null
- ticketArticleTypes: AppSpecificTicketArticleType[]
- isTicketCustomer?: boolean
- hasInternalArticle?: boolean
- parentReachedBottomScroll: boolean
- }
- const props = defineProps<Props>()
- const emit = defineEmits<{
- 'show-article-form': [
- articleType: string,
- performReply: AppSpecificTicketArticleType['performReply'],
- ]
- 'discard-form': []
- }>()
- const currentTicketArticleType = computed(() => {
- if (props.isTicketCustomer) return 'web'
- if (props.createArticleType === 'phone') return 'email'
- return props.createArticleType
- })
- const allowedArticleTypes = computed(() => {
- return ['note', 'phone', currentTicketArticleType.value]
- })
- const availableArticleTypes = computed(() => {
- const availableArticleTypes = props.ticketArticleTypes.filter((type) =>
- allowedArticleTypes.value.includes(type.value),
- )
- const hasEmail = availableArticleTypes.some((type) => type.value === 'email')
- let primaryTicketArticleType = currentTicketArticleType.value
- if (availableArticleTypes.length === 2) {
- primaryTicketArticleType = props.createArticleType
- }
- return availableArticleTypes.map((type) => {
- return {
- articleType: type.value,
- label:
- primaryTicketArticleType === type.value && hasEmail
- ? __('Add reply')
- : type.buttonLabel,
- icon: type.icon,
- variant:
- primaryTicketArticleType === type.value ||
- (type.value === 'phone' &&
- !hasEmail &&
- availableArticleTypes.length === 2)
- ? 'primary'
- : 'secondary',
- performReply: (() =>
- type.performReply?.(
- props.ticket,
- )) as AppSpecificTicketArticleType['performReply'],
- }
- })
- })
- const { userId } = useSessionStore()
- const pinned = useLocalStorage(`${userId}-article-reply-pinned`, false)
- const togglePinned = () => {
- pinned.value = !pinned.value
- }
- const articlePanel = ref<HTMLElement>()
- // Scroll the new article panel into view whenever:
- // - an article is being added
- // - the panel is being unpinned
- watch(
- () => [props.newArticlePresent, pinned.value],
- ([newArticlePresent, pinned]) => {
- if (!newArticlePresent || pinned) return
- nextTick(() => {
- // NB: Give editor a chance to initialize its height.
- setTimeout(() => {
- articlePanel.value?.scrollIntoView?.(true)
- }, 300)
- })
- },
- )
- const DEFAULT_ARTICLE_PANEL_HEIGHT = 290
- const MINIMUM_ARTICLE_PANEL_HEIGHT = 150
- const articlePanelHeight = useLocalStorage(
- `${userId}-article-reply-height`,
- DEFAULT_ARTICLE_PANEL_HEIGHT,
- )
- const { height: screenHeight } = useWindowSize()
- const articlePanelMaxHeight = computed(() => screenHeight.value / 2)
- const resizeLine = ref<InstanceType<typeof ResizeLine>>()
- const resizeCallback = (valueY: number) => {
- if (
- valueY >= articlePanelMaxHeight.value ||
- valueY < MINIMUM_ARTICLE_PANEL_HEIGHT
- )
- return
- articlePanelHeight.value = valueY
- }
- // a11y keyboard navigation
- const activeElement = useActiveElement()
- const handleKeyStroke = (e: KeyboardEvent, adjustment: number) => {
- if (
- !articlePanelHeight.value ||
- activeElement.value !== resizeLine.value?.resizeLine
- )
- return
- e.preventDefault()
- const newHeight = articlePanelHeight.value + adjustment
- if (newHeight >= articlePanelMaxHeight.value) return
- resizeCallback(newHeight)
- }
- const { startResizing } = useResizeLine(
- resizeCallback,
- resizeLine.value?.resizeLine,
- handleKeyStroke,
- { orientation: 'horizontal', offsetThreshold: 56 }, // bottom bar height in px
- )
- const resetHeight = () => {
- articlePanelHeight.value = DEFAULT_ARTICLE_PANEL_HEIGHT
- }
- const articleForm = ref<HTMLElement>()
- const { reachedTop: articleFormReachedTop } = useElementScroll(
- articleForm as MaybeRef<HTMLElement>,
- )
- defineExpose({
- articlePanel,
- })
- </script>
- <template>
- <div
- v-if="newArticlePresent"
- ref="articlePanel"
- class="mx-auto w-full"
- :class="{
- 'max-w-6xl px-12 py-4': !pinned,
- 'sticky bottom-0 z-20 border-t border-t-neutral-300 bg-neutral-50 dark:border-t-gray-900 dark:bg-gray-500':
- pinned,
- }"
- aria-labelledby="article-reply-form-title"
- role="complementary"
- :aria-expanded="!pinned"
- >
- <div
- :class="{
- 'bg-stripes relative z-0 rounded-xl outline outline-1 outline-blue-700 before:rounded-2xl':
- hasInternalArticle && !pinned,
- 'border-stripes': hasInternalArticle && pinned,
- }"
- :style="{
- height: pinned ? `${articlePanelHeight}px` : 'auto',
- }"
- >
- <ResizeLine
- v-if="pinned"
- ref="resizeLine"
- class="group absolute h-3 w-full -translate-y-1.5 py-1"
- :label="$t('Resize article panel')"
- orientation="horizontal"
- :values="{
- max: articlePanelMaxHeight,
- min: MINIMUM_ARTICLE_PANEL_HEIGHT,
- current: articlePanelHeight,
- }"
- @mousedown-event="startResizing"
- @touchstart-event="startResizing"
- @dblclick="resetHeight"
- />
- <div
- class="flex h-full flex-col"
- :class="{
- 'rounded-xl border border-neutral-300 bg-neutral-50 dark:border-gray-900 dark:bg-gray-500':
- !pinned,
- }"
- >
- <div
- class="flex h-10 items-center p-3"
- :class="{
- 'bg-neutral-50 dark:bg-gray-500': pinned,
- 'border-b border-b-transparent': pinned && articleFormReachedTop,
- 'border-b border-b-neutral-300 dark:border-b-gray-900':
- pinned && !articleFormReachedTop,
- }"
- >
- <CommonLabel
- id="article-reply-form-title"
- class="text-stone-200 ltr:mr-auto rtl:ml-auto dark:text-neutral-500"
- tag="h2"
- size="small"
- >
- {{ $t('Reply') }}
- </CommonLabel>
- <CommonButton
- v-tooltip="$t('Discard unsaved reply')"
- class="text-red-500 ltr:mr-2 rtl:ml-2"
- variant="none"
- icon="trash"
- @click="$emit('discard-form')"
- />
- <CommonButton
- v-tooltip="pinned ? $t('Unpin this panel') : $t('Pin this panel')"
- :icon="pinned ? 'pin' : 'pin-angle'"
- variant="neutral"
- size="small"
- @click="togglePinned"
- />
- </div>
- <div
- id="ticketArticleReplyForm"
- ref="articleForm"
- class="h-full px-3 pb-3"
- :class="{
- 'overflow-y-auto': pinned,
- 'my-[5px] px-4 pt-2': hasInternalArticle && pinned,
- }"
- ></div>
- </div>
- </div>
- </div>
- <div
- v-else-if="newArticlePresent !== undefined"
- class="-:border-t-transparent sticky bottom-0 z-20 flex w-full justify-center gap-2.5 border-t py-1.5"
- :class="{
- 'border-t-neutral-100 bg-neutral-50 dark:border-t-gray-900 dark:bg-gray-500':
- !parentReachedBottomScroll,
- }"
- >
- <CommonButton
- v-for="button in availableArticleTypes"
- :key="button.articleType"
- :prefix-icon="button.icon"
- :variant="button.variant as ButtonVariant"
- size="large"
- @click="
- emit('show-article-form', button.articleType, button.performReply)
- "
- >
- {{ $t(button.label) }}
- </CommonButton>
- </div>
- </template>
- <style scoped>
- .border-stripes {
- position: relative;
- background-color: theme('colors.neutral.50');
- &::before {
- content: '';
- position: absolute;
- left: 0;
- top: 40px;
- bottom: 0;
- right: 0;
- border: 5px solid transparent;
- background-image: repeating-linear-gradient(
- 45deg,
- theme('colors.blue.400'),
- theme('colors.blue.400') 5px,
- theme('colors.blue.700') 5px,
- theme('colors.blue.700') 10px
- );
- background-position: -1px;
- background-attachment: fixed;
- mask:
- linear-gradient(#fff 0 0) padding-box,
- linear-gradient(#fff 0 0);
- mask-composite: exclude;
- }
- &::after {
- content: '';
- position: absolute;
- left: 0;
- top: 40px;
- bottom: 0;
- right: 0;
- outline: 1px solid theme('colors.blue.700');
- outline-offset: -5px;
- pointer-events: none;
- }
- }
- [data-theme='dark'] .border-stripes {
- background-color: theme('colors.gray.500');
- &::before {
- background-image: repeating-linear-gradient(
- 45deg,
- theme('colors.blue.700'),
- theme('colors.blue.700') 5px,
- theme('colors.blue.900') 5px,
- theme('colors.blue.900') 10px
- );
- }
- }
- </style>
|