mirror of
https://github.com/koel/koel
synced 2025-01-20 00:14:02 +00:00
95 lines
3.3 KiB
Vue
95 lines
3.3 KiB
Vue
<template>
|
|
<article
|
|
:class="{ me: isCurrentUser }"
|
|
class="apply p-4 flex items-center rounded-md bg-k-bg-secondary border border-k-border
|
|
gap-3 transition-[border-color] duration-200 ease-in-out hover:border-white/15"
|
|
>
|
|
<UserAvatar :user="user" width="48" />
|
|
|
|
<main class="flex flex-col justify-between relative flex-1 gap-1">
|
|
<h3 class="font-medium flex gap-2 items-center">
|
|
<span v-if="user.name" class="name">{{ user.name }}</span>
|
|
<span v-else class="name font-light text-k-text-secondary">Anonymous</span>
|
|
<Icon v-if="isCurrentUser" :icon="faCircleCheck" class="you text-k-highlight" title="This is you!" />
|
|
<Icon
|
|
v-if="user.is_admin"
|
|
:icon="faShield"
|
|
class="is-admin text-k-primary"
|
|
title="User has admin privileges"
|
|
/>
|
|
<img
|
|
v-if="user.sso_provider === 'Google'"
|
|
title="Google SSO"
|
|
:src="googleLogo"
|
|
alt="Google"
|
|
width="14"
|
|
height="14"
|
|
>
|
|
</h3>
|
|
|
|
<p class="text-k-text-secondary">{{ user.email }}</p>
|
|
</main>
|
|
|
|
<div class="space-x-2">
|
|
<template v-if="user.is_prospect">
|
|
<Btn class="btn-revoke" danger small @click="revokeInvite">Revoke</Btn>
|
|
</template>
|
|
<template v-else>
|
|
<Btn v-if="!user.is_prospect" highlight small @click="edit">
|
|
{{ isCurrentUser ? 'Your Profile' : 'Edit' }}
|
|
</Btn>
|
|
<Btn v-if="!isCurrentUser" danger small @click="destroy">Delete</Btn>
|
|
</template>
|
|
</div>
|
|
</article>
|
|
</template>
|
|
|
|
<script lang="ts" setup>
|
|
import googleLogo from '@/../img/logos/google.svg'
|
|
import { faCircleCheck, faShield } from '@fortawesome/free-solid-svg-icons'
|
|
import { computed, toRefs } from 'vue'
|
|
import { userStore } from '@/stores'
|
|
import { invitationService } from '@/services'
|
|
import { useAuthorization, useDialogBox, useMessageToaster, useRouter } from '@/composables'
|
|
import { eventBus, parseValidationError } from '@/utils'
|
|
|
|
import Btn from '@/components/ui/form/Btn.vue'
|
|
import UserAvatar from '@/components/user/UserAvatar.vue'
|
|
|
|
const props = defineProps<{ user: User }>()
|
|
const { user } = toRefs(props)
|
|
|
|
const { toastSuccess } = useMessageToaster()
|
|
const { showConfirmDialog, showErrorDialog } = useDialogBox()
|
|
const { go } = useRouter()
|
|
|
|
const { currentUser } = useAuthorization()
|
|
|
|
const isCurrentUser = computed(() => user.value.id === currentUser.value.id)
|
|
|
|
const edit = () => isCurrentUser.value ? go('profile') : eventBus.emit('MODAL_SHOW_EDIT_USER_FORM', user.value)
|
|
|
|
const destroy = async () => {
|
|
if (!await showConfirmDialog(`Unperson ${user.value.name}?`)) return
|
|
|
|
await userStore.destroy(user.value)
|
|
toastSuccess(`User "${user.value.name}" deleted.`)
|
|
}
|
|
|
|
const revokeInvite = async () => {
|
|
if (!await showConfirmDialog(`Revoke the invite for ${user.value.email}?`)) return
|
|
|
|
try {
|
|
await invitationService.revoke(user.value)
|
|
toastSuccess(`Invitation for ${user.value.email} revoked.`)
|
|
} catch (err: any) {
|
|
if (err.response.status === 404) {
|
|
showErrorDialog('Cannot revoke the invite. Maybe it has been accepted?', 'Revocation Failed')
|
|
return
|
|
}
|
|
|
|
const msg = err.response.status === 422 ? parseValidationError(err.response.data)[0] : 'Unknown error.'
|
|
showErrorDialog(msg, 'Error')
|
|
}
|
|
}
|
|
</script>
|