Polarcraft/webv2/components/team/Default.vue
Xeovalyte fbcd0d1bb9
Some checks failed
Build and Deploy / Deploy Web (push) Failing after 1s
Build and Deploy / Deploy Discord Bot (push) Failing after 1s
feat: Added team page
2023-06-24 13:27:33 +02:00

198 lines
6.2 KiB
Vue

<template>
<div v-if="user.team" class="grid w-full grid-cols-12 gap-6">
<UModal v-model="isInviteModalOpen">
<UCard>
<template #header>
<h1 class="text-lg font-bold">Invite User</h1>
</template>
<div class="flex flex-col items-center divide-y divide-gray-700">
<div v-for="newUser in users" :key="newUser._id" class="flex w-full max-w-sm py-2">
<span class="mr-auto">
{{ newUser.username }}
</span>
<span v-if="newUser.team" class="text-gray-500">Already in Team</span>
<span v-if="!newUser.team && newUser.teamInvites.includes(team._id)" class="text-gray-500">Already invited</span>
<UButton v-if="!newUser.team && !newUser.teamInvites.includes(team._id)" :disabled="disableButtons" @click="inviteUser(newUser)">Invite</UButton>
</div>
</div>
<template #footer>
<UButton :disabled="disableButtons" @click="isInviteModalOpen = false">OK</UButton>
</template>
</UCard>
</UModal>
<UModal v-model="editModal.open">
<UCard>
<template #header>
<h1 class="text-lg font-bold">Edit Team</h1>
</template>
<form class="flex flex-col gap-y-5">
<UFormGroup name="name" label="Name" :error="teamNameError">
<UInput v-model="editModal.name" placeholder=" " />
</UFormGroup>
<UFormGroup name="color" label="Color" :error="teamColorError">
<div class="flex">
<input v-model="editModal.color" type="color" class="mr-2">
<UInput v-model="editModal.color" placeholder=" " class="w-full" />
</div>
</UFormGroup>
</form>
<template #footer>
<UButton :disabled="disableButtons" @click="submitEditTeam">Edit Team</UButton>
</template>
</UCard>
</UModal>
<div class="col-span-6 h-min rounded-lg border border-gray-700 bg-gray-800 p-5">
<h3 class="mb-2 text-xl font-bold">
Team Information
</h3>
<ul>
<li>Name: {{ user.team.name }}</li>
<li class="relative">
Color: <span :style="{ 'text-decoration-color': user.team.color }" class="underline underline-offset-4">{{ user.team.color }}</span>
</li>
</ul>
<div class="mt-5 space-x-3">
<UButton @click="openModal">Edit Team</UButton>
<UButton color="red" variant="outline" @click="leaveTeam">Leave Team</UButton>
</div>
</div>
<div v-if="team" class="col-span-6 rounded-lg border border-gray-700 bg-gray-800 p-5">
<h3 class="mb-2 text-xl font-bold">
Team Members
</h3>
<ul class="divide-y divide-gray-700">
<li v-for="member in team.members" :key="member._id" class="flex items-center py-3 text-lg">
<UAvatarGroup size="sm" class="mr-3">
<UAvatar :src="discordAvatarUrl(member)" alt="Discord Avatar" placeholder="DC" />
<UAvatar v-if="minecraftAvatarUrl(member)" :src="minecraftAvatarUrl(member)" alt="Minecraft Avatar" placeholder="MC" />
</UAvatarGroup>
{{ member.username }}
</li>
</ul>
<div class="my-2 rounded border-2 border-dashed border-gray-500 px-5 py-2 text-center font-bold hover:cursor-pointer hover:bg-gray-700" @click="isInviteModalOpen = true">Invite User</div>
</div>
</div>
</template>
<script lang="ts" setup>
const disableButtons = ref(false)
const user = useState<IUser>('user')
const { data: team }: { data: any } = useFetch('/api/teams/@current')
const { data: users }: { data: any } = useFetch('/api/users')
const isInviteModalOpen = ref(false)
const editModal = ref({
open: false,
name: '',
color: ''
})
const teamNameError = computed(() => {
if (!editModal.value.name) {
return ''
}
if (!/^[a-zA-Z0-9]+$/.test(editModal.value.name)) {
return 'Team name must only include alfanumeric characters'
}
if (editModal.value.name.length < 3 || editModal.value.name.length > 16) {
return 'Team name must be between 3 and 16 characters'
}
})
const teamColorError = computed(() => {
if (!editModal.value.color) {
return ''
}
if (!/^#([A-Fa-f0-9]{6}|[A-Fa-f0-9]{3})$/.test(editModal.value.color)) {
return 'Team color is not a valid hex code'
}
})
const discordAvatarUrl = (member: IUser) => {
return 'https://cdn.discordapp.com/avatars/' + member.discord.id + '/' + member.discord.avatarHash + '.png'
}
const minecraftAvatarUrl = (member: IUser) => {
if (!member.minecraft) {
return ''
}
return 'https://api.mineatar.io/face/' + member.minecraft.uuid + '?scale=16'
}
const leaveTeam = async () => {
disableButtons.value = true
try {
await $fetch('/api/teams/@current/members/@me', {
method: 'DELETE'
})
user.value.team = undefined
useToast().add({ title: 'Successfully left team', color: 'green' })
} catch (e: any) {
console.error(e)
useToast().add({ title: e.statusMessage, color: 'red' })
}
disableButtons.value = false
}
const inviteUser = async (newUser: IUser) => {
disableButtons.value = true
try {
await $fetch(`/api/users/${newUser._id}/teamInvites`, {
method: 'POST',
body: {
teamId: team.value._id
}
})
newUser.teamInvites.push(team.value._id)
useToast().add({ title: 'Successfully invited user to team', color: 'green' })
} catch (e: any) {
console.error(e)
useToast().add({ title: e.statusMessage, color: 'red' })
}
disableButtons.value = false
}
const openModal = () => {
editModal.value.name = team.value.name
editModal.value.color = team.value.color
editModal.value.open = true
}
const submitEditTeam = async () => {
disableButtons.value = true
try {
const response: any = await $fetch('/api/teams/@current', {
method: 'PUT',
body: {
teamName: editModal.value.name,
teamColor: editModal.value.color
}
})
user.value.team = response
useToast().add({ title: 'Successfully edited team', color: 'green' })
} catch (e: any) {
console.error(e)
useToast().add({ title: e.statusMessage, color: 'red' })
}
editModal.value = { open: false, name: '', color: '' }
disableButtons.value = false
}
</script>