koel/resources/assets/js/components/profile-preferences/EditableProfileAvatar.vue

154 lines
3.4 KiB
Vue
Raw Normal View History

2024-03-19 22:48:12 +00:00
<template>
<div class="avatar">
<UserAvatar v-if="profile.avatar" :user="profile" style="width: var(--w)" />
<div class="buttons">
<button class="upload" title="Pick a new avatar" type="button" @click.prevent="openFileDialog">
2024-03-19 22:48:12 +00:00
<Icon :icon="faUpload" />
</button>
<button v-if="avatarChanged" class="reset" title="Reset avatar" type="button" @click.prevent="resetAvatar">
2024-03-19 22:48:12 +00:00
<Icon :icon="faRefresh" />
</button>
<button v-else class="remove" title="Remove avatar" type="button" @click.prevent="removeAvatar">
2024-03-19 22:48:12 +00:00
<Icon :icon="faTimes" />
</button>
</div>
<ImageCropper v-if="cropperSource" :source="cropperSource" @cancel="onCancel" @crop="onCrop" />
2024-03-19 22:48:12 +00:00
</div>
</template>
<script setup lang="ts">
import {
faRefresh,
faTimes,
faUpload
} from '@fortawesome/free-solid-svg-icons'
import 'vue-advanced-cropper/dist/style.css'
import { computed, ref, toRefs } from 'vue'
import { useFileDialog } from '@vueuse/core'
import { userStore } from '@/stores'
import { useFileReader } from '@/composables'
import { gravatar } from '@/utils'
import UserAvatar from '@/components/user/UserAvatar.vue'
import ImageCropper from '@/components/utils/ImageCropper.vue'
2024-03-19 22:48:12 +00:00
const props = defineProps<{ profile: Pick<User, 'name' | 'avatar'> }>()
const { profile } = toRefs(props)
const { open: openFileDialog, onChange, reset } = useFileDialog({
2024-03-19 22:48:12 +00:00
accept: 'image/*',
multiple: false,
reset: true
2024-03-19 22:48:12 +00:00
})
const cropperSource = ref<string | null>(null)
onChange(files => {
if (!files?.length) {
profile.value.avatar = userStore.current.avatar
cropperSource.value = null
return
}
useFileReader().readAsDataUrl(files[0], url => {
cropperSource.value = url
})
})
const removeAvatar = () => profile.value.avatar = gravatar(userStore.current.email)
const resetAvatar = () => {
profile.value.avatar = userStore.current.avatar
cropperSource.value = null
}
const avatarChanged = computed(() => profile.value.avatar !== userStore.current.avatar)
const onCrop = (result: string) => {
profile.value.avatar = result
2024-03-19 22:48:12 +00:00
cropperSource.value = null
}
const onCancel = () => (cropperSource.value = null)
2024-03-19 22:48:12 +00:00
</script>
<style scoped lang="scss">
.avatar {
--w: 105px;
outline: rgba(255, 255, 255, .1) solid 3px;
margin-top: 2rem;
border-radius: 50%;
position: relative;
overflow: hidden;
background: rgba(0, 0, 0, .1);
aspect-ratio: 1 / 1;
width: var(--w);
.buttons {
position: absolute;
top: 0;
border-radius: 50%;
width: 100%;
aspect-ratio: 1 / 1;
display: flex;
place-items: center;
justify-content: center;
gap: .5rem;
padding-top: 50%;
opacity: 0;
transition: opacity .3s;
button {
background: rgba(0, 0, 0, .3);
width: 28px;
aspect-ratio: 1 / 1;
border-radius: 50%;
padding: 2px 4px;
&:hover {
background: rgba(0, 0, 0, .7);
}
}
&:hover {
opacity: 1;
}
}
.cropper-wrapper {
width: 100%;
height: 100%;
position: fixed;
top: 0;
left: 0;
display: flex;
align-items: center;
justify-content: center;
z-index: 99;
background: rgba(0, 0, 0, .5);
> div {
position: relative;
max-width: 100%;
max-height: 100%;
border-radius: 5px;
display: flex;
}
.controls {
position: fixed;
right: 1.5rem;
top: 1.5rem;
display: flex;
gap: .5rem;
flex: 1;
}
}
}
</style>