123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import { visitView } from '#tests/support/components/visitView.ts'
- import { mockPermissions } from '#tests/support/mock-permissions.ts'
- import { getCurrentUserUpdatesSubscriptionHandler } from '#shared/graphql/subscriptions/currentUserUpdates.mocks.ts'
- import { useSessionStore } from '#shared/stores/session.ts'
- import {
- mockUserCurrentCalendarSubscriptionUpdate,
- waitForUserCurrentCalendarSubscriptionUpdateCalls,
- } from '../graphql/mutations/userCurrentCalendarSubscriptionUpdate.mocks.ts'
- import { mockUserCurrentCalendarSubscriptionList } from '../graphql/queries/userCurrentCalendarSubscriptionList.mocks.ts'
- describe('personal calendar subscription settings', () => {
- beforeEach(() => {
- mockPermissions(['user_preferences.calendar+ticket.agent'])
- mockUserCurrentCalendarSubscriptionList({
- userCurrentCalendarSubscriptionList: {
- combinedUrl: 'https://zammad.example.com/ical/tickets',
- globalOptions: {
- alarm: false,
- },
- escalation: {
- url: 'https://zammad.example.com/ical/tickets/escalation',
- options: {
- own: true,
- notAssigned: false,
- },
- },
- newOpen: {
- url: 'https://zammad.example.com/ical/tickets/new_open',
- options: {
- own: false,
- notAssigned: true,
- },
- },
- pending: {
- url: 'https://zammad.example.com/ical/tickets/pending',
- options: {
- own: true,
- notAssigned: true,
- },
- },
- },
- })
- })
- it('renders view correctly', async () => {
- const view = await visitView('personal-setting/calendar-subscriptions')
- expect(view.getByLabelText('Combined subscription URL')).toHaveValue(
- 'https://zammad.example.com/ical/tickets',
- )
- expect(
- view.getByLabelText(
- 'Add alarm to pending reminder and escalated tickets',
- ),
- ).toBeInTheDocument()
- expect(
- view.getByRole('tab', { name: 'Escalated Tickets' }),
- ).toBeInTheDocument()
- expect(
- view.getByRole('tab', { name: 'New & Open Tickets' }),
- ).toBeInTheDocument()
- expect(
- view.getByRole('tab', { name: 'Pending Tickets' }),
- ).toBeInTheDocument()
- })
- it('switches tab panels correctly', async () => {
- const view = await visitView('personal-setting/calendar-subscriptions')
- const escalationTab = view.getByRole('tab', { name: 'Escalated Tickets' })
- expect(escalationTab).toHaveAttribute('aria-selected', 'true')
- expect(view.getByLabelText('Direct subscription URL')).toHaveValue(
- 'https://zammad.example.com/ical/tickets/escalation',
- )
- expect(view.getAllByLabelText('My tickets')[0]).toBeChecked()
- expect(view.getAllByLabelText('Not assigned')[0]).not.toBeChecked()
- const newOpenTab = view.getByRole('tab', { name: 'New & Open Tickets' })
- await view.events.click(newOpenTab)
- expect(escalationTab).toHaveAttribute('aria-selected', 'false')
- expect(newOpenTab).toHaveAttribute('aria-selected', 'true')
- expect(view.getByLabelText('Direct subscription URL')).toHaveValue(
- 'https://zammad.example.com/ical/tickets/new_open',
- )
- expect(view.getAllByLabelText('My tickets')[1]).not.toBeChecked()
- expect(view.getAllByLabelText('Not assigned')[1]).toBeChecked()
- const pendingTab = view.getByRole('tab', { name: 'Pending Tickets' })
- await view.events.click(pendingTab)
- expect(newOpenTab).toHaveAttribute('aria-selected', 'false')
- expect(pendingTab).toHaveAttribute('aria-selected', 'true')
- expect(view.getByLabelText('Direct subscription URL')).toHaveValue(
- 'https://zammad.example.com/ical/tickets/pending',
- )
- expect(view.getAllByLabelText('My tickets')[2]).toBeChecked()
- expect(view.getAllByLabelText('Not assigned')[2]).toBeChecked()
- })
- it('updates calendar subscription when alarm is toggled', async () => {
- const view = await visitView('personal-setting/calendar-subscriptions')
- mockUserCurrentCalendarSubscriptionUpdate({
- userCurrentCalendarSubscriptionUpdate: {
- success: true,
- errors: null,
- },
- })
- await view.events.click(
- view.getByLabelText(
- 'Add alarm to pending reminder and escalated tickets',
- ),
- )
- const calls = await waitForUserCurrentCalendarSubscriptionUpdateCalls()
- expect(calls.at(-1)?.variables).toEqual({
- input: expect.objectContaining({
- alarm: true,
- }),
- })
- })
- it('updates calendar subscription when form is changed', async () => {
- const view = await visitView('personal-setting/calendar-subscriptions')
- mockUserCurrentCalendarSubscriptionUpdate({
- userCurrentCalendarSubscriptionUpdate: {
- success: true,
- errors: null,
- },
- })
- await view.events.click(view.getAllByLabelText('My tickets')[0])
- const calls = await waitForUserCurrentCalendarSubscriptionUpdateCalls()
- expect(calls.at(-1)?.variables).toEqual({
- input: expect.objectContaining({
- escalation: expect.objectContaining({
- own: false,
- }),
- }),
- })
- })
- it('updates calendar subscription when different tab is changed', async () => {
- const view = await visitView('personal-setting/calendar-subscriptions')
- await view.events.click(
- view.getByRole('tab', { name: 'New & Open Tickets' }),
- )
- mockUserCurrentCalendarSubscriptionUpdate({
- userCurrentCalendarSubscriptionUpdate: {
- success: true,
- errors: null,
- },
- })
- await view.events.click(view.getAllByLabelText('My tickets')[1])
- const calls = await waitForUserCurrentCalendarSubscriptionUpdateCalls()
- expect(calls.at(-1)?.variables).toEqual({
- input: expect.objectContaining({
- escalation: expect.objectContaining({
- own: true,
- }),
- }),
- })
- })
- it('resets state when session store is updated', async () => {
- const view = await visitView('personal-setting/calendar-subscriptions')
- // Mock opposite states than what was loaded with the initial request.
- mockUserCurrentCalendarSubscriptionList({
- userCurrentCalendarSubscriptionList: {
- combinedUrl: 'https://zammad.example.com/ical/tickets',
- globalOptions: {
- alarm: true,
- },
- escalation: {
- url: 'https://zammad.example.com/ical/tickets/escalation',
- options: {
- own: false,
- notAssigned: true,
- },
- },
- newOpen: {
- url: 'https://zammad.example.com/ical/tickets/new_open',
- options: {
- own: true,
- notAssigned: false,
- },
- },
- pending: {
- url: 'https://zammad.example.com/ical/tickets/pending',
- options: {
- own: false,
- notAssigned: false,
- },
- },
- },
- })
- // Trigger the current user query, so the user updates subscription is put in place.
- // Normally, this is not needed in the test environment, but here we depend on the mechanism, so we can trigger
- // the subscription located in the session store and in turn form updates.
- const { getCurrentUser } = useSessionStore()
- const user = await getCurrentUser()
- // Just trigger the subscription with the same data, new state will come from its own query anyway.
- await getCurrentUserUpdatesSubscriptionHandler().trigger({
- userUpdates: {
- user,
- },
- })
- expect(
- view.getByLabelText(
- 'Add alarm to pending reminder and escalated tickets',
- ),
- ).toBeChecked()
- expect(view.getAllByLabelText('My tickets')[0]).not.toBeChecked()
- expect(view.getAllByLabelText('Not assigned')[0]).toBeChecked()
- expect(view.getAllByLabelText('My tickets')[1]).toBeChecked()
- expect(view.getAllByLabelText('Not assigned')[1]).not.toBeChecked()
- expect(view.getAllByLabelText('My tickets')[2]).not.toBeChecked()
- expect(view.getAllByLabelText('Not assigned')[2]).not.toBeChecked()
- })
- })
|