123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import { NotificationTypes } from '#shared/components/CommonNotifications/types.ts'
- import { useNotifications } from '#shared/components/CommonNotifications/useNotifications.ts'
- import { populateEditorNewLines } from '#shared/components/Form/fields/FieldEditor/utils.ts'
- import type { FormRef, FormSubmitData } from '#shared/components/Form/types.ts'
- import { setErrors } from '#shared/components/Form/utils.ts'
- import { useCheckBodyAttachmentReference } from '#shared/composables/form/useCheckBodyAttachmentReference.ts'
- import { useObjectAttributeFormData } from '#shared/entities/object-attributes/composables/useObjectAttributeFormData.ts'
- import { useObjectAttributes } from '#shared/entities/object-attributes/composables/useObjectAttributes.ts'
- import { ticketCreateArticleType } from '#shared/entities/ticket/composables/useTicketCreateArticleType.ts'
- import { useTicketCreateMutation } from '#shared/entities/ticket/graphql/mutations/create.api.ts'
- import UserError from '#shared/errors/UserError.ts'
- import {
- EnumObjectManagerObjects,
- type TicketCreateInput,
- } from '#shared/graphql/types.ts'
- import { isGraphQLId, convertToGraphQLId } from '#shared/graphql/utils.ts'
- import MutationHandler from '#shared/server/apollo/handler/MutationHandler.ts'
- import { GraphQLErrorTypes } from '#shared/types/error.ts'
- import { convertFilesToAttachmentInput } from '#shared/utils/files.ts'
- import { useTicketCreateView } from './useTicketCreateView.ts'
- import type { TicketFormData } from '../types.ts'
- import type { ApolloError } from '@apollo/client/core'
- import type { Ref } from 'vue'
- export const useTicketCreate = (
- form: Ref<FormRef | undefined>,
- redirectAfterCreate: (internalId?: number) => void,
- ) => {
- const { isTicketCustomer } = useTicketCreateView()
- const { notify } = useNotifications()
- const notifySuccess = () => {
- notify({
- id: 'ticket-create-success',
- type: NotificationTypes.Success,
- message: __('Ticket has been created successfully.'),
- })
- }
- const handleTicketCreateError = (error: UserError | ApolloError) => {
- if ('graphQLErrors' in error) {
- const graphQLErrors = error.graphQLErrors?.[0]
- // Treat this as successful, because it happens when you create a ticket inside a group, where you only
- // have create permission, but not view permission.
- if (graphQLErrors?.extensions?.type === GraphQLErrorTypes.Forbidden) {
- notifySuccess()
- return () => redirectAfterCreate()
- }
- notify({
- id: 'ticket-create-error',
- message: __('Ticket could not be created.'),
- type: NotificationTypes.Error,
- })
- } else {
- if (error instanceof UserError && form.value?.formNode) {
- setErrors(form.value?.formNode, error)
- return
- }
- notify({
- id: 'ticket-create-error',
- message: error.generalErrors[0],
- type: NotificationTypes.Error,
- })
- }
- }
- const ticketCreateMutation = new MutationHandler(
- useTicketCreateMutation({}),
- {
- errorShowNotification: false,
- },
- )
- const {
- missingBodyAttachmentReference,
- bodyAttachmentReferenceConfirmation,
- } = useCheckBodyAttachmentReference()
- const getCustomerVariable = (customerId: string) => {
- return isGraphQLId(customerId) ? { id: customerId } : { email: customerId }
- }
- const createTicket = async (formData: FormSubmitData<TicketFormData>) => {
- // Check for possible missing attached files and ask for confirmation.
- // With return false, the form submit is stopped.
- if (
- missingBodyAttachmentReference(formData.body, formData.attachments) &&
- (await bodyAttachmentReferenceConfirmation())
- ) {
- return false
- }
- const { attributesLookup: ticketObjectAttributesLookup } =
- useObjectAttributes(EnumObjectManagerObjects.Ticket)
- const { internalObjectAttributeValues, additionalObjectAttributeValues } =
- useObjectAttributeFormData(ticketObjectAttributesLookup.value, formData)
- // The customerId has an special handling, so we need to extract it from the internalObjectAttributeValues.
- const { customerId, ...internalValues } = internalObjectAttributeValues
- let sharedDraftId
- if (formData.shared_draft_id) {
- sharedDraftId = convertToGraphQLId(
- 'Ticket::SharedDraftStart',
- formData.shared_draft_id as string | number,
- )
- }
- const input = {
- ...internalValues,
- sharedDraftId,
- customer: customerId
- ? getCustomerVariable(customerId as string)
- : undefined,
- article: {
- cc: formData.cc,
- body: populateEditorNewLines(formData.body),
- sender: isTicketCustomer.value
- ? 'Customer'
- : ticketCreateArticleType[formData.articleSenderType].sender,
- type: isTicketCustomer.value
- ? 'web'
- : ticketCreateArticleType[formData.articleSenderType].type,
- contentType: 'text/html',
- security: formData.security,
- },
- objectAttributeValues: additionalObjectAttributeValues,
- } as TicketCreateInput
- if (formData.attachments && input.article && form.value?.formId) {
- input.article.attachments = convertFilesToAttachmentInput(
- form.value.formId,
- formData.attachments,
- )
- }
- if (formData.link_ticket_id) {
- const linkObjectId = convertToGraphQLId(
- 'Ticket',
- formData.link_ticket_id as string | number,
- )
- input.links = [
- {
- linkObjectId,
- linkType: 'child',
- },
- ]
- }
- return ticketCreateMutation
- .send({ input })
- .then((result) => {
- if (result?.ticketCreate?.ticket) {
- notifySuccess()
- return () => {
- const ticket = result.ticketCreate?.ticket
- redirectAfterCreate(
- ticket?.policy.update ? ticket.internalId : undefined,
- )
- }
- }
- return null
- })
- .catch(handleTicketCreateError)
- }
- return {
- createTicket,
- isTicketCustomer,
- }
- }
|