123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256 |
- // 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 { mockUserCurrent } from '#tests/support/mock-userCurrent.ts'
- import { waitForNextTick } from '#tests/support/utils.ts'
- import { mockFormUpdaterQuery } from '#shared/components/Form/graphql/queries/formUpdater.mocks.ts'
- import { mockCurrentUserQuery } from '#shared/graphql/queries/currentUser.mocks.ts'
- import {
- EnumFormUpdaterId,
- EnumNotificationSoundFile,
- } from '#shared/graphql/types.ts'
- import { convertToGraphQLId } from '#shared/graphql/utils.ts'
- import { waitForUserCurrentNotificationPreferencesResetMutationCalls } from '#desktop/pages/personal-setting/graphql/mutations/userCurrentNotificationPreferencesReset.mocks.ts'
- import {
- mockUserCurrentNotificationPreferencesUpdateMutation,
- waitForUserCurrentNotificationPreferencesUpdateMutationCalls,
- } from '#desktop/pages/personal-setting/graphql/mutations/userCurrentNotificationPreferencesUpdate.mocks.ts'
- const mockPersonalSettings = (withGroups = true) => {
- return {
- personalSettings: {
- notificationConfig: {
- groupIds: withGroups ? [1, 2] : undefined,
- matrix: {
- create: {
- criteria: {
- ownedByMe: true,
- ownedByNobody: true,
- subscribed: true,
- no: false,
- },
- channel: { email: true, online: true },
- },
- update: {
- criteria: {
- ownedByMe: true,
- ownedByNobody: true,
- subscribed: true,
- no: false,
- },
- channel: { email: true, online: true },
- },
- reminderReached: {
- criteria: {
- ownedByMe: true,
- ownedByNobody: false,
- subscribed: false,
- no: false,
- },
- channel: { email: true, online: true },
- },
- escalation: {
- criteria: {
- ownedByMe: true,
- ownedByNobody: false,
- subscribed: false,
- no: false,
- },
- channel: { email: true, online: true },
- },
- },
- },
- notificationSound: {
- enabled: true,
- file: EnumNotificationSoundFile.Xylo,
- },
- },
- }
- }
- const mockUser = () => ({
- firstname: 'John',
- lastname: 'Doe',
- })
- describe('personal notifications settings', () => {
- beforeEach(() => {
- mockPermissions(['user_preferences.notifications'])
- mockFormUpdaterQuery({
- formUpdater: {
- fields: {
- EnumFormUpdaterId:
- EnumFormUpdaterId.FormUpdaterUpdaterUserNotifications,
- group_ids: {
- options: [
- {
- label: 'Testers Group',
- value: 1,
- },
- {
- label: 'Developers Group',
- value: 2,
- },
- ],
- },
- },
- },
- })
- })
- it('renders view correctly', async () => {
- mockUserCurrent({
- ...mockUser(),
- ...mockPersonalSettings(),
- })
- await waitForNextTick()
- const view = await visitView('personal-setting/notifications')
- // TABLE
- expect(view.getByText('New ticket')).toBeInTheDocument()
- expect(view.getByText('Ticket update')).toBeInTheDocument()
- expect(view.getByText('Ticket reminder reached')).toBeInTheDocument()
- expect(view.getByText('Ticket escalation')).toBeInTheDocument()
- expect(view.getByText('Name')).toBeInTheDocument()
- expect(view.getByText('My tickets')).toBeInTheDocument()
- expect(view.getByText('Not assigned')).toBeInTheDocument()
- expect(view.getByText('All tickets')).toBeInTheDocument()
- expect(view.getByText('Also notify via email')).toBeInTheDocument()
- expect(view.getByText('Developers Group')).toBeInTheDocument()
- expect(view.getByText('Testers Group')).toBeInTheDocument()
- const checkboxes = view.getAllByRole('checkbox')
- expect(checkboxes).toHaveLength(20)
- // Fields
- expect(view.getByText('Notification sound')).toBeInTheDocument()
- expect(
- view.getByText('Limit notifications to specific groups'),
- ).toBeInTheDocument()
- expect(
- view.getByText('Play user interface sound effects'),
- ).toBeInTheDocument()
- })
- it("doesn't render user groups if groups are not provided", async () => {
- mockUserCurrent({
- ...mockUser(),
- ...mockPersonalSettings(false),
- })
- const view = await visitView('personal-setting/notifications')
- expect(view.queryByText('User groups')).not.toBeInTheDocument()
- })
- it('resets values to default', async () => {
- mockUserCurrent({
- ...mockUser(),
- ...mockPersonalSettings(),
- })
- const playSound = vi.fn()
- // Set up a mock for playing sound effects in a test environment
- // This is necessary because the audio API is not available in the test environment
- window.HTMLMediaElement.prototype.play = () => playSound()
- const view = await visitView('personal-setting/notifications')
- await view.events.click(view.getByLabelText('Notification sound'))
- await view.events.click(view.getByText('Plop'))
- const checkboxes = view.getAllByTestId('checkbox-label')
- await view.events.click(checkboxes.at(-1)!)
- expect(
- (view.getByLabelText('Notification sound') as HTMLInputElement).value,
- ).toEqual('Plop')
- await view.events.click(
- view.getByRole('button', { name: 'Reset to Default Settings' }),
- )
- expect(
- await view.findByRole('dialog', { name: 'Confirmation' }),
- ).toBeInTheDocument()
- expect(
- view.getByText(
- 'Are you sure? Your notifications settings will be reset to default.',
- ),
- ).toBeInTheDocument()
- await view.events.click(view.getByRole('button', { name: 'Yes' }))
- const mocks =
- await waitForUserCurrentNotificationPreferencesResetMutationCalls()
- expect(mocks.at(-1)?.variables).toEqual({})
- mockCurrentUserQuery({
- currentUser: {
- ...mockUser(),
- ...mockPersonalSettings(),
- },
- })
- await waitForNextTick()
- expect(playSound).toHaveBeenCalled()
- expect(checkboxes.at(-1)).toBeEnabled()
- })
- it('submits notification form successfully', async () => {
- mockUserCurrent({
- ...mockUser(),
- ...mockPersonalSettings(),
- })
- const view = await visitView('personal-setting/notifications')
- const checkboxes = view.getAllByTestId('checkbox-label')
- await view.events.click(checkboxes.at(-1)!)
- mockUserCurrentNotificationPreferencesUpdateMutation({
- userCurrentNotificationPreferencesUpdate: {
- user: {
- ...mockUser(),
- ...mockPersonalSettings(),
- },
- errors: null,
- },
- })
- await view.events.click(
- view.getByRole('button', { name: 'Save Notifications' }),
- )
- const previousMockedData = mockPersonalSettings()
- // Convert group ids to GraphQL ids to match payload format
- // eslint-disable-next-line @typescript-eslint/ban-ts-comment
- // @ts-expect-error
- previousMockedData.personalSettings.notificationConfig.groupIds =
- previousMockedData.personalSettings.notificationConfig?.groupIds?.map(
- (id) => convertToGraphQLId('Group', id),
- )
- const mocks =
- await waitForUserCurrentNotificationPreferencesUpdateMutationCalls()
- // Last checkbox in table got updated
- expect(mocks.at(-1)?.variables).not.toEqual(previousMockedData)
- })
- })
|