123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643 |
- <!-- Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ -->
- <script setup lang="ts">
- import { useLocalStorage } from '@vueuse/core'
- import { cloneDeep } from 'lodash-es'
- import {
- computed,
- toRef,
- provide,
- Teleport,
- markRaw,
- type Component,
- reactive,
- nextTick,
- watch,
- useTemplateRef,
- ref,
- type Ref,
- } from 'vue'
- import {
- NotificationTypes,
- useNotifications,
- } from '#shared/components/CommonNotifications/index.ts'
- import Form from '#shared/components/Form/Form.vue'
- import type {
- FormSubmitData,
- FormValues,
- } from '#shared/components/Form/types.ts'
- import { useForm } from '#shared/components/Form/useForm.ts'
- import { setErrors } from '#shared/components/Form/utils.ts'
- import { useConfirmation } from '#shared/composables/useConfirmation.ts'
- import {
- useTicketMacros,
- macroScreenBehaviourMapping,
- } from '#shared/entities/macro/composables/useMacros.ts'
- import { useTicketArticleReplyAction } from '#shared/entities/ticket/composables/useTicketArticleReplyAction.ts'
- import { useTicketEdit } from '#shared/entities/ticket/composables/useTicketEdit.ts'
- import { useTicketEditForm } from '#shared/entities/ticket/composables/useTicketEditForm.ts'
- import { useTicketLiveUserList } from '#shared/entities/ticket/composables/useTicketLiveUserList.ts'
- import type {
- TicketArticleTimeAccountingFormData,
- TicketUpdateFormData,
- } from '#shared/entities/ticket/types.ts'
- import type { AppSpecificTicketArticleType } from '#shared/entities/ticket-article/action/plugins/types.ts'
- import {
- useArticleDataHandler,
- type AddArticleCallbackArgs,
- } from '#shared/entities/ticket-article/composables/useArticleDataHandler.ts'
- import UserError from '#shared/errors/UserError.ts'
- import {
- EnumTaskbarEntity,
- EnumFormUpdaterId,
- EnumTaskbarApp,
- EnumUserErrorException,
- } from '#shared/graphql/types.ts'
- import { convertToGraphQLId } from '#shared/graphql/utils.ts'
- import { QueryHandler } from '#shared/server/apollo/handler/index.ts'
- import { useSessionStore } from '#shared/stores/session.ts'
- import { useFlyout } from '#desktop/components/CommonFlyout/useFlyout.ts'
- import CommonLoader from '#desktop/components/CommonLoader/CommonLoader.vue'
- import LayoutContent from '#desktop/components/layout/LayoutContent.vue'
- import { useElementScroll } from '#desktop/composables/useElementScroll.ts'
- import { useTaskbarTab } from '#desktop/entities/user/current/composables/useTaskbarTab.ts'
- import { useTaskbarTabStateUpdates } from '#desktop/entities/user/current/composables/useTaskbarTabStateUpdates.ts'
- import TicketDetailBottomBar from '#desktop/pages/ticket/components/TicketDetailView/TicketDetailBottomBar/TicketDetailBottomBar.vue'
- import { useTicketScreenBehavior } from '#desktop/pages/ticket/components/TicketDetailView/TicketScreenBehavior/useTicketScreenBehavior.ts'
- import ArticleList from '../components/TicketDetailView/ArticleList.vue'
- import ArticleReply from '../components/TicketDetailView/ArticleReply.vue'
- import TicketDetailTopBar from '../components/TicketDetailView/TicketDetailTopBar/TicketDetailTopBar.vue'
- import TicketSidebar from '../components/TicketSidebar.vue'
- import { ARTICLES_INFORMATION_KEY } from '../composables/useArticleContext.ts'
- import { useTicketArticleReply } from '../composables/useTicketArticleReply.ts'
- import {
- initializeTicketInformation,
- provideTicketInformation,
- } from '../composables/useTicketInformation.ts'
- import {
- useTicketSidebar,
- useProvideTicketSidebar,
- } from '../composables/useTicketSidebar.ts'
- import {
- type TicketSidebarContext,
- TicketSidebarScreenType,
- } from '../types/sidebar.ts'
- interface Props {
- internalId: string
- }
- const props = defineProps<Props>()
- const {
- activeTaskbarTab,
- activeTaskbarTabFormId,
- activeTaskbarTabNewArticlePresent,
- } = useTaskbarTab(EnumTaskbarEntity.TicketZoom)
- const internalId = toRef(props, 'internalId')
- const { ticket, ticketId, ...ticketInformation } =
- initializeTicketInformation(internalId)
- const onAddArticleCallback = ({ articlesQuery }: AddArticleCallbackArgs) => {
- return (articlesQuery as QueryHandler).refetch()
- }
- const { articleResult, articlesQuery, isLoadingArticles } =
- useArticleDataHandler(ticketId, { pageSize: 20, onAddArticleCallback })
- provide(ARTICLES_INFORMATION_KEY, {
- articles: computed(() => articleResult.value),
- articlesQuery,
- })
- const {
- form,
- values,
- flags,
- isDisabled,
- isDirty,
- isInitialSettled,
- formReset,
- formSubmit,
- triggerFormUpdater,
- } = useForm()
- const groupId = computed(() =>
- isInitialSettled.value && values.value.group_id
- ? convertToGraphQLId('Group', values.value.group_id as number)
- : undefined,
- )
- const { setSkipNextStateUpdate } = useTaskbarTabStateUpdates(triggerFormUpdater)
- const sidebarContext = computed<TicketSidebarContext>(() => ({
- screenType: TicketSidebarScreenType.TicketDetailView,
- form: form.value,
- formValues: {
- // TODO: Workaround, to make the sidebars working for now.
- customer_id: ticket.value?.customer.internalId,
- organization_id: ticket.value?.organization?.internalId,
- },
- }))
- useProvideTicketSidebar(sidebarContext)
- const { hasSidebar, activeSidebar, switchSidebar } = useTicketSidebar()
- const {
- ticketSchema,
- articleSchema,
- currentArticleType,
- ticketArticleTypes,
- securityIntegration,
- isTicketAgent,
- isTicketCustomer,
- isTicketEditable,
- articleTypeHandler,
- articleTypeSelectHandler,
- } = useTicketEditForm(ticket, form)
- const hasInternalArticle = computed(
- () => (values.value as TicketUpdateFormData).article?.internal,
- )
- const formEditAttributeLocation = computed(() => {
- if (activeSidebar.value === 'information') return '#ticketEditAttributeForm'
- return '#wrapper-form-ticket-edit'
- })
- const {
- isArticleFormGroupValid,
- newTicketArticlePresent,
- showTicketArticleReplyForm,
- } = useTicketArticleReply(form, activeTaskbarTabNewArticlePresent)
- const { liveUserList } = useTicketLiveUserList(
- internalId,
- isTicketAgent,
- EnumTaskbarApp.Desktop,
- )
- provideTicketInformation({
- ticket,
- ticketId,
- isTicketEditable,
- form,
- newTicketArticlePresent,
- showTicketArticleReplyForm,
- ...ticketInformation,
- })
- const ticketEditSchemaData = reactive({
- formEditAttributeLocation,
- securityIntegration,
- newTicketArticlePresent,
- currentArticleType,
- })
- const ticketEditSchema = [
- {
- isLayout: true,
- component: 'Teleport',
- props: {
- to: '$formEditAttributeLocation',
- },
- children: [
- {
- isLayout: true,
- component: 'FormGroup',
- props: {
- class: '@container/form-group',
- showDirtyMark: true,
- },
- children: [ticketSchema],
- },
- ],
- },
- {
- if: '$newTicketArticlePresent',
- isLayout: true,
- component: 'Teleport',
- props: {
- to: '#ticketArticleReplyForm',
- },
- children: [
- {
- isLayout: true,
- component: 'FormGroup',
- props: {
- class: '@container/form-group',
- },
- children: [articleSchema],
- },
- ],
- },
- ]
- const { waitForConfirmation, waitForVariantConfirmation } = useConfirmation()
- const { handleScreenBehavior } = useTicketScreenBehavior()
- const canUseDraft = computed(() => {
- return flags.value.hasSharedDraft
- })
- const hasAvailableDraft = computed(() => {
- const sharedDraftZoomId = ticket.value?.sharedDraftZoomId
- if (!sharedDraftZoomId) return false
- return canUseDraft.value
- })
- const discardChanges = async () => {
- const confirm = await waitForVariantConfirmation('unsaved')
- if (confirm) {
- newTicketArticlePresent.value = false
- await nextTick()
- // Skip subscription for the current tab, to avoid not needed form updater requests.
- setSkipNextStateUpdate(true)
- formReset()
- }
- }
- const { isTicketFormGroupValid, initialTicketValue, editTicket } =
- useTicketEdit(ticket, form)
- const { openReplyForm } = useTicketArticleReplyAction(
- form,
- showTicketArticleReplyForm,
- )
- const isFormValid = computed(() => {
- if (!newTicketArticlePresent.value) return isTicketFormGroupValid.value
- return isTicketFormGroupValid.value && isArticleFormGroupValid.value
- })
- const formAdditionalRouteQueryParams = computed(() => ({
- taskbarId: activeTaskbarTab.value?.taskbarTabId,
- }))
- const { notify } = useNotifications()
- const { userId } = useSessionStore()
- const articleReplyPinned = useLocalStorage(
- `${userId}-article-reply-pinned`,
- false,
- )
- const contentContainerElement = useTemplateRef('content-container')
- const topBarInstance = useTemplateRef('top-bar')
- const { isScrollingDown: hideDetails } = useElementScroll(
- contentContainerElement as Ref<HTMLElement>,
- {
- scrollStartThreshold: computed(
- () => topBarInstance.value?.$el.clientHeight,
- ),
- },
- )
- const { reachedBottom } = useElementScroll(
- contentContainerElement as Ref<HTMLElement>,
- )
- const scrollToArticlesEnd = () => {
- nextTick(() => {
- const scrollHeight = contentContainerElement.value?.scrollHeight
- if (scrollHeight)
- contentContainerElement.value?.scrollTo({
- top: scrollHeight,
- })
- })
- }
- const checkSubmitEditTicket = () => {
- if (!isFormValid.value) {
- if (activeSidebar.value !== 'information') switchSidebar('information')
- if (
- newTicketArticlePresent.value &&
- !isArticleFormGroupValid.value &&
- !articleReplyPinned.value
- )
- scrollToArticlesEnd()
- }
- formSubmit()
- }
- const skipValidators = ref<EnumUserErrorException[]>([])
- const handleIncompleteChecklist = async (error: UserError) => {
- const confirmed = await waitForConfirmation(
- __(
- 'You have unchecked items in the checklist. Do you want to handle them before closing this ticket?',
- ),
- {
- headerTitle: __('Incomplete Ticket Checklist'),
- headerIcon: 'checklist',
- buttonLabel: __('Yes, open the checklist'),
- cancelLabel: __('No, just close the ticket'),
- },
- )
- if (confirmed) {
- if (activeSidebar.value !== 'checklist') switchSidebar('checklist')
- return false
- }
- if (confirmed === false) {
- const exception = error.getFirstErrorException()
- if (exception) skipValidators.value?.push(exception)
- formSubmit()
- return true
- }
- return false
- }
- const timeAccountingData = ref<TicketArticleTimeAccountingFormData>()
- const timeAccountingFlyout = useFlyout({
- name: 'ticket-time-accounting',
- component: () =>
- import('../components/TicketDetailView/TimeAccountingFlyout.vue'),
- })
- const handleTimeAccounting = (error: UserError) => {
- timeAccountingFlyout.open({
- onAccountTime: (data: TicketArticleTimeAccountingFormData) => {
- timeAccountingData.value = data
- formSubmit()
- },
- onSkip: () => {
- const exception = error.getFirstErrorException()
- if (exception) skipValidators.value?.push(exception)
- formSubmit()
- },
- })
- return false
- }
- const handleUserErrorException = (error: UserError) => {
- if (
- error.getFirstErrorException() ===
- EnumUserErrorException.ServiceTicketUpdateValidatorChecklistCompletedError
- )
- return handleIncompleteChecklist(error)
- if (
- error.getFirstErrorException() ===
- EnumUserErrorException.ServiceTicketUpdateValidatorTimeAccountingError
- )
- return handleTimeAccounting(error)
- return true
- }
- const { activeMacro, executeMacro, disposeActiveMacro } =
- useTicketMacros(formSubmit)
- const submitEditTicket = async (
- formData: FormSubmitData<TicketUpdateFormData>,
- ) => {
- let data = cloneDeep(formData)
- if (currentArticleType.value?.updateForm)
- data = currentArticleType.value.updateForm(data)
- if (data.article && timeAccountingData.value) {
- data.article = {
- ...data.article,
- timeUnit:
- timeAccountingData.value.time_unit !== undefined
- ? parseFloat(timeAccountingData.value.time_unit)
- : undefined,
- accountedTimeTypeId: timeAccountingData.value.accounted_time_type_id
- ? convertToGraphQLId(
- 'Ticket::TimeAccounting::Type',
- timeAccountingData.value.accounted_time_type_id,
- )
- : undefined,
- }
- }
- return editTicket(data, {
- macroId: activeMacro.value?.id,
- skipValidators: skipValidators.value,
- })
- .then((result) => {
- if (result?.ticketUpdate?.ticket) {
- notify({
- id: 'ticket-update',
- type: NotificationTypes.Success,
- message: __('Ticket updated successfully.'),
- })
- const screenBehaviour = activeMacro.value
- ? macroScreenBehaviourMapping[activeMacro.value?.uxFlowNextUp]
- : undefined
- handleScreenBehavior({
- screenBehaviour,
- ticket: result.ticketUpdate.ticket,
- })
- skipValidators.value.length = 0
- timeAccountingData.value = undefined
- // Await subscription to update article list before we scroll to the bottom.
- watch(articleResult, scrollToArticlesEnd, {
- once: true,
- })
- // Reset article form after ticket update and reset form.
- newTicketArticlePresent.value = false
- return {
- reset: (
- values: FormSubmitData<TicketUpdateFormData>,
- formNodeValues: FormValues,
- ) => {
- nextTick(() => {
- formReset({ values: { ticket: formNodeValues.ticket } })
- })
- },
- }
- }
- return false
- })
- .catch((error) => {
- if (error instanceof UserError) {
- if (error.getFirstErrorException())
- return handleUserErrorException(error)
- skipValidators.value.length = 0
- timeAccountingData.value = undefined
- if (form.value?.formNode) {
- setErrors(form.value.formNode, error)
- return
- }
- }
- skipValidators.value.length = 0
- timeAccountingData.value = undefined
- notify({
- id: 'ticket-update-failed',
- type: NotificationTypes.Error,
- message: __('Ticket update failed.'),
- })
- })
- .finally(() => {
- disposeActiveMacro()
- })
- }
- const discardReplyForm = async () => {
- const confirm = await waitForVariantConfirmation('unsaved')
- if (!confirm) return
- newTicketArticlePresent.value = false
- await nextTick()
- // Skip subscription for the current tab, to avoid not needed form updater requests.
- setSkipNextStateUpdate(true)
- return triggerFormUpdater()
- }
- const handleShowArticleForm = (
- articleType: string,
- performReply: AppSpecificTicketArticleType['performReply'],
- ) => {
- openReplyForm({ articleType, ...performReply?.(ticket.value) })
- }
- const onEditFormSettled = () => {
- watch(
- () => flags.value.newArticlePresent,
- (newValue) => {
- newTicketArticlePresent.value = newValue
- },
- )
- }
- // Reset newTicketArticlePresent when ticket changed, that the
- // taskbar information is used for the start.
- watch(ticketId, () => {
- initialTicketValue.value = undefined
- newTicketArticlePresent.value = undefined
- })
- </script>
- <template>
- <LayoutContent
- name="ticket-detail"
- no-padding
- background-variant="primary"
- :show-sidebar="hasSidebar"
- content-alignment="center"
- no-scrollable
- >
- <CommonLoader class="mt-8" :loading="!ticket">
- <div
- ref="content-container"
- class="relative grid h-full w-full overflow-y-auto"
- :class="{
- 'grid-rows-[max-content_max-content_max-content]':
- !newTicketArticlePresent || !articleReplyPinned,
- 'grid-rows-[max-content_1fr_max-content]':
- newTicketArticlePresent && articleReplyPinned,
- }"
- >
- <TicketDetailTopBar
- ref="top-bar"
- :hide-details="hideDetails"
- class="sticky left-0 right-0 top-0 w-full"
- />
- <ArticleList :aria-busy="isLoadingArticles" />
- <ArticleReply
- v-if="ticket?.id && isTicketEditable"
- :ticket="ticket"
- :new-article-present="newTicketArticlePresent"
- :create-article-type="ticket.createArticleType?.name"
- :ticket-article-types="ticketArticleTypes"
- :is-ticket-customer="isTicketCustomer"
- :has-internal-article="hasInternalArticle"
- :parent-reached-bottom-scroll="reachedBottom"
- @show-article-form="handleShowArticleForm"
- @discard-form="discardReplyForm"
- />
- <div id="wrapper-form-ticket-edit" class="hidden" aria-hidden="true">
- <Form
- v-if="ticket?.id && initialTicketValue"
- id="form-ticket-edit"
- :key="ticket.id"
- ref="form"
- :form-id="activeTaskbarTabFormId"
- :schema="ticketEditSchema"
- :disabled="!isTicketEditable"
- :flatten-form-groups="['ticket']"
- :handlers="[articleTypeHandler()]"
- :form-kit-plugins="[articleTypeSelectHandler]"
- :schema-data="ticketEditSchemaData"
- :initial-values="initialTicketValue"
- :initial-entity-object="ticket"
- :form-updater-id="EnumFormUpdaterId.FormUpdaterUpdaterTicketEdit"
- :form-updater-additional-params="formAdditionalRouteQueryParams"
- use-object-attributes
- :schema-component-library="{
- Teleport: markRaw(Teleport) as unknown as Component,
- }"
- @submit="
- submitEditTicket($event as FormSubmitData<TicketUpdateFormData>)
- "
- @settled="onEditFormSettled"
- @changed="setSkipNextStateUpdate(true)"
- />
- </div>
- </div>
- </CommonLoader>
- <template #sideBar="{ isCollapsed, toggleCollapse }">
- <TicketSidebar
- :is-collapsed="isCollapsed"
- :toggle-collapse="toggleCollapse"
- :context="sidebarContext"
- />
- </template>
- <template #bottomBar>
- <TicketDetailBottomBar
- :can-use-draft="canUseDraft"
- :dirty="isDirty"
- :disabled="isDisabled"
- :form="form"
- :group-id="groupId"
- :is-ticket-agent="isTicketAgent"
- :is-ticket-editable="isTicketEditable"
- :has-available-draft="hasAvailableDraft"
- :live-user-list="liveUserList"
- :shared-draft-id="ticket?.sharedDraftZoomId"
- :ticket-id="ticketId"
- @submit="checkSubmitEditTicket"
- @discard="discardChanges"
- @execute-macro="executeMacro"
- />
- </template>
- </LayoutContent>
- </template>
|