// Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ import { getAttachmentLinks } from '#shared/composables/getAttachmentLinks.ts' import type { TicketArticle, TicketById, } from '#shared/entities/ticket/types.ts' import { useTicketArticleEmailForwardReplyMutation } from '#shared/entities/ticket-article/graphql/mutations/ticketArticleEmailForwardReply.api.ts' import type { TicketArticleEmailForwardReplyMutation } from '#shared/graphql/types.ts' import { i18n } from '#shared/i18n.ts' import { MutationHandler } from '#shared/server/apollo/handler/index.ts' import type { ConfigList } from '#shared/types/store.ts' import type { ConfidentTake } from '#shared/types/utils.ts' import { textCleanup, textToHtml } from '#shared/utils/helpers.ts' import type { TicketArticlePerformOptions } from '../types.ts' const forwardMutation = new MutationHandler( useTicketArticleEmailForwardReplyMutation({}), { errorShowNotification: true }, ) export const buildEmailForwardHeader = ( article: TicketArticle, meta: ConfidentTake< TicketArticleEmailForwardReplyMutation, 'ticketArticleEmailForwardReply' >, ) => { const { quotableFrom, quotableCc, quotableTo } = meta || {} const fields = [ [__('Subject'), article.subject], [__('Date'), i18n.dateTime(article.createdAt)], [__('From'), quotableFrom], [__('To'), quotableTo], [__('CC'), quotableCc], ] as const const output = fields.reduce((acc, [key, value]) => { if (value) { acc.append(i18n.t(key), ': ', value, document.createElement('br')) } return acc }, document.createElement('p')) output.appendChild(document.createElement('br')) return output.outerHTML } export const forwardEmail = async ( ticket: TicketById, article: TicketArticle, options: TicketArticlePerformOptions, config: ConfigList, ) => { let body = article.contentType === 'text/html' ? textCleanup(article.bodyWithUrls) : textToHtml(textCleanup(article.bodyWithUrls)) // TODO: standardise this in https://github.com/zammad/coordination-feature-mobile-view/issues/396 body = body.replace(/


<\/p>/g, '

') // cleanup const result = await forwardMutation .send({ formId: options.formId, articleId: article.id, }) .then((r) => r?.ticketArticleEmailForwardReply) // show attachment previews, but don't save its content const attachments = (result?.attachments || []).map((file, idx) => { const originalAttachment = article.attachmentsWithoutInline[idx] if (!originalAttachment || originalAttachment.name !== file.name) return file const { previewUrl, inlineUrl } = getAttachmentLinks( { internalId: originalAttachment.internalId, type: file.type, }, config.api_path, ) return { ...file, preview: previewUrl, inline: inlineUrl, } }) const quotedHeader = config.ui_ticket_zoom_article_email_full_quote_header && result ? buildEmailForwardHeader(article, result) : '' const quotedBody = `

---${i18n.t( 'Begin forwarded message', )}:---

${quotedHeader}${body}
` return options.openReplyForm({ articleType: 'email', subject: config.ui_ticket_zoom_article_email_subject ? article.subject || ticket.title : '', subtype: 'forward', attachments, body: quotedBody, }) }