123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382 |
- <!-- Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ -->
- <script setup lang="ts">
- import { cloneDeep, noop } from 'lodash-es'
- import { computed, provide, ref, reactive, toRef, nextTick } from 'vue'
- import {
- onBeforeRouteLeave,
- onBeforeRouteUpdate,
- RouterView,
- useRoute,
- useRouter,
- } from 'vue-router'
- 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 { useConfirmation } from '#shared/composables/useConfirmation.ts'
- import { useOnlineNotificationSeen } from '#shared/composables/useOnlineNotificationSeen.ts'
- import { useTicketEdit } from '#shared/entities/ticket/composables/useTicketEdit.ts'
- import { useTicketEditForm } from '#shared/entities/ticket/composables/useTicketEditForm.ts'
- import { useTicketView } from '#shared/entities/ticket/composables/useTicketView.ts'
- import { TicketUpdatesDocument } from '#shared/entities/ticket/graphql/subscriptions/ticketUpdates.api.ts'
- import type { TicketUpdateFormData } from '#shared/entities/ticket/types.ts'
- import { useErrorHandler } from '#shared/errors/useErrorHandler.ts'
- import UserError from '#shared/errors/UserError.ts'
- import type {
- TicketUpdatesSubscription,
- TicketUpdatesSubscriptionVariables,
- } from '#shared/graphql/types.ts'
- import {
- EnumFormUpdaterId,
- EnumUserErrorException,
- } from '#shared/graphql/types.ts'
- import { convertToGraphQLId } from '#shared/graphql/utils.ts'
- import { QueryHandler } from '#shared/server/apollo/handler/index.ts'
- import CommonLoader from '#mobile/components/CommonLoader/CommonLoader.vue'
- import { useCommonSelect } from '#mobile/components/CommonSelect/useCommonSelect.ts'
- import { getOpenedDialogs } from '#mobile/composables/useDialog.ts'
- import { useTicketWithMentionLimitQuery } from '#mobile/entities/ticket/graphql/queries/ticketWithMentionLimit.api.ts'
- import type { TicketInformation } from '#mobile/entities/ticket/types.ts'
- import TicketDetailViewActions from '../components/TicketDetailView/TicketDetailViewActions.vue'
- import { useTicketArticleReply } from '../composable/useTicketArticleReply.ts'
- import { TICKET_INFORMATION_SYMBOL } from '../composable/useTicketInformation.ts'
- import { useTicketLiveUser } from '../composable/useTicketLiveUser.ts'
- interface Props {
- internalId: string
- }
- const props = defineProps<Props>()
- const ticketId = computed(() => convertToGraphQLId('Ticket', props.internalId))
- const MENTIONS_LIMIT = 5
- const { createQueryErrorHandler } = useErrorHandler()
- const ticketQuery = new QueryHandler(
- useTicketWithMentionLimitQuery(() => ({
- ticketId: ticketId.value,
- mentionsCount: MENTIONS_LIMIT,
- })),
- {
- errorCallback: createQueryErrorHandler({
- notFound: __(
- 'Ticket with specified ID was not found. Try checking the URL for errors.',
- ),
- forbidden: __('You have insufficient rights to view this ticket.'),
- }),
- },
- )
- const ticketResult = ticketQuery.result()
- const ticket = computed(() => ticketResult.value?.ticket)
- ticketQuery.subscribeToMore<
- TicketUpdatesSubscriptionVariables,
- TicketUpdatesSubscription
- >(() => ({
- document: TicketUpdatesDocument,
- variables: {
- ticketId: ticketId.value,
- },
- onError: noop,
- }))
- const formLocation = ref('body')
- const formVisible = computed(() => formLocation.value !== 'body')
- const { form, canSubmit, isDirty, formSubmit, formReset } = useForm()
- const { initialTicketValue, isTicketFormGroupValid, editTicket } =
- useTicketEdit(ticket, form)
- const {
- currentArticleType,
- ticketSchema,
- articleSchema,
- securityIntegration,
- isTicketEditable,
- articleTypeHandler,
- articleTypeSelectHandler,
- } = useTicketEditForm(ticket, form)
- const needSpaceForSaveBanner = computed(
- () => isTicketEditable.value && isDirty.value,
- )
- const {
- articleReplyDialog,
- newTicketArticleRequested,
- newTicketArticlePresent,
- isArticleFormGroupValid,
- openArticleReplyDialog,
- closeArticleReplyDialog,
- } = useTicketArticleReply(ticket, form, needSpaceForSaveBanner)
- const ticketEditSchema = [
- {
- isLayout: true,
- component: 'FormGroup',
- props: {
- style: {
- if: '$formLocation !== "[data-ticket-edit-form]"',
- then: 'display: none;',
- },
- showDirtyMark: true,
- },
- children: [ticketSchema],
- },
- {
- isLayout: true,
- component: 'FormGroup',
- props: {
- style: {
- if: '$formLocation !== "[data-ticket-article-reply-form]"',
- then: 'display: none;',
- },
- },
- children: [articleSchema],
- },
- ]
- const { isTicketAgent } = useTicketView(ticket)
- const { notify } = useNotifications()
- const saveTicketForm = async (
- formData: FormSubmitData<TicketUpdateFormData>,
- ) => {
- let data = cloneDeep(formData)
- if (currentArticleType.value?.updateForm)
- data = currentArticleType.value.updateForm(formData)
- try {
- const result = await editTicket(
- data,
- { skipValidators: Object.values(EnumUserErrorException) }, // skip all validators, they are irrelevant for mobile view
- )
- if (result?.ticketUpdate?.ticket) {
- notify({
- id: 'ticket-update',
- type: NotificationTypes.Success,
- message: __('Ticket updated successfully.'),
- })
- // Reset article form after ticket update and reset form.
- newTicketArticlePresent.value = false
- return {
- reset: (
- values: FormSubmitData<TicketUpdateFormData>,
- formNodeValues: FormValues,
- ) => {
- nextTick(() => {
- closeArticleReplyDialog().then(() => {
- formReset({ values: { ticket: formNodeValues.ticket } })
- })
- })
- },
- }
- }
- } catch (errors) {
- if (errors instanceof UserError) {
- notify({
- id: 'ticket-update-error',
- message: errors.generalErrors[0],
- type: NotificationTypes.Error,
- })
- }
- }
- }
- const updateFormLocation = (newLocation: string) => {
- formLocation.value = newLocation
- }
- const isFormValid = computed(() => {
- if (!newTicketArticlePresent.value) return isTicketFormGroupValid.value
- return isTicketFormGroupValid.value && isArticleFormGroupValid.value
- })
- const showArticleReplyDialog = () => {
- return openArticleReplyDialog({ updateFormLocation })
- }
- const { liveUserList } = useTicketLiveUser(
- toRef(() => props.internalId),
- isTicketAgent,
- isDirty,
- )
- const refetchingStatus = ref(false)
- const updateRefetchingStatus = (status: boolean) => {
- refetchingStatus.value = status
- }
- const scrolledToBottom = ref(false)
- const scrollDownState = ref(false)
- onBeforeRouteUpdate((to, from) => {
- // reset if we opened another ticket from the same page (via ticket merge, for example)
- if (to.params.internalId !== from.params.internalId) {
- scrolledToBottom.value = false
- }
- scrollDownState.value = false
- })
- const newArticlesIds = reactive(new Set<string>())
- provide<TicketInformation>(TICKET_INFORMATION_SYMBOL, {
- ticketQuery,
- initialFormTicketValue: initialTicketValue,
- ticket,
- form,
- scrolledToBottom,
- newTicketArticleRequested,
- newTicketArticlePresent,
- updateFormLocation,
- isTicketEditable,
- showArticleReplyDialog,
- liveUserList,
- refetchingStatus,
- newArticlesIds,
- scrollDownState,
- updateRefetchingStatus,
- })
- useOnlineNotificationSeen(ticket)
- onBeforeRouteLeave(async () => {
- if (!isDirty.value) return true
- const { waitForConfirmation } = useConfirmation()
- const confirmed = await waitForConfirmation(
- __('Are you sure? You have unsaved changes that will get lost.'),
- {
- buttonLabel: __('Discard changes'),
- buttonVariant: 'danger',
- },
- )
- return confirmed
- })
- const router = useRouter()
- const route = useRoute()
- const submitForm = () => {
- if (!isTicketFormGroupValid.value && route.name !== 'Edit') {
- if (articleReplyDialog.isOpened.value) {
- closeArticleReplyDialog(true)
- }
- router.push(`/tickets/${ticket.value?.internalId}/information`)
- } else if (
- newTicketArticlePresent.value &&
- !isArticleFormGroupValid.value &&
- !articleReplyDialog.isOpened.value
- ) {
- showArticleReplyDialog()
- }
- formSubmit()
- }
- const ticketEditSchemaData = reactive({
- formLocation,
- securityIntegration,
- newTicketArticleRequested,
- newTicketArticlePresent,
- currentArticleType,
- })
- const { isOpened: commonSelectOpened } = useCommonSelect()
- const showReplyButton = computed(() => {
- if (articleReplyDialog.isOpened.value) return false
- return isTicketEditable.value
- })
- const showScrollDown = computed(() => {
- if (articleReplyDialog.isOpened.value) return false
- return scrollDownState.value
- })
- // show banner only in "articles list", "ticket information" and "create article" views
- const showBottomBanner = computed(() => {
- const dialogs = getOpenedDialogs()
- if (
- commonSelectOpened.value ||
- dialogs.size > 1 ||
- (dialogs.size === 1 && !articleReplyDialog.isOpened.value)
- )
- return false
- return (
- (isTicketEditable.value && isDirty.value) ||
- showReplyButton.value ||
- showScrollDown.value
- )
- })
- </script>
- <template>
- <RouterView />
- <div class="pb-safe-16"></div>
- <!-- submit form is always present in the DOM, so we can access FormKit validity state -->
- <!-- if it's visible, it's moved to the [data-ticket-edit-form] element, which is in TicketInformationDetail -->
- <Teleport v-if="isTicketEditable" :to="formLocation">
- <CommonLoader
- :class="formVisible ? 'visible' : 'hidden'"
- :loading="!ticket"
- >
- <Form
- v-if="ticket?.id && initialTicketValue"
- id="form-ticket-edit"
- :key="ticket.id"
- ref="form"
- :schema="ticketEditSchema"
- :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"
- use-object-attributes
- :aria-hidden="!formVisible"
- :class="formVisible ? 'visible' : 'hidden'"
- @submit="saveTicketForm($event as FormSubmitData<TicketUpdateFormData>)"
- />
- </CommonLoader>
- </Teleport>
- <Teleport v-if="form?.formNode" to="body">
- <TicketDetailViewActions
- :form-invalid="canSubmit && !isFormValid"
- :new-replies-count="newArticlesIds.size"
- :new-article-present="newTicketArticlePresent"
- :can-reply="showReplyButton"
- :can-save="isTicketEditable && isDirty"
- :can-scroll-down="showScrollDown"
- :hidden="!showBottomBanner"
- @reply="showArticleReplyDialog"
- @save="submitForm"
- />
- </Teleport>
- </template>
|