123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- 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 { mockObjectManagerFrontendAttributesQuery } from '#shared/entities/object-attributes/graphql/queries/objectManagerFrontendAttributes.mocks.ts'
- import { waitForTicketCreateMutationCalls } from '#shared/entities/ticket/graphql/mutations/create.mocks.ts'
- import {
- EnumTaskbarEntity,
- EnumTaskbarEntityAccess,
- } from '#shared/graphql/types.ts'
- import { convertToGraphQLId } from '#shared/graphql/utils.ts'
- import getUuid from '#shared/utils/getUuid.ts'
- import { waitForUserCurrentTaskbarItemUpdateMutationCalls } from '#desktop/entities/user/current/graphql/mutations/userCurrentTaskbarItemUpdate.mocks.ts'
- import { mockUserCurrentTaskbarItemListQuery } from '#desktop/entities/user/current/graphql/queries/userCurrentTaskbarItemList.mocks.ts'
- import {
- handleMockUserQuery,
- handleCustomerMock,
- handleMockFormUpdaterQuery,
- rendersFields,
- handleMockOrganizationQuery,
- } from '#desktop/pages/ticket/__tests__/support/ticket-create-helpers.ts'
- vi.hoisted(() => {
- vi.setSystemTime('2024-11-11T00:00:00Z')
- })
- describe('ticket create view', async () => {
- describe('view with granted access', async () => {
- beforeEach(() => {
- mockApplicationConfig({
- ui_task_mananger_max_task_count: 30,
- ui_ticket_create_available_types: [
- 'phone-in',
- 'phone-out',
- 'email-out',
- ],
- })
- mockPermissions(['ticket.agent'])
- })
- // FIXME: This test example has to be first, otherwise it will start to fail.
- // This is probably due to the leftover router instance, which has to be set up in a different way for this test.
- it('supports updating dirty flag in the associated taskbar tab', async () => {
- const uid = getUuid()
- mockUserCurrentTaskbarItemListQuery({
- userCurrentTaskbarItemList: [
- {
- __typename: 'UserTaskbarItem',
- id: convertToGraphQLId('Taskbar', 1),
- key: `TicketCreateScreen-${uid}`,
- callback: EnumTaskbarEntity.TicketCreate,
- entityAccess: EnumTaskbarEntityAccess.Granted,
- entity: {
- __typename: 'UserTaskbarItemEntityTicketCreate',
- uid,
- title: '',
- createArticleTypeKey: 'phone-in',
- },
- dirty: false,
- },
- ],
- })
- handleMockFormUpdaterQuery()
- const view = await visitView(`/ticket/create/${uid}`)
- expect(
- await view.findByRole('button', { name: 'Cancel & Go Back' }),
- ).toBeInTheDocument()
- await view.events.type(view.getByLabelText('Title'), 'Test Ticket')
- const calls = await waitForUserCurrentTaskbarItemUpdateMutationCalls()
- expect(calls.at(-1)?.variables).toEqual(
- expect.objectContaining({
- input: expect.objectContaining({
- dirty: true,
- }),
- }),
- )
- })
- it('renders view correctly', async () => {
- const view = await visitView('/ticket/create')
- expect(
- await view.findByRole('heading', { level: 1, name: 'New Ticket' }),
- ).toBeInTheDocument()
- expect(view.getByRole('tablist')).toBeInTheDocument()
- // Default tab is the first one
- expect(
- view.getByRole('tab', { selected: true, name: 'Received Call' }),
- ).toBeInTheDocument()
- rendersFields(view)
- })
- it('cancels ticket creation', async () => {
- const view = await visitView('/ticket/create')
- expect(
- await view.findByRole('heading', { level: 1, name: 'New Ticket' }),
- ).toBeInTheDocument()
- await view.events.click(
- view.getByRole('button', { name: 'Cancel & Go Back' }),
- )
- await waitFor(() =>
- expect(
- view.queryByRole('heading', { level: 1, name: 'New Ticket' }),
- ).not.toBeInTheDocument(),
- )
- })
- it('shows send email article type', async () => {
- const view = await visitView('/ticket/create')
- await view.events.click(await view.findByText('Send Email'))
- expect(
- view.getByRole('tab', { selected: true, name: 'Send Email' }),
- ).toBeInTheDocument()
- expect(view.getByLabelText('CC')).toBeInTheDocument()
- rendersFields(view)
- })
- it('shows outbound call article type', async () => {
- const view = await visitView('/ticket/create')
- await view.events.click(await view.findByText('Outbound Call'))
- expect(
- view.getByRole('tab', { selected: true, name: 'Outbound Call' }),
- ).toBeInTheDocument()
- rendersFields(view)
- })
- it('detects duplicate ticket', async () => {
- await mockApplicationConfig({
- ticket_duplicate_detection: true,
- ticket_duplicate_detection_title: 'Similar tickets found',
- ticket_duplicate_detection_body:
- 'Tickets with the same attributes were found.',
- })
- handleMockFormUpdaterQuery({
- ticket_duplicate_detection: {
- show: true,
- hidden: false,
- value: { count: 1, items: [[1, '123,', 'foo title']] },
- },
- })
- const view = await visitView('/ticket/create')
- await view.events.type(await view.findByLabelText('Title'), 'foo title')
- await waitFor(() =>
- expect(view.getByText('Similar tickets found')).toBeInTheDocument(),
- )
- expect(view.getByTestId('common-alert')).toHaveTextContent('foo title')
- expect(view.getByIconName('exclamation-triangle')).toBeInTheDocument()
- expect(
- view.getByText('Tickets with the same attributes were found.'),
- ).toBeInTheDocument()
- })
- it('prevents submission on incomplete form', async () => {
- handleMockFormUpdaterQuery()
- const view = await visitView('/ticket/create')
- await view.events.type(await view.findByLabelText('Title'), 'Test Ticket')
- await view.events.click(view.getByRole('button', { name: 'Create' }))
- expect(await view.findAllByText('This field is required.')).toHaveLength(
- 4,
- )
- })
- it('discards unsaved changes', async () => {
- handleMockFormUpdaterQuery()
- const view = await visitView('/ticket/create')
- expect(
- await view.findByRole('button', { name: 'Cancel & Go Back' }),
- ).toBeInTheDocument()
- await view.events.type(view.getByLabelText('Title'), 'Test Ticket')
- await waitFor(() =>
- expect(
- view.queryByRole('button', { name: 'Cancel & Go Back' }),
- ).not.toBeInTheDocument(),
- )
- await view.events.click(
- await view.findByRole('button', { name: 'Discard Changes' }),
- )
- const dialog = await view.findByRole('dialog', {
- name: 'Unsaved Changes',
- })
- expect(dialog).toBeInTheDocument()
- const dialogView = within(dialog)
- expect(
- await dialogView.findByText(
- 'Are you sure? You have unsaved changes that will get lost.',
- ),
- )
- await view.events.click(
- dialogView.getByRole('button', { name: 'Discard Changes' }),
- )
- // should not be in the document anymore
- await waitFor(() =>
- expect(view.getByLabelText('Title')).not.toHaveValue('Test Ticket'),
- )
- })
- it('keeps form values on cancel unsaved changes', async () => {
- handleMockFormUpdaterQuery()
- const view = await visitView('/ticket/create')
- await view.events.type(await view.findByLabelText('Title'), 'Test Ticket')
- await view.events.click(
- await view.findByRole('button', { name: 'Discard Changes' }),
- )
- const dialog = await view.findByRole('dialog', {
- name: 'Unsaved Changes',
- })
- const dialogView = within(dialog)
- await view.events.click(
- dialogView.getByRole('button', { name: 'Cancel & Go Back' }),
- )
- expect(view.getByText('Test Ticket')).toBeInTheDocument()
- })
- it('creates a new ticket', async () => {
- handleMockFormUpdaterQuery()
- const view = await visitView('/ticket/create')
- await view.events.type(await view.findByLabelText('Title'), 'Test Ticket')
- // Page title updates when title is set
- expect(
- await view.findByRole('heading', { level: 1, name: 'Test Ticket' }),
- ).toBeInTheDocument()
- // Page title defaults back when title is cleared
- await view.events.clear(view.getByLabelText('Title'))
- await waitFor(() =>
- expect(
- view.getByRole('heading', { level: 1, name: 'New Ticket' }),
- ).toBeInTheDocument(),
- )
- await view.events.type(view.getByLabelText('Title'), 'Test Ticket')
- // Customer field
- await handleCustomerMock(view)
- handleMockUserQuery()
- await view.events.click(
- view.getByRole('option', {
- name: 'Avatar (Nicole Braun) Nicole Braun – Zammad Foundation',
- }),
- )
- // Sidebar CUSTOMER
- expect(view.getByLabelText('Avatar (Nicole Braun)')).toBeInTheDocument()
- expect(view.getByText('Zammad Foundation')).toBeInTheDocument()
- expect(view.getByText('open tickets')).toBeInTheDocument()
- expect(view.getByText('nicole.braun@zammad.org')).toBeInTheDocument()
- expect(view.getByText('closed tickets')).toBeInTheDocument()
- expect(view.getByLabelText('Open tickets')).toHaveTextContent('17')
- // Sidebar Organization
- handleMockOrganizationQuery()
- await view.events.click(view.getByLabelText('Organization'))
- expect(view.getByText('Organization')).toBeInTheDocument()
- expect(view.getByText('Members')).toBeInTheDocument()
- expect(view.getByLabelText('Avatar (Nicole Braun)')).toBeInTheDocument()
- // Text field
- await view.events.type(
- view.getByRole('textbox', { name: 'Text' }),
- 'Test ticket text',
- )
- // Group field
- await view.events.click(view.getByLabelText('Group'))
- await view.events.click(view.getByRole('option', { name: 'Users' }))
- // State field
- await view.events.click(view.getByLabelText('Priority'))
- await view.events.click(view.getByRole('option', { name: '2 normal' }))
- // Priority Field
- await view.events.click(view.getByLabelText('State'))
- await view.events.click(
- view.getByRole('option', { name: 'pending reminder' }),
- )
- // Date selection Field on pending reminder
- await view.events.click(view.getByText('Pending till'))
- await waitFor(() => expect(view.getByRole('dialog')).toBeInTheDocument())
- const dateCells: Element[] = view.getAllByRole('gridcell', { name: '29' })
- await view.events.click(<Element>dateCells.at(-1))
- // Submission
- await view.events.click(view.getByRole('button', { name: 'Create' }))
- const calls = await waitForTicketCreateMutationCalls()
- expect(calls.at(-1)?.variables).toEqual({
- input: {
- article: {
- body: 'Test ticket text',
- cc: undefined,
- contentType: 'text/html',
- security: undefined,
- sender: 'Customer',
- type: 'phone',
- },
- customer: {
- id: 'gid://zammad/User/2',
- },
- groupId: 'gid://zammad/Group/1',
- objectAttributeValues: [],
- pendingTime: '2024-11-29T00:00:00.000Z',
- priorityId: 'gid://zammad/Ticket::Priority/2',
- stateId: 'gid://zammad/Ticket::State/3',
- title: 'Test Ticket',
- },
- })
- await waitFor(() =>
- expect(
- view.getByText('Ticket has been created successfully.'),
- ).toBeInTheDocument(),
- )
- })
- })
- describe('with customer permission', () => {
- beforeEach(() => {
- mockPermissions(['ticket.customer'])
- })
- describe('view disabled customer ticket create', async () => {
- beforeEach(() => {
- mockApplicationConfig({
- customer_ticket_create: false,
- })
- mockPermissions(['ticket.customer'])
- })
- it('redirects to error page', async () => {
- const view = await visitView('/ticket/create')
- expect(view.getByText('Not Found')).toBeInTheDocument()
- expect(view.getByText("This page doesn't exist.")).toBeInTheDocument()
- })
- })
- describe('view enabled customer ticket create', async () => {
- beforeEach(() => {
- mockApplicationConfig({
- customer_ticket_create: true,
- })
- })
- it('creates a new ticket', async () => {
- // Mock frontend attributes for customer context.
- // TODO: check if we can mock the query twice based on the variable?
- mockObjectManagerFrontendAttributesQuery({
- objectManagerFrontendAttributes: ticketCustomerObjectAttributes(),
- })
- handleMockFormUpdaterQuery()
- const view = await visitView('/ticket/create')
- await view.events.type(
- await view.findByLabelText('Title'),
- 'Test Customer Ticket',
- )
- // Text field
- await view.events.type(
- view.getByRole('textbox', { name: 'Text' }),
- 'Test customer ticket text',
- )
- await view.events.click(view.getByLabelText('Group'))
- await view.events.click(view.getByRole('option', { name: 'Users' }))
- // Submission
- await view.events.click(view.getByRole('button', { name: 'Create' }))
- const calls = await waitForTicketCreateMutationCalls()
- expect(calls.at(-1)?.variables).toEqual({
- input: {
- article: {
- body: 'Test customer ticket text',
- cc: undefined,
- contentType: 'text/html',
- security: undefined,
- sender: 'Customer',
- type: 'web',
- },
- customer: undefined,
- groupId: 'gid://zammad/Group/1',
- objectAttributeValues: [],
- stateId: 'gid://zammad/Ticket::State/2',
- title: 'Test Customer Ticket',
- },
- })
- await waitFor(() =>
- expect(
- view.getByText('Ticket has been created successfully.'),
- ).toBeInTheDocument(),
- )
- })
- })
- })
- })
|