// Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ import { defineComponent } from 'vue' import type { ExtendedRenderResult } from '#tests/support/components/index.ts' import { visitView } from '#tests/support/components/visitView.ts' import { mockGraphQLApi } from '#tests/support/mock-graphql-api.ts' import { mockUserCurrent } from '#tests/support/mock-userCurrent.ts' import { UserCurrentAvatarAddDocument } from '#shared/entities/user/current/graphql/mutations/userCurrentAvatarAdd.api.ts' import { UserCurrentAvatarDeleteDocument } from '#shared/entities/user/current/graphql/mutations/userCurrentAvatarDelete.api.ts' import { UserCurrentAvatarActiveDocument } from '../graphql/queries/userCurrentAvatarActive.api.ts' vi.mock('vue-advanced-cropper', () => { const Cropper = defineComponent({ emits: ['change'], mounted() { this.$emit('change', { canvas: { toDataURL() { return 'cropped image url' }, }, }) }, template: '
', }) return { Cropper, } }) const mockAvatarImage = '' const getAvatarObject = (deletable: boolean) => { return { id: 'Z2lkOi8vemFtbWFkL0F2YXRhci8yNA', default: true, deletable, initial: false, imageFull: mockAvatarImage, imageResize: mockAvatarImage, imageHash: '123456', createdAt: '2022-07-12T06:54:45Z', updatedAt: '2022-07-12T06:54:45Z', } } const mockActiveAvatar = async (deletable = true) => { mockGraphQLApi(UserCurrentAvatarActiveDocument).willResolve({ userCurrentAvatarActive: getAvatarObject(deletable), }) } const mockAddAvatar = async () => { mockGraphQLApi(UserCurrentAvatarAddDocument).willResolve({ userCurrentAvatarAdd: { avatar: getAvatarObject(true), errors: null, }, }) } const mockDeleteAvatar = async () => { mockGraphQLApi(UserCurrentAvatarDeleteDocument).willResolve({ userCurrentAvatarDelete: { success: true, errors: null, }, }) } const checkShownAvatar = async (view: ExtendedRenderResult, image: string) => { const waitForAvatar = await view.findByTestId('common-avatar') expect(waitForAvatar).toHaveStyle({ 'background-image': image, }) } const uploadFile = async (view: ExtendedRenderResult, testFlag: string) => { expect(view.queryByText('Save')).not.toBeInTheDocument() const file = new File([], 'test.jpg', { type: 'image/jpeg' }) await view.events.upload(view.getByTestId(testFlag), file) const saveButton = await view.findByText('Save') expect(saveButton).toBeVisible() await view.events.click(saveButton) } const removeAvatar = async (view: ExtendedRenderResult) => { await view.events.click(view.getByText('Delete')) await view.findByText('Delete avatar') await view.events.click(view.getByText('Delete avatar')) await checkShownAvatar(view, '') } describe('editing avatar', () => { beforeEach(() => { mockUserCurrent({ firstname: 'John', lastname: 'Doe', }) }) afterEach(() => { vi.spyOn(console, 'log').mockRestore() }) it('shows the avatar', async () => { mockActiveAvatar() const view = await visitView('/account/avatar') await checkShownAvatar(view, `url(${mockAvatarImage})`) }) it('can remove avatar', async () => { mockActiveAvatar() mockDeleteAvatar() const view = await visitView('/account/avatar') await view.findByText('Delete') await checkShownAvatar(view, `url(${mockAvatarImage})`) await removeAvatar(view) const avatar = await view.findByTestId('common-avatar') expect(avatar).toHaveTextContent('JD') }) it('can not remove undeletable avatars', async () => { mockActiveAvatar(false) mockDeleteAvatar() const view = await visitView('/account/avatar') await view.findByTestId('common-avatar') const deleteButton = await view.findByRole('button', { name: 'Delete' }) expect(deleteButton).toHaveAttribute('disabled') }) it('can upload image from camera', async () => { mockAddAvatar() mockActiveAvatar() const view = await visitView('/account/avatar') await uploadFile(view, 'fileCameraInput') await checkShownAvatar(view, `url(${mockAvatarImage})`) }) it('can upload image from gallery', async () => { mockAddAvatar() mockActiveAvatar() const view = await visitView('/account/avatar') await uploadFile(view, 'fileGalleryInput') await checkShownAvatar(view, `url(${mockAvatarImage})`) }) it('even after deleting I can upload an image', async () => { mockAddAvatar() mockActiveAvatar() mockDeleteAvatar() const view = await visitView('/account/avatar') await view.findByTestId('common-avatar') await checkShownAvatar(view, `url(${mockAvatarImage})`) await removeAvatar(view) await uploadFile(view, 'fileGalleryInput') await checkShownAvatar(view, `url(${mockAvatarImage})`) }) it('after selecting image I can cancel the cropping', async () => { mockActiveAvatar() const view = await visitView('/account/avatar') const file = new File([], 'test.jpg', { type: 'image/jpeg' }) await view.events.upload(view.getByTestId('fileGalleryInput'), file) const cancelButton = await view.findByText('Cancel') expect(cancelButton).toBeInTheDocument() await view.events.click(view.getByText('Cancel')) await checkShownAvatar(view, `url(${mockAvatarImage})`) }) })