
* - polishing calendar * wip * wip * #74 - wip * #74 - wip * #74 - wip * #74 - fix icons Co-authored-by: EwelinaLasowy <ewelina.lasowy@blumilk.pl>
348 lines
13 KiB
Vue
348 lines
13 KiB
Vue
<template>
|
|
<InertiaHead title="Strona główna" />
|
|
<div class="grid grid-cols-1 gap-4 items-start lg:grid-cols-3 lg:gap-8">
|
|
<div class="grid grid-cols-1 gap-4 lg:col-span-2">
|
|
<section>
|
|
<div class=" bg-white overflow-hidden shadow">
|
|
<div class="bg-white p-6">
|
|
<div class="sm:flex sm:items-center sm:justify-between">
|
|
<div class="sm:flex sm:space-x-5">
|
|
<div class="flex-shrink-0">
|
|
<img
|
|
class="mx-auto h-20 w-20 rounded-full"
|
|
:src="user.avatar"
|
|
>
|
|
</div>
|
|
<div class="mt-4 text-center sm:mt-0 sm:pt-1 sm:text-left">
|
|
<p class="text-sm font-medium text-gray-600">
|
|
Cześć,
|
|
</p>
|
|
<p class="text-xl font-bold text-gray-900 sm:text-2xl">
|
|
{{ user.name }}
|
|
</p>
|
|
<p class="text-sm font-medium text-gray-600">
|
|
{{ user.role }}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
<section>
|
|
<div class="grid grid-cols-2 gap-4">
|
|
<div class="bg-white shadow-md p-4">
|
|
<VacationChart :stats="stats" />
|
|
</div>
|
|
<div class="h-full">
|
|
<div class="grid grid-cols-2 gap-4 h-full">
|
|
<div class="px-4 py-5 bg-white shadow-md sm:p-6">
|
|
<dd class="mt-1 text-4xl font-semibold text-blumilk-500">
|
|
{{ stats.remaining }}
|
|
</dd>
|
|
<dt class="text-md font-medium text-gray-700 truncate">
|
|
Pozostało
|
|
</dt>
|
|
<dt class="text-sm font-medium text-gray-500 mt-2">
|
|
Dni do wykorzystania teraz.
|
|
</dt>
|
|
</div>
|
|
<div class="px-4 py-5 bg-white shadow-md sm:p-6">
|
|
<dd class="mt-1 text-4xl font-semibold text-blumilk-700">
|
|
{{ stats.used }}
|
|
</dd>
|
|
<dt class="text-md font-medium text-gray-700 truncate">
|
|
Dni wykorzystane
|
|
</dt>
|
|
<dt class="text-sm font-medium text-gray-500 mt-2">
|
|
Dni, które zostały już wykorzystane na urlop wypoczynkowy.
|
|
</dt>
|
|
</div>
|
|
<div class="px-4 py-5 bg-white shadow-md sm:p-6">
|
|
<dt class="mt-1 text-4xl font-semibold text-blumilk-200">
|
|
{{ stats.pending }}
|
|
</dt>
|
|
<dd class="text-md font-medium text-gray-500 truncate">
|
|
Rozpatrywane
|
|
</dd>
|
|
<dt class="text-sm font-medium text-gray-500 mt-2">
|
|
Dni czekające na akceptację przełożonych.
|
|
</dt>
|
|
</div>
|
|
<div class="px-4 py-5 bg-white shadow-md sm:p-6">
|
|
<dt class="mt-1 text-4xl font-semibold text-gray-900">
|
|
{{ stats.limit }}
|
|
</dt>
|
|
<dd class="text-md font-medium text-gray-500 truncate">
|
|
Limit urlopu
|
|
</dd>
|
|
<dt class="text-sm font-medium text-gray-500 mt-2">
|
|
Twój roczny limit urlopu wypoczynkowego.
|
|
</dt>
|
|
</div>
|
|
<div class="px-4 py-5 bg-white shadow-md sm:p-6 col-span-2">
|
|
<dt class="mt-1 text-4xl font-semibold text-gray-900">
|
|
{{ stats.other }}
|
|
</dt>
|
|
<dd class="text-md font-medium text-gray-500 truncate">
|
|
Inne urlopy
|
|
</dd>
|
|
<dt class="text-sm font-medium text-gray-500 mt-2">
|
|
Urlopy bezpłatne, okolicznościowe, zwolnienia lekarskie, itd., które zostały już zatwierdzone.
|
|
</dt>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
</div>
|
|
<div class="grid grid-cols-1 gap-4">
|
|
<section v-if="can.listAllVacationRequests">
|
|
<div class="bg-white shadow-md">
|
|
<div class="p-4 sm:px-6">
|
|
<h2 class="text-lg leading-6 font-medium text-gray-900">
|
|
Wnioski oczekujące na akcje
|
|
</h2>
|
|
</div>
|
|
<div class="border-t border-gray-200 pb-5 px-4 sm:px-6">
|
|
<div class="flow-root mt-6">
|
|
<ul class="-my-5 divide-y divide-gray-200">
|
|
<li
|
|
v-for="request in vacationRequests.data"
|
|
:key="request.id"
|
|
class="py-5"
|
|
>
|
|
<div class="relative focus-within:ring-2 focus-within:ring-blumilk-500">
|
|
<h3 class="text-sm font-semibold text-blumilk-600 hover:text-blumilk-500">
|
|
<InertiaLink
|
|
:href="`/vacation-requests/${request.id}`"
|
|
class="hover:underline focus:outline-none"
|
|
>
|
|
<span class="absolute inset-0" />
|
|
Wniosek o {{ findType(request.type).text.toLowerCase() }}
|
|
[{{ request.name }}]
|
|
</InertiaLink>
|
|
</h3>
|
|
<p class="mt-1 text-sm text-gray-600">
|
|
{{ request.from }} - {{ request.to }}
|
|
</p>
|
|
<div class="mt-3 text-sm text-gray-600">
|
|
<div class="flex">
|
|
<img
|
|
class="h-10 w-10 rounded-full"
|
|
:src="request.user.avatar"
|
|
>
|
|
<div class="ml-3">
|
|
<p class="text-sm font-medium text-gray-900">
|
|
{{ request.user.name }}
|
|
</p>
|
|
<p class="text-sm text-gray-500">
|
|
{{ request.user.email }}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</li>
|
|
<li v-if="! vacationRequests.data.length">
|
|
<p class="py-2">
|
|
Brak danych
|
|
</p>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
<div class="mt-6">
|
|
<InertiaLink
|
|
href="/vacation-requests"
|
|
:data="{status: 'waiting_for_action'}"
|
|
class="w-full flex justify-center items-center px-4 py-2 border border-gray-300 shadow-sm text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-blumilk-500"
|
|
>
|
|
Zobacz wszystkie
|
|
</InertiaLink>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
<section v-else>
|
|
<div class="bg-white shadow-md">
|
|
<div class="p-4 sm:px-6">
|
|
<h2 class="text-lg leading-6 font-medium text-gray-900">
|
|
Twoje wnioski
|
|
</h2>
|
|
</div>
|
|
<div class="border-t border-gray-200 pb-5 px-4 sm:px-6">
|
|
<div class="flow-root mt-6">
|
|
<ul class="-my-5 divide-y divide-gray-200">
|
|
<li
|
|
v-for="request in vacationRequests.data"
|
|
:key="request.id"
|
|
class="py-5"
|
|
>
|
|
<div class="relative focus-within:ring-2 focus-within:ring-blumilk-500">
|
|
<h3 class="text-sm font-semibold text-blumilk-600 hover:text-blumilk-500">
|
|
<InertiaLink
|
|
:href="`/vacation-requests/${request.id}`"
|
|
class="hover:underline focus:outline-none"
|
|
>
|
|
<span class="absolute inset-0" />
|
|
Wniosek o {{ findType(request.type).text.toLowerCase() }}
|
|
[{{ request.name }}]
|
|
</InertiaLink>
|
|
</h3>
|
|
<p class="mt-1 text-sm text-gray-600">
|
|
{{ request.from }} - {{ request.to }}
|
|
</p>
|
|
<p class="mt-2 text-sm text-gray-600">
|
|
<Status :status="request.state" />
|
|
</p>
|
|
</div>
|
|
</li>
|
|
<li v-if="! vacationRequests.data.length">
|
|
<p class="py-2">
|
|
Brak danych
|
|
</p>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
<div class="mt-6">
|
|
<InertiaLink
|
|
href="/vacation-requests/me"
|
|
class="w-full flex justify-center items-center px-4 py-2 border border-gray-300 shadow-sm text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-blumilk-500"
|
|
>
|
|
Zobacz wszystkie
|
|
</InertiaLink>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
<section>
|
|
<div class="bg-white shadow-md">
|
|
<div class="p-4 sm:px-6">
|
|
<h2 class="text-lg leading-6 font-medium text-gray-900">
|
|
Dzisiejsze nieobecności
|
|
</h2>
|
|
</div>
|
|
<div class="border-t border-gray-200 px-4 sm:px-6">
|
|
<ul class="divide-y divide-gray-200">
|
|
<li
|
|
v-for="absence in absences.data"
|
|
:key="absence.user.id"
|
|
class="py-4 flex"
|
|
>
|
|
<img
|
|
class="h-10 w-10 rounded-full"
|
|
:src="absence.user.avatar"
|
|
>
|
|
<div class="ml-3">
|
|
<p class="text-sm font-medium text-gray-900">
|
|
{{ absence.user.name }}
|
|
</p>
|
|
<p class="text-sm text-gray-500">
|
|
{{ absence.user.email }}
|
|
</p>
|
|
</div>
|
|
</li>
|
|
<li v-if="! absences.data.length">
|
|
<p class="py-2">
|
|
Brak danych
|
|
</p>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
<section>
|
|
<div class="bg-white shadow-md">
|
|
<div>
|
|
<div class="p-4 sm:px-6">
|
|
<h2 class="text-lg leading-6 font-medium text-gray-900">
|
|
Najbliższe dni wolne
|
|
</h2>
|
|
</div>
|
|
<div class="border-t border-gray-200 px-4 pb-5 sm:px-6">
|
|
<ul class="divide-y divide-gray-200">
|
|
<li
|
|
v-for="holiday in holidays.data"
|
|
:key="holiday.id.id"
|
|
class="py-4 flex"
|
|
>
|
|
<div>
|
|
<p class="text-sm font-medium text-gray-900">
|
|
{{ holiday.name }}
|
|
</p>
|
|
<p class="text-sm text-gray-500">
|
|
{{ holiday.displayDate }}
|
|
</p>
|
|
</div>
|
|
</li>
|
|
<li v-if="! holidays.data.length">
|
|
<p class="py-2">
|
|
Brak danych
|
|
</p>
|
|
</li>
|
|
</ul>
|
|
<div>
|
|
<InertiaLink
|
|
href="/holidays"
|
|
class="w-full flex justify-center items-center px-4 py-2 border border-gray-300 shadow-sm text-sm font-medium rounded-md text-gray-700 bg-white hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-blumilk-500"
|
|
>
|
|
Zobacz wszystkie
|
|
</InertiaLink>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</section>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import {computed} from 'vue'
|
|
import {usePage} from '@inertiajs/inertia-vue3'
|
|
import Status from '@/Shared/Status'
|
|
import VacationChart from '@/Shared/VacationChart'
|
|
import {useVacationTypeInfo} from '@/Composables/vacationTypeInfo'
|
|
|
|
export default {
|
|
name: 'DashboardPage',
|
|
components: {Status, VacationChart},
|
|
props: {
|
|
absences: {
|
|
type: Object,
|
|
default: null,
|
|
},
|
|
vacationRequests: {
|
|
type: Object,
|
|
default: null,
|
|
},
|
|
holidays: {
|
|
type: Object,
|
|
default: null,
|
|
},
|
|
can: {
|
|
type: Object,
|
|
default: null,
|
|
},
|
|
stats: {
|
|
type: Object,
|
|
default: () => ({
|
|
used: 0,
|
|
pending: 0,
|
|
remaining: 0,
|
|
}),
|
|
},
|
|
},
|
|
setup() {
|
|
const user = computed(() => usePage().props.value.auth.user)
|
|
|
|
const { findType } = useVacationTypeInfo()
|
|
|
|
return {
|
|
user,
|
|
findType,
|
|
}
|
|
},
|
|
}
|
|
</script>
|