123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199 |
- // 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: '<div></div>',
- })
- 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})`)
- })
- })
|