// Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ import { getNode } from '@formkit/core' import { waitFor, within } from '@testing-library/vue' import ticketCustomerObjectAttributes from '#tests/graphql/factories/fixtures/ticket-customer-object-attributes.ts' import { visitView } from '#tests/support/components/visitView.ts' import { mockApplicationConfig } from '#tests/support/mock-applicationConfig.ts' import { mockPermissions } from '#tests/support/mock-permissions.ts' import { waitForNextTick } from '#tests/support/utils.ts' import { waitForFormUpdaterQueryCalls } from '#shared/components/Form/graphql/queries/formUpdater.mocks.ts' import { mockObjectManagerFrontendAttributesQuery } from '#shared/entities/object-attributes/graphql/queries/objectManagerFrontendAttributes.mocks.ts' import { waitForTicketCreateMutationCalls } from '#shared/entities/ticket/graphql/mutations/create.mocks.ts' import { waitForTicketSharedDraftStartCreateMutationCalls } from '#shared/entities/ticket-shared-draft-start/graphql/mutations/ticketSharedDraftStartCreate.mocks.ts' import { waitForTicketSharedDraftStartDeleteMutationCalls } from '#shared/entities/ticket-shared-draft-start/graphql/mutations/ticketSharedDraftStartDelete.mocks.ts' import { waitForTicketSharedDraftStartUpdateMutationCalls } from '#shared/entities/ticket-shared-draft-start/graphql/mutations/ticketSharedDraftStartUpdate.mocks.ts' import { mockTicketSharedDraftStartListQuery, waitForTicketSharedDraftStartListQueryCalls, } from '#shared/entities/ticket-shared-draft-start/graphql/queries/ticketSharedDraftStartList.mocks.ts' import { mockTicketSharedDraftStartSingleQuery, waitForTicketSharedDraftStartSingleQueryCalls, } from '#shared/entities/ticket-shared-draft-start/graphql/queries/ticketSharedDraftStartSingle.mocks.ts' import { getTicketSharedDraftStartUpdateByGroupSubscriptionHandler } from '#shared/entities/ticket-shared-draft-start/graphql/subscriptions/ticketSharedDraftStartUpdateByGroup.mocks.ts' import { convertToGraphQLId, getIdFromGraphQLId, } from '#shared/graphql/utils.ts' import { handleMockFormUpdaterQuery } from '#desktop/pages/ticket/__tests__/support/ticket-create-helpers.ts' vi.hoisted(() => { vi.setSystemTime('2024-07-03T13:48:09Z') }) describe('ticket create view - shared drafts sidebar', async () => { describe('with agent permissions', async () => { beforeEach(() => { mockApplicationConfig({ ui_ticket_create_available_types: [ 'phone-in', 'phone-out', 'email-out', ], }) mockPermissions(['ticket.agent']) handleMockFormUpdaterQuery() }) it('supports creating shared drafts', async () => { const view = await visitView('/ticket/create') await view.events.type( await view.findByLabelText('Text'), 'foobar
Signature here
', ) const formUpdaterCalls = await waitForFormUpdaterQueryCalls() await vi.waitUntil(() => formUpdaterCalls.length === 2) mockTicketSharedDraftStartListQuery({ ticketSharedDraftStartList: [], }) await view.events.click(view.getByLabelText('Group')) await view.events.click(view.getByRole('option', { name: 'Users' })) await waitForTicketSharedDraftStartListQueryCalls() const aside = within( view.getByRole('complementary', { name: 'Content sidebar', }), ) await view.events.type( aside.getByLabelText('Create a shared draft'), 'Test shared draft 1', ) await getNode('sharedDraftTitle')?.settled await view.events.click( aside.getByRole('link', { name: 'Create Shared Draft' }), ) const calls = await waitForTicketSharedDraftStartCreateMutationCalls() expect(calls.at(-1)?.variables).toEqual({ name: 'Test shared draft 1', input: expect.objectContaining({ groupId: convertToGraphQLId('Group', 1), content: expect.objectContaining({ body: 'foobar', }), }), }) expect(view.getByRole('alert')).toHaveTextContent( 'Shared draft has been created successfully.', ) await getTicketSharedDraftStartUpdateByGroupSubscriptionHandler().trigger( { ticketSharedDraftStartUpdateByGroup: { sharedDraftStarts: [ { id: convertToGraphQLId('Ticket::SharedDraftStart', 1), name: 'Test shared draft 1', updatedAt: '2024-07-03T13:48:09Z', updatedBy: { fullname: 'Erika Mustermann', }, }, ], }, }, ) await waitForNextTick() expect( aside.getByRole('link', { name: 'Test shared draft 1' }), ).toBeInTheDocument() }) it('supports applying shared drafts', async () => { const view = await visitView('/ticket/create') const draftToMock = { id: convertToGraphQLId('Ticket::SharedDraftStart', 1), name: 'Test shared draft 1', content: { title: 'foobar', customer_id: 'test@example.com', body: 'body', }, updatedAt: '2024-07-03T13:48:09Z', updatedBy: { fullname: 'Erika Mustermann', }, } mockTicketSharedDraftStartListQuery({ ticketSharedDraftStartList: [draftToMock], }) await view.events.click(await view.findByLabelText('Group')) await view.events.click(view.getByRole('option', { name: 'Users' })) await waitForTicketSharedDraftStartListQueryCalls() const aside = within( view.getByRole('complementary', { name: 'Content sidebar', }), ) mockTicketSharedDraftStartSingleQuery({ ticketSharedDraftStartSingle: draftToMock, }) await view.events.click( aside.getByRole('link', { name: draftToMock.name }), ) await waitForTicketSharedDraftStartSingleQueryCalls() const flyout = within( view.getByRole('complementary', { name: 'Preview Shared Draft', }), ) expect( flyout.getByText(draftToMock.updatedBy.fullname), ).toBeInTheDocument() expect(flyout.getByText('just now')).toBeInTheDocument() expect(flyout.getByText(draftToMock.content.body)).toBeInTheDocument() await view.events.click(flyout.getByRole('button', { name: 'Apply' })) expect( await view.findByRole('dialog', { name: 'Apply Draft' }), ).toBeInTheDocument() const dialog = within( view.getByRole('dialog', { name: 'Apply Draft', }), ) handleMockFormUpdaterQuery({ title: { value: draftToMock.content.title }, customer_id: { value: draftToMock.content.customer_id, options: [{ value: draftToMock.content.customer_id }], }, body: { value: draftToMock.content.body }, pending_time: { show: false }, shared_draft_id: { value: getIdFromGraphQLId(draftToMock.id) }, }) await view.events.click( dialog.getByRole('button', { name: 'Overwrite Content' }), ) await waitFor(() => { expect( view.queryByRole('dialog', { name: 'Apply Draft', }), ).not.toBeInTheDocument() }) const formUpdaterCalls = await waitForFormUpdaterQueryCalls() expect(formUpdaterCalls.at(-1)?.variables).toEqual( expect.objectContaining({ meta: expect.objectContaining({ additionalData: expect.objectContaining({ sharedDraftId: draftToMock.id, draftType: 'start', }), }), }), ) await waitForNextTick() expect(view.getByLabelText('Title')).toHaveValue( draftToMock.content.title, ) await view.events.click(view.getByRole('button', { name: 'Create' })) const ticketCreateCalls = await waitForTicketCreateMutationCalls() expect(ticketCreateCalls.at(-1)?.variables).toEqual( expect.objectContaining({ input: expect.objectContaining({ title: draftToMock.content.title, sharedDraftId: draftToMock.id, }), }), ) }) it('supports updating shared drafts', async () => { const view = await visitView('/ticket/create') mockTicketSharedDraftStartListQuery({ ticketSharedDraftStartList: [ { id: convertToGraphQLId('Ticket::SharedDraftStart', 1), name: 'Test shared draft 1', updatedAt: '2024-07-03T13:48:09Z', updatedBy: { fullname: 'Erika Mustermann', }, }, ], }) await view.events.click(await view.findByLabelText('Group')) await view.events.click(view.getByRole('option', { name: 'Users' })) await waitForTicketSharedDraftStartListQueryCalls() const aside = within( view.getByRole('complementary', { name: 'Content sidebar', }), ) mockTicketSharedDraftStartSingleQuery({ ticketSharedDraftStartSingle: { name: 'Test shared draft 1', content: { body: 'foobar', }, updatedAt: '2024-07-03T13:48:09Z', updatedBy: { fullname: 'Erika Mustermann', }, }, }) await view.events.click( aside.getByRole('link', { name: 'Test shared draft 1' }), ) await waitForTicketSharedDraftStartSingleQueryCalls() const flyout = within( view.getByRole('complementary', { name: 'Preview Shared Draft', }), ) await view.events.click(flyout.getByRole('button', { name: 'Apply' })) const dialog = within( await view.findByRole('dialog', { name: 'Apply Draft' }), ) handleMockFormUpdaterQuery({ shared_draft_id: { value: 1, }, body: { value: 'foobar', }, }) await view.events.click( dialog.getByRole('button', { name: 'Overwrite Content' }), ) await waitFor(() => { expect( view.queryByRole('complementary', { name: 'Preview Shared Draft', }), ).not.toBeInTheDocument() }) await view.events.click( aside.getByRole('button', { name: 'Update Shared Draft' }), ) const calls = await waitForTicketSharedDraftStartUpdateMutationCalls() expect(calls.at(-1)?.variables).toEqual({ sharedDraftId: convertToGraphQLId('Ticket::SharedDraftStart', 1), input: expect.objectContaining({ content: expect.objectContaining({ body: 'foobar', }), }), }) expect(view.getByRole('alert')).toHaveTextContent( 'Shared draft has been updated successfully.', ) expect( aside.getByRole('button', { name: 'Update Shared Draft' }), ).toBeInTheDocument() }) it('supports deleting shared drafts', async () => { const view = await visitView('/ticket/create') mockTicketSharedDraftStartListQuery({ ticketSharedDraftStartList: [ { id: convertToGraphQLId('Ticket::SharedDraftStart', 1), name: 'Test shared draft 1', updatedAt: '2024-07-03T13:48:09Z', updatedBy: { fullname: 'Erika Mustermann', }, }, ], }) await view.events.click(await view.findByLabelText('Group')) await view.events.click(view.getByRole('option', { name: 'Users' })) await waitForTicketSharedDraftStartListQueryCalls() const aside = within( view.getByRole('complementary', { name: 'Content sidebar', }), ) mockTicketSharedDraftStartSingleQuery({ ticketSharedDraftStartSingle: { name: 'Test shared draft 1', content: { body: 'foobar', }, updatedAt: '2024-07-03T13:48:09Z', updatedBy: { fullname: 'Erika Mustermann', }, }, }) await view.events.click( aside.getByRole('link', { name: 'Test shared draft 1' }), ) await waitForTicketSharedDraftStartSingleQueryCalls() const flyout = within( view.getByRole('complementary', { name: 'Preview Shared Draft', }), ) await view.events.click(flyout.getByRole('button', { name: 'Delete' })) const dialog = within( await view.findByRole('dialog', { name: 'Delete Object' }), ) await view.events.click( dialog.getByRole('button', { name: 'Delete Object' }), ) const calls = await waitForTicketSharedDraftStartDeleteMutationCalls() expect(calls.at(-1)?.variables).toEqual({ sharedDraftId: convertToGraphQLId('Ticket::SharedDraftStart', 1), }) await waitFor(() => { expect( view.queryByRole('complementary', { name: 'Preview Shared Draft', }), ).not.toBeInTheDocument() }) // FIXME: Check why returning an empty array triggers the following console error in test environment only. // Cache data may be lost when replacing the ticketSharedDraftStartList field of a Query object. await getTicketSharedDraftStartUpdateByGroupSubscriptionHandler().trigger( { ticketSharedDraftStartUpdateByGroup: {}, }, ) expect( aside.queryByRole('link', { name: 'Test shared draft 1' }), ).not.toBeInTheDocument() }) }) describe('with customer permission', () => { beforeEach(() => { mockApplicationConfig({ customer_ticket_create: true, }) mockPermissions(['ticket.customer']) // Mock frontend attributes for customer context. // TODO: check if we can mock the query twice based on the variable? mockObjectManagerFrontendAttributesQuery({ objectManagerFrontendAttributes: ticketCustomerObjectAttributes(), }) handleMockFormUpdaterQuery() }) it('does not show', async () => { const view = await visitView('/ticket/create') await view.events.click(await view.findByLabelText('Group')) await view.events.click(view.getByRole('option', { name: 'Users' })) await waitForFormUpdaterQueryCalls() expect( view.queryByRole('complementary', { name: 'Content sidebar', }), ).not.toBeInTheDocument() }) }) })