123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import { keyBy } from 'lodash-es'
- import { computed, shallowRef } from 'vue'
- import type { FieldEditorContext } from '#shared/components/Form/fields/FieldEditor/types.ts'
- import { FormHandlerExecution } from '#shared/components/Form/types.ts'
- import type {
- ChangedField,
- ReactiveFormSchemData,
- FormHandlerFunction,
- FormRef,
- } from '#shared/components/Form/types.ts'
- import { useAppName } from '#shared/composables/useAppName.ts'
- import { useTicketView } from '#shared/entities/ticket/composables/useTicketView.ts'
- import type { TicketById } from '#shared/entities/ticket/types.ts'
- import { createArticleTypes } from '#shared/entities/ticket-article/action/plugins/index.ts'
- import type {
- AppSpecificTicketArticleType,
- TicketArticleTypeFields,
- } from '#shared/entities/ticket-article/action/plugins/types.ts'
- import { EnumObjectManagerObjects } from '#shared/graphql/types.ts'
- import { useApplicationStore } from '#shared/stores/application.ts'
- import type { FormKitNode } from '@formkit/core'
- import type { Ref } from 'vue'
- export const useTicketEditForm = (
- ticket: Ref<TicketById | undefined>,
- form: Ref<FormRef | undefined>,
- ) => {
- const appName = useAppName()
- const ticketArticleTypes = computed(() => {
- return ticket.value ? createArticleTypes(ticket.value, appName) : []
- })
- const ticketArticleTypeValueLookup = computed(() =>
- keyBy(ticketArticleTypes.value, 'value'),
- )
- const currentArticleType = shallowRef<AppSpecificTicketArticleType>()
- const recipientContact = computed(
- () => currentArticleType.value?.options?.recipientContact,
- )
- const editorType = computed(() => currentArticleType.value?.contentType)
- const editorMeta = computed(() => {
- return {
- mentionUser: {
- groupNodeName: 'group_id',
- },
- mentionKnowledgeBase: {
- attachmentsNodeName: 'attachments',
- },
- ...currentArticleType.value?.editorMeta,
- }
- })
- const articleTypeFields = [
- 'to',
- 'cc',
- 'subject',
- 'body',
- 'attachments',
- 'security',
- ] as const
- const articleTypeFieldProps = articleTypeFields.reduce((acc, field) => {
- acc[field] = {
- validation: computed(
- () => currentArticleType.value?.fields?.[field]?.validation || null,
- ),
- required: computed(
- () => !!currentArticleType.value?.fields?.[field]?.required,
- ),
- }
- return acc
- }, {} as TicketArticleTypeFields)
- const { isTicketAgent, isTicketCustomer, isTicketEditable } =
- useTicketView(ticket)
- const isMobileApp = appName === 'mobile'
- const ticketSchema = {
- type: 'group',
- name: 'ticket', // will be flattened in the form submit result
- isGroupOrList: true,
- children: [
- ...(isMobileApp
- ? [
- {
- name: 'title',
- type: 'text',
- label: __('Ticket title'),
- required: true,
- },
- ]
- : []),
- {
- type: 'hidden',
- name: 'isDefaultFollowUpStateSet',
- },
- {
- screen: 'edit',
- object: EnumObjectManagerObjects.Ticket,
- },
- ],
- }
- const articleSchema = {
- // Desktop is handling the condition on top for the teleport.
- if: isMobileApp
- ? '$newTicketArticleRequested || $newTicketArticlePresent'
- : undefined,
- type: 'group',
- name: 'article',
- isGroupOrList: true,
- children: [
- {
- type: 'hidden',
- name: 'inReplyTo',
- },
- {
- if: '$currentArticleType.fields.subtype',
- type: 'hidden',
- name: 'subtype',
- },
- {
- name: 'articleType',
- label: __('Channel'),
- labelSrOnly: isMobileApp,
- type: 'select',
- hidden: computed(() => ticketArticleTypes.value.length === 1),
- props: {
- // We need to disable the auto preselection when the field
- // is initialized, so that we have a correct dirty state.
- noInitialAutoPreselect: true,
- options: ticketArticleTypes,
- },
- },
- {
- name: 'internal',
- label: __('Visibility'),
- labelSrOnly: isMobileApp,
- hidden: isTicketCustomer,
- type: 'select',
- props: {
- options: [
- {
- value: true,
- label: __('Internal'),
- icon: 'lock',
- },
- {
- value: false,
- label: __('Public'),
- icon: 'unlock',
- },
- ],
- },
- },
- {
- if: '$currentArticleType.fields.to',
- name: 'to',
- label: __('To'),
- type: 'recipient',
- validation: articleTypeFieldProps.to.validation,
- props: {
- contact: recipientContact,
- multiple: true,
- },
- required: articleTypeFieldProps.to.required,
- },
- {
- if: '$currentArticleType.fields.cc',
- name: 'cc',
- label: __('CC'),
- type: 'recipient',
- validation: articleTypeFieldProps.cc.validation,
- props: {
- contact: recipientContact,
- multiple: true,
- },
- },
- {
- if: '$currentArticleType.fields.subject',
- name: 'subject',
- label: __('Subject'),
- type: 'text',
- validation: articleTypeFieldProps.subject.validation,
- props: {
- maxlength: 200,
- },
- required: articleTypeFieldProps.subject.required,
- },
- {
- if: '$securityIntegration === true && $currentArticleType.fields.security',
- name: 'security',
- label: __('Security'),
- type: 'security',
- validation: articleTypeFieldProps.security.validation,
- },
- {
- name: 'body',
- screen: 'edit',
- object: EnumObjectManagerObjects.TicketArticle,
- validation: articleTypeFieldProps.body.validation,
- props: {
- ticketId: computed(() => ticket.value?.internalId),
- customerId: computed(() => ticket.value?.customer.internalId),
- contentType: editorType,
- meta: editorMeta,
- },
- required: articleTypeFieldProps.body.required,
- },
- {
- if: '$currentArticleType.fields.attachments',
- type: 'file',
- name: 'attachments',
- label: __('Attachment'),
- labelSrOnly: true,
- validation: articleTypeFieldProps.attachments.validation,
- props: {
- multiple: computed(() =>
- Boolean(
- typeof currentArticleType.value?.fields?.attachments?.multiple ===
- 'boolean'
- ? currentArticleType.value?.fields?.attachments?.multiple
- : true,
- ),
- ),
- allowedFiles: computed(
- () =>
- currentArticleType.value?.fields?.attachments?.allowedFiles ||
- null,
- ),
- accept: computed(
- () => currentArticleType.value?.fields?.attachments?.accept || null,
- ),
- },
- required: articleTypeFieldProps.attachments.required,
- },
- ],
- }
- const articleTypeChangeHandler = () => {
- const executeTypeChangeHandler = (
- execution: FormHandlerExecution,
- schemaData: ReactiveFormSchemData,
- changedField?: ChangedField,
- ) => {
- if (!schemaData.fields.articleType) return false
- return !(
- execution === FormHandlerExecution.FieldChange &&
- (!changedField || changedField.name !== 'articleType')
- )
- }
- const handleArticleType: FormHandlerFunction = (
- execution,
- reactivity,
- data,
- ) => {
- const { formNode, changedField, formUpdaterData } = data
- const { schemaData } = reactivity
- if (
- execution === FormHandlerExecution.Initial &&
- formUpdaterData?.fields.articleType?.value
- ) {
- currentArticleType.value =
- ticketArticleTypeValueLookup.value[
- formUpdaterData.fields.articleType.value
- ]
- }
- if (
- !executeTypeChangeHandler(execution, schemaData, changedField) ||
- !ticket.value ||
- !formNode
- )
- return
- const body = formNode.find('body', 'name')
- const context = {
- body: body?.context as unknown as FieldEditorContext,
- }
- if (changedField?.newValue !== changedField?.oldValue) {
- currentArticleType.value?.onDeselected?.(ticket.value, context)
- }
- if (!changedField?.newValue) return
- const newType =
- ticketArticleTypeValueLookup.value[changedField?.newValue as string]
- if (!newType) return
- if (!formNode.context?._open) {
- newType.onSelected?.(ticket.value, context, form.value)
- }
- currentArticleType.value = newType
- formNode.find('internal')?.input(newType.internal, false)
- }
- return {
- execution: [
- FormHandlerExecution.Initial,
- FormHandlerExecution.FieldChange,
- ],
- callback: handleArticleType,
- }
- }
- const articleTypeSelectHandler = (formNode: FormKitNode) => {
- // this is called only when user replied to an article, but the type inside form did not change
- // (because dialog was opened before, and type was changed then, but we still need to trigger select, because visually it's what happens)
- formNode.on('article-reply-open', ({ payload }) => {
- if (!payload || !ticket.value) return
- const articleType = ticketArticleTypeValueLookup.value[payload as string]
- if (!articleType) return
- const body = formNode.find('body', 'name') as FormKitNode
- const context = {
- body: body.context as unknown as FieldEditorContext,
- }
- articleType.onOpened?.(ticket.value, context, form.value)
- })
- }
- const application = useApplicationStore()
- const securityIntegration = computed<boolean>(
- () =>
- (application.config.smime_integration ||
- application.config.pgp_integration) ??
- false,
- )
- return {
- ticketSchema,
- articleSchema,
- currentArticleType,
- ticketArticleTypes,
- securityIntegration,
- isTicketAgent,
- isTicketCustomer,
- isTicketEditable,
- articleTypeHandler: articleTypeChangeHandler,
- articleTypeSelectHandler,
- }
- }
|