123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298 |
- <!-- Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ -->
- <script setup lang="ts">
- import { storeToRefs } from 'pinia'
- import { reactive, shallowRef, watch, ref, computed } from 'vue'
- import { Cropper, type CropperResult } from 'vue-advanced-cropper'
- import { useRouter } from 'vue-router'
- import 'vue-advanced-cropper/dist/style.css'
- import CommonAvatar from '#shared/components/CommonAvatar/CommonAvatar.vue'
- import {
- useNotifications,
- NotificationTypes,
- } from '#shared/components/CommonNotifications/index.ts'
- import CommonUserAvatar from '#shared/components/CommonUserAvatar/CommonUserAvatar.vue'
- import { useConfirmation } from '#shared/composables/useConfirmation.ts'
- import { useUserCurrentAvatarAddMutation } from '#shared/entities/user/current/graphql/mutations/userCurrentAvatarAdd.api.ts'
- import { useUserCurrentAvatarDeleteMutation } from '#shared/entities/user/current/graphql/mutations/userCurrentAvatarDelete.api.ts'
- import type UserError from '#shared/errors/UserError.ts'
- import type { UserCurrentAvatarActiveQuery } from '#shared/graphql/types.ts'
- import {
- MutationHandler,
- QueryHandler,
- } from '#shared/server/apollo/handler/index.ts'
- import { useSessionStore } from '#shared/stores/session.ts'
- import type { ImageFileData } from '#shared/utils/files.ts'
- import {
- convertFileList,
- allowedImageTypesString,
- } from '#shared/utils/files.ts'
- import CommonButton from '#mobile/components/CommonButton/CommonButton.vue'
- import CommonButtonGroup from '#mobile/components/CommonButtonGroup/CommonButtonGroup.vue'
- import type { CommonButtonOption } from '#mobile/components/CommonButtonGroup/types.ts'
- import CommonLoader from '#mobile/components/CommonLoader/CommonLoader.vue'
- import { useHeader } from '#mobile/composables/useHeader.ts'
- import { useUserCurrentAvatarActiveQuery } from '../graphql/queries/userCurrentAvatarActive.api.ts'
- const router = useRouter()
- const fileCameraInput = shallowRef<HTMLInputElement>()
- const fileGalleryInput = shallowRef<HTMLInputElement>()
- const avatarImage = shallowRef<ImageFileData>()
- const activeAvatarQuery = new QueryHandler(useUserCurrentAvatarActiveQuery(), {
- errorNotificationMessage: __('The avatar could not be fetched.'),
- })
- const activeAvatar =
- ref<UserCurrentAvatarActiveQuery['userCurrentAvatarActive']>()
- activeAvatarQuery.watchOnResult((data) => {
- activeAvatar.value = data?.userCurrentAvatarActive
- })
- const avatarLoading = activeAvatarQuery.loading()
- const state = reactive({
- resizedImage: activeAvatar.value?.imageResize || '',
- })
- watch(activeAvatar, (newValue) => {
- state.resizedImage = newValue?.imageResize || ''
- })
- const { user } = storeToRefs(useSessionStore())
- const avatarDeleteDisabled = computed(() => {
- return !activeAvatar.value?.deletable
- })
- const addAvatar = () => {
- if (!state.resizedImage) return
- if (!avatarImage.value) return
- const addAvatarMutation = new MutationHandler(
- useUserCurrentAvatarAddMutation({
- variables: {
- images: {
- original: avatarImage.value,
- resized: {
- name: 'resized_avatar.png',
- type: 'image/png',
- content: state.resizedImage,
- },
- },
- },
- }),
- {
- errorNotificationMessage: __('The avatar could not be uploaded.'),
- },
- )
- const { notify, clearAllNotifications } = useNotifications()
- // Clear notifications to avoid duplicated error messages.
- clearAllNotifications()
- addAvatarMutation
- .send()
- .then((data) => {
- if (data?.userCurrentAvatarAdd?.avatar) {
- activeAvatar.value = data.userCurrentAvatarAdd.avatar
- avatarImage.value = undefined
- if (user.value) {
- user.value.image = data.userCurrentAvatarAdd.avatar.imageHash
- }
- }
- })
- .catch((errors: UserError) => {
- notify({
- id: 'avatar-add-error',
- message: errors.generalErrors[0],
- type: NotificationTypes.Error,
- })
- })
- }
- const canRemoveAvatar = () => {
- if (!user.value) return false
- if (!activeAvatar.value?.id) return false
- if (!activeAvatar.value?.deletable) return false
- return true
- }
- const removeAvatar = () => {
- if (!canRemoveAvatar()) return
- if (!activeAvatar.value?.id) return
- const removeAvatarMutation = new MutationHandler(
- useUserCurrentAvatarDeleteMutation({
- variables: { id: activeAvatar.value.id },
- }),
- {
- errorNotificationMessage: __('The avatar could not be deleted.'),
- },
- )
- removeAvatarMutation.send().then((data) => {
- if (data?.userCurrentAvatarDelete?.success) {
- state.resizedImage = ''
- avatarImage.value = undefined
- activeAvatar.value = undefined
- // reset image value in user store
- if (user.value) {
- user.value.image = undefined
- }
- }
- })
- }
- const { waitForConfirmation } = useConfirmation()
- const confirmRemoveAvatar = async () => {
- if (!canRemoveAvatar()) return
- const confirmed = await waitForConfirmation(
- __('Do you really want to delete your current avatar?'),
- {
- buttonLabel: __('Delete avatar'),
- buttonVariant: 'danger',
- },
- )
- if (confirmed) removeAvatar()
- }
- const saveButtonActive = computed(() => {
- if (state.resizedImage && avatarImage.value) return true
- return false
- })
- useHeader({
- title: __('Avatar'),
- backUrl: '/account',
- actionTitle: __('Done'),
- backIgnore: ['/user/current/avatar'],
- refetch: computed(
- () => avatarLoading.value && !!activeAvatarQuery.result().value,
- ),
- onAction() {
- router.push('/account')
- },
- })
- const loadAvatar = async (input?: HTMLInputElement) => {
- const files = input?.files
- if (!files) return
- const [avatar] = await convertFileList(files)
- avatarImage.value = avatar
- // Reset input value to allow selecting the same file again
- input.value = ''
- }
- const imageCropped = (crop: CropperResult) => {
- if (!crop.canvas) return
- state.resizedImage = crop.canvas.toDataURL('image/png')
- }
- const cancelCropping = () => {
- avatarImage.value = undefined
- state.resizedImage = activeAvatar.value?.imageResize || ''
- }
- const actions = computed<CommonButtonOption[]>(() => [
- {
- label: __('Library'),
- icon: 'photos',
- value: 'library',
- onAction: () => fileGalleryInput.value?.click(),
- },
- {
- label: __('Camera'),
- icon: 'camera',
- value: 'camera',
- onAction: () => fileCameraInput.value?.click(),
- },
- {
- label: __('Delete'),
- icon: 'delete',
- value: 'delete',
- disabled: avatarDeleteDisabled.value,
- class: 'bg-red-dark !text-red-bright',
- onAction: confirmRemoveAvatar,
- },
- ])
- </script>
- <template>
- <div v-if="user" class="px-4">
- <div class="flex flex-col items-center py-6">
- <CommonLoader
- :loading="avatarLoading && !activeAvatarQuery.result().value"
- >
- <CommonAvatar
- v-if="state.resizedImage"
- :image="state.resizedImage"
- size="xl"
- />
- <CommonUserAvatar v-else :entity="user" size="xl" personal />
- <CommonButtonGroup class="mt-6" mode="full" :options="actions" />
- </CommonLoader>
- <input
- ref="fileGalleryInput"
- data-test-id="fileGalleryInput"
- type="file"
- class="hidden"
- aria-hidden="true"
- :accept="allowedImageTypesString()"
- @change="loadAvatar(fileGalleryInput)"
- />
- <input
- ref="fileCameraInput"
- data-test-id="fileCameraInput"
- type="file"
- class="hidden"
- aria-hidden="true"
- :accept="allowedImageTypesString()"
- capture="user"
- @change="loadAvatar(fileCameraInput)"
- />
- <div
- v-if="avatarImage"
- class="flex w-full flex-col items-center justify-center"
- >
- <Cropper
- class="mb-4 mt-4 !max-h-[250px] !max-w-[400px]"
- :src="avatarImage.content"
- :stencil-props="{
- aspectRatio: 1,
- }"
- :transitions="false"
- background-class="!bg-black"
- foreground-class="!bg-black"
- @change="imageCropped"
- />
- <div class="flex w-full gap-2">
- <CommonButton class="h-10 flex-1" @click="cancelCropping">
- {{ $t('Cancel') }}
- </CommonButton>
- <CommonButton
- variant="primary"
- :disabled="!saveButtonActive"
- class="h-10 flex-1"
- @click="addAvatar"
- >
- {{ $t('Save') }}
- </CommonButton>
- </div>
- </div>
- </div>
- </div>
- </template>
|