yunohost-portal/pages/index.vue
2023-07-19 18:42:05 +02:00

105 lines
3 KiB
Vue

<script setup lang="ts">
const apiEndpoint = useApiEndpoint()
const isLoggedIn = useIsLoggedIn()
let me = {}
const { data, error } = await useFetch(apiEndpoint + '/me', {
credentials: 'include',
})
if (error.value && error.value.statusCode >= 400) {
isLoggedIn.value = false // FIXME : not confident this actually mutates the state ...
// FIXME : we probably want different handlings between 401/403, 500, 502, ...
await navigateTo('/login')
} else {
me = data.value
Object.keys(me.apps).forEach((appId) => {
const appTileColors = [
'red',
'orange',
'yellow',
'lime',
'green',
'teal',
'indigo',
'sky',
'purple',
'rose',
]
const randomColorNumber =
parseInt(me.apps[appId].label, 36) % appTileColors.length
me.apps[appId].color = appTileColors[randomColorNumber]
})
}
async function logout() {
const { error } = await useFetch(apiEndpoint + '/logout', {
method: 'GET',
credentials: 'include',
})
if (error.value && error.value.statusCode !== 200) {
// FIXME : display an error or something
} else {
// FIXME : meh, turns out the cookie is still valid after successfully calling the route for some reason ... !?
isLoggedIn.value = false // FIXME : not confident this actually mutates the state ...
await navigateTo('/login')
}
}
</script>
<template>
<div>
<div class="flex flex-row items-center min-w-full">
<span class="flex-none pr-5">
<Icon name="mdi:account-circle" size="5em" class="text-gray-500" />
</span>
<div class="grow">
<h2 class="text-2xl font-extrabold leading-none tracking-tight">
{{ me.username }}
</h2>
<h3>{{ me.fullname }}</h3>
<h4 class="opacity-50">{{ me.mail }}</h4>
</div>
<div class="flex-none">
<button class="btn bg-gray-800" @click.prevent="logout">
<Icon name="mdi:logout" class="text-gray-500" />
Logout
</button>
</div>
</div>
<div id="apps" class="p-10">
<div v-if="Object.keys(me.apps).length == 0">
<em class="text-gray-400"
>There is no app to list here, either because no web app yet is
installed on the server, or because you don't have access to any.
Please check with the admins of the server for more infos!</em
>
</div>
<ul class="flex space-x-4">
<!-- NB : because of the usage of dynamic colors, gotta force tailwind to expose those, cf 'safelisting' -->
<li
v-for="app in me.apps"
:key="app.id"
:class="
'text-center leading-none p-5 card h-32 w-32 bg-' +
app.color +
'-500'
"
>
<a>
<div class="text-6xl font-extrabold">
{{ app.label.substring(0, 2) }}
</div>
<span class="leading-tight">{{ app.label }}</span>
</a>
</li>
</ul>
</div>
</div>
</template>