feat: enhance point registration and management features

- Added functionality to capture and display images during point registration, improving user experience.
- Implemented error handling for image uploads and webcam access, ensuring smoother operation.
- Introduced a justification field for point registration, allowing users to provide context for their entries.
- Enhanced the backend to support new features, including image handling and justification storage.
- Updated UI components for better layout and responsiveness, improving overall usability.
This commit is contained in:
2025-11-18 15:28:26 -03:00
parent f0c6e4468f
commit b01d2d6786
10 changed files with 941 additions and 187 deletions

View File

@@ -1,68 +1,68 @@
<script lang="ts">
import { useQuery } from 'convex-svelte';
import { api } from '@sgse-app/backend/convex/_generated/api';
import { Clock, ArrowRight, CheckCircle2, XCircle } from 'lucide-svelte';
import { goto } from '$app/navigation';
import { Clock } from 'lucide-svelte';
import { resolve } from '$app/paths';
// Estatísticas do dia atual
const hoje = new Date().toISOString().split('T')[0]!;
const estatisticasQuery = useQuery(api.pontos.obterEstatisticas, {
dataInicio: hoje,
dataFim: hoje,
});
const estatisticas = $derived(estatisticasQuery?.data);
function abrirDashboard() {
goto(resolve('/(dashboard)/recursos-humanos/registro-pontos'));
}
</script>
<div class="card bg-gradient-to-br from-blue-500 to-cyan-600 text-white shadow-xl hover:shadow-2xl transition-all duration-300">
<div class="card bg-base-100 shadow-xl hover:shadow-2xl transition-all duration-300">
<div class="card-body">
<div class="flex items-center justify-between mb-4">
<div class="flex items-center gap-3">
<div class="p-3 bg-white/20 rounded-xl">
<Clock class="h-6 w-6" strokeWidth={2} />
</div>
<div>
<h3 class="card-title text-white">Gestão de Pontos</h3>
<p class="text-white/80 text-sm">Registros de ponto do dia</p>
<!-- Cabeçalho da Categoria -->
<div class="flex items-start gap-6 mb-6">
<div class="p-4 bg-blue-500/20 rounded-2xl">
<div class="text-blue-600">
<Clock class="h-12 w-12" strokeWidth={2} />
</div>
</div>
<div class="flex-1">
<h2 class="card-title text-2xl mb-2 text-blue-600">
Gestão de Pontos
</h2>
<p class="text-base-content/70">Registros de ponto do dia</p>
</div>
</div>
{#if estatisticas}
<div class="grid grid-cols-2 gap-4 mb-4">
<div class="bg-white/10 rounded-lg p-3">
<div class="flex items-center gap-2 mb-1">
<CheckCircle2 class="h-4 w-4" />
<span class="text-sm text-white/80">Dentro do Prazo</span>
<!-- Grid de Opções -->
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4">
<a
href={resolve('/(dashboard)/recursos-humanos/registro-pontos')}
class="group relative overflow-hidden rounded-xl border-2 border-base-300 bg-linear-to-br from-blue-500/10 to-blue-600/20 p-6 hover:border-primary hover:shadow-lg transition-all duration-300 transform hover:-translate-y-1"
>
<div class="flex flex-col h-full">
<div class="flex items-start justify-between mb-4">
<div
class="p-3 bg-base-100 rounded-lg group-hover:bg-primary group-hover:text-white transition-colors duration-300"
>
<div
class="text-blue-600 group-hover:text-white"
>
<Clock class="h-5 w-5" strokeWidth={2} />
</div>
</div>
<svg
xmlns="http://www.w3.org/2000/svg"
class="h-5 w-5 text-base-content/30 group-hover:text-primary transition-colors duration-300"
fill="none"
viewBox="0 0 24 24"
stroke="currentColor"
>
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M9 5l7 7-7 7"
/>
</svg>
</div>
<div class="text-2xl font-bold">{estatisticas.dentroDoPrazo}</div>
<h3
class="text-lg font-bold text-base-content mb-2 group-hover:text-primary transition-colors duration-300"
>
Gestão de Pontos
</h3>
<p class="text-sm text-base-content/70 flex-1">
Visualizar e gerenciar registros de ponto
</p>
</div>
<div class="bg-white/10 rounded-lg p-3">
<div class="flex items-center gap-2 mb-1">
<XCircle class="h-4 w-4" />
<span class="text-sm text-white/80">Fora do Prazo</span>
</div>
<div class="text-2xl font-bold">{estatisticas.foraDoPrazo}</div>
</div>
</div>
<div class="text-sm text-white/80 mb-4">
Total: {estatisticas.totalRegistros} registros de {estatisticas.totalFuncionarios} funcionários
</div>
{:else}
<div class="text-white/80 text-sm mb-4">Carregando estatísticas...</div>
{/if}
<button class="btn btn-white btn-sm w-full" onclick={abrirDashboard}>
Ver Dashboard Completo
<ArrowRight class="h-4 w-4" />
</button>
</a>
</div>
</div>
</div>