2412 lines
78 KiB
Svelte
2412 lines
78 KiB
Svelte
<script lang="ts">
|
|
import { useConvexClient, useQuery } from 'convex-svelte';
|
|
import { api } from '@sgse-app/backend/convex/_generated/api';
|
|
import AprovarFerias from '$lib/components/AprovarFerias.svelte';
|
|
import WizardSolicitacaoFerias from '$lib/components/ferias/WizardSolicitacaoFerias.svelte';
|
|
import WizardSolicitacaoAusencia from '$lib/components/ausencias/WizardSolicitacaoAusencia.svelte';
|
|
import AprovarAusencias from '$lib/components/AprovarAusencias.svelte';
|
|
import CalendarioAusencias from '$lib/components/ausencias/CalendarioAusencias.svelte';
|
|
import { generateAvatarGallery } from '$lib/utils/avatars';
|
|
import ProtectedRoute from '$lib/components/ProtectedRoute.svelte';
|
|
import type { Id } from '@sgse-app/backend/convex/_generated/dataModel';
|
|
import type { FunctionReturnType } from 'convex/server';
|
|
import { X, Calendar } from 'lucide-svelte';
|
|
|
|
const client = useConvexClient();
|
|
const currentUser = useQuery(api.auth.getCurrentUser, {});
|
|
|
|
type FuncionarioAtual = FunctionReturnType<typeof api.funcionarios.getCurrent>;
|
|
type TimeAtual = FunctionReturnType<typeof api.times.obterTimeFuncionario>;
|
|
type MinhasSolicitacoes = FunctionReturnType<typeof api.ferias.listarMinhasSolicitacoes>;
|
|
type MinhasAusencias = FunctionReturnType<typeof api.ausencias.listarMinhasSolicitacoes>;
|
|
let funcionarioEstavel = $state<FuncionarioAtual | null>(null);
|
|
let meuTimeEstavel = $state<TimeAtual | null>(null);
|
|
let minhasSolicitacoesEstaveis = $state<MinhasSolicitacoes>([]);
|
|
let minhasAusenciasEstaveis = $state<MinhasAusencias>([]);
|
|
|
|
let abaAtiva = $state<
|
|
'meu-perfil' | 'minhas-ferias' | 'minhas-ausencias' | 'aprovar-ferias' | 'aprovar-ausencias'
|
|
>('meu-perfil');
|
|
|
|
let periodoSelecionado = $state<Id<'ferias'> | null>(null);
|
|
|
|
let mostrarModalFoto = $state(false);
|
|
let uploadandoFoto = $state(false);
|
|
let erroUpload = $state('');
|
|
let modoFoto = $state<'upload' | 'avatar'>('avatar');
|
|
let avatarSelecionado = $state<string>('');
|
|
let mostrarBotaoCamera = $state(false);
|
|
|
|
// Estados para Minhas Férias
|
|
let mostrarWizard = $state(false);
|
|
let filtroStatusFerias = $state<string>('todos');
|
|
|
|
// Estados para Minhas Ausências
|
|
let mostrarWizardAusencia = $state(false);
|
|
let filtroStatusAusencias = $state<string>('todos');
|
|
|
|
// Estados para Aprovar Ausências (Gestores)
|
|
let solicitacaoAusenciaAprovar = $state<Id<'solicitacoesAusencias'> | null>(null);
|
|
|
|
// Galeria de avatares (30 avatares profissionais 3D realistas)
|
|
const avatarGallery = generateAvatarGallery(30);
|
|
|
|
// FuncionarioId disponível diretamente do usuário atual
|
|
const funcionarioIdDisponivel = $derived(currentUser?.data?.funcionarioId ?? null);
|
|
|
|
// Queries
|
|
const funcionarioQuery = $derived(
|
|
currentUser?.data?.funcionarioId ? useQuery(api.funcionarios.getCurrent, {}) : { data: null }
|
|
);
|
|
|
|
$effect(() => {
|
|
if (funcionarioQuery?.data) {
|
|
funcionarioEstavel = funcionarioQuery.data;
|
|
} else if (!currentUser?.data?.funcionarioId) {
|
|
funcionarioEstavel = null;
|
|
}
|
|
});
|
|
|
|
const solicitacoesSubordinadosQuery = $derived(
|
|
currentUser?.data?._id
|
|
? useQuery(api.ferias.listarSolicitacoesSubordinados, {
|
|
gestorId: currentUser.data._id as Id<'usuarios'>
|
|
})
|
|
: { data: [] }
|
|
);
|
|
|
|
const ausenciasSubordinadosQuery = $derived(
|
|
currentUser?.data?._id
|
|
? useQuery(api.ausencias.listarSolicitacoesSubordinados, {
|
|
gestorId: currentUser.data._id as Id<'usuarios'>
|
|
})
|
|
: { data: [] }
|
|
);
|
|
|
|
const minhasSolicitacoesQuery = $derived(
|
|
funcionarioEstavel?._id
|
|
? useQuery(api.ferias.listarMinhasSolicitacoes, {
|
|
funcionarioId: funcionarioEstavel._id
|
|
})
|
|
: { data: [] }
|
|
);
|
|
|
|
const minhasAusenciasQuery = $derived(
|
|
funcionarioEstavel?._id
|
|
? useQuery(api.ausencias.listarMinhasSolicitacoes, {
|
|
funcionarioId: funcionarioEstavel._id
|
|
})
|
|
: { data: [] }
|
|
);
|
|
|
|
const meuTimeQuery = $derived(
|
|
funcionarioEstavel?._id
|
|
? useQuery(api.times.obterTimeFuncionario, {
|
|
funcionarioId: funcionarioEstavel._id
|
|
})
|
|
: { data: null }
|
|
);
|
|
|
|
$effect(() => {
|
|
if (meuTimeQuery?.data) {
|
|
meuTimeEstavel = meuTimeQuery.data;
|
|
} else if (!funcionarioEstavel?._id) {
|
|
meuTimeEstavel = null;
|
|
}
|
|
});
|
|
|
|
const funcionario = $derived(funcionarioEstavel);
|
|
const solicitacoesSubordinados = $derived(solicitacoesSubordinadosQuery?.data || []);
|
|
const ausenciasSubordinados = $derived(ausenciasSubordinadosQuery?.data || []);
|
|
const meuTime = $derived(meuTimeEstavel);
|
|
|
|
$effect(() => {
|
|
if (Array.isArray(minhasSolicitacoesQuery?.data)) {
|
|
minhasSolicitacoesEstaveis = minhasSolicitacoesQuery.data;
|
|
} else if (!funcionarioEstavel?._id) {
|
|
minhasSolicitacoesEstaveis = [];
|
|
}
|
|
});
|
|
|
|
$effect(() => {
|
|
if (Array.isArray(minhasAusenciasQuery?.data)) {
|
|
minhasAusenciasEstaveis = minhasAusenciasQuery.data;
|
|
} else if (!funcionarioEstavel?._id) {
|
|
minhasAusenciasEstaveis = [];
|
|
}
|
|
});
|
|
|
|
const minhasSolicitacoes = $derived(minhasSolicitacoesEstaveis);
|
|
const minhasAusencias = $derived(minhasAusenciasEstaveis);
|
|
|
|
const timesSubordinadosQuery = $derived(useQuery(api.times.listarSubordinadosDoGestorAtual, {}));
|
|
const timesSubordinados = $derived(timesSubordinadosQuery?.data || []);
|
|
// Times gerenciados usam a query que já infere o gestor logado
|
|
const meusTimesGestor = $derived(timesSubordinados);
|
|
|
|
const rolePermiteAprovacao = $derived(
|
|
currentUser?.data?.role?.nome === 'TI_MASTER' || currentUser?.data?.role?.nome === 'TI_ADMIN'
|
|
);
|
|
|
|
const ehGestor = $derived((timesSubordinados || []).length > 0 || rolePermiteAprovacao);
|
|
|
|
// Filtrar minhas solicitações
|
|
const solicitacoesFiltradas = $derived(
|
|
minhasSolicitacoes.filter((s) => {
|
|
if (filtroStatusFerias !== 'todos' && s.status !== filtroStatusFerias) return false;
|
|
return true;
|
|
})
|
|
);
|
|
|
|
// Filtrar minhas ausências
|
|
const ausenciasFiltradas = $derived(
|
|
minhasAusencias.filter((a) => {
|
|
if (filtroStatusAusencias !== 'todos' && a.status !== filtroStatusAusencias) return false;
|
|
return true;
|
|
})
|
|
);
|
|
|
|
// Formatar ausências para o calendário
|
|
const ausenciasParaCalendario = $derived(
|
|
minhasAusencias.map((a) => ({
|
|
dataInicio: a.dataInicio,
|
|
dataFim: a.dataFim,
|
|
status: a.status as 'aguardando_aprovacao' | 'aprovado' | 'reprovado'
|
|
}))
|
|
);
|
|
|
|
// Estatísticas das minhas férias (períodos individuais)
|
|
const statsMinhasFerias = $derived({
|
|
total: minhasSolicitacoes.length,
|
|
aguardando: minhasSolicitacoes.filter((s) => s.status === 'aguardando_aprovacao').length,
|
|
aprovadas: minhasSolicitacoes.filter(
|
|
(s) =>
|
|
s.status === 'aprovado' || s.status === 'data_ajustada_aprovada' || s.status === 'EmFérias'
|
|
).length,
|
|
reprovadas: minhasSolicitacoes.filter((s) => s.status === 'reprovado').length,
|
|
emFerias: funcionario?.statusFerias === 'em_ferias' ? 1 : 0
|
|
});
|
|
|
|
// Estatísticas das minhas ausências
|
|
const statsMinhasAusencias = $derived({
|
|
total: minhasAusencias.length,
|
|
aguardando: minhasAusencias.filter((a) => a.status === 'aguardando_aprovacao').length,
|
|
aprovadas: minhasAusencias.filter((a) => a.status === 'aprovado').length,
|
|
reprovadas: minhasAusencias.filter((a) => a.status === 'reprovado').length
|
|
});
|
|
|
|
async function recarregar() {
|
|
periodoSelecionado = null;
|
|
}
|
|
|
|
async function selecionarPeriodo(feriasId: Id<'ferias'>) {
|
|
periodoSelecionado = feriasId;
|
|
}
|
|
|
|
// Função para formatar data sem problemas de timezone
|
|
function formatarDataString(dataString: string): string {
|
|
if (!dataString) return '';
|
|
// Dividir a string da data (formato YYYY-MM-DD)
|
|
const partes = dataString.split('-');
|
|
if (partes.length !== 3) return dataString;
|
|
// Retornar no formato DD/MM/YYYY
|
|
return `${partes[2]}/${partes[1]}/${partes[0]}`;
|
|
}
|
|
|
|
function getStatusBadge(status: string) {
|
|
const badges: Record<string, string> = {
|
|
aguardando_aprovacao: 'badge-warning',
|
|
aprovado: 'badge-success',
|
|
reprovado: 'badge-error',
|
|
data_ajustada_aprovada: 'badge-info',
|
|
EmFérias: 'badge-info'
|
|
};
|
|
return badges[status] || 'badge-neutral';
|
|
}
|
|
|
|
function getStatusTexto(status: string) {
|
|
const textos: Record<string, string> = {
|
|
aguardando_aprovacao: 'Aguardando',
|
|
aprovado: 'Aprovado',
|
|
reprovado: 'Reprovado',
|
|
data_ajustada_aprovada: 'Ajustado',
|
|
EmFérias: 'Em Férias'
|
|
};
|
|
return textos[status] || status;
|
|
}
|
|
|
|
async function handleUploadFoto(event: Event) {
|
|
const input = event.target as HTMLInputElement;
|
|
const file = input.files?.[0];
|
|
|
|
if (!file) return;
|
|
|
|
// Validar tipo de arquivo
|
|
if (!file.type.startsWith('image/')) {
|
|
erroUpload = 'Por favor, selecione uma imagem válida';
|
|
return;
|
|
}
|
|
|
|
// Validar tamanho (max 5MB)
|
|
if (file.size > 5 * 1024 * 1024) {
|
|
erroUpload = 'A imagem deve ter no máximo 5MB';
|
|
return;
|
|
}
|
|
|
|
uploadandoFoto = true;
|
|
erroUpload = '';
|
|
|
|
try {
|
|
// 2. Gerar URL de upload
|
|
const uploadUrl = await client.mutation(api.usuarios.uploadFotoPerfil, {});
|
|
|
|
// 3. Upload do arquivo
|
|
const response = await fetch(uploadUrl, {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': file.type },
|
|
body: file
|
|
});
|
|
|
|
if (!response.ok) {
|
|
throw new Error('Falha no upload da imagem');
|
|
}
|
|
|
|
const { storageId } = await response.json();
|
|
|
|
// 4. Atualizar perfil com o novo storageId
|
|
await client.mutation(api.usuarios.atualizarPerfil, {
|
|
fotoPerfil: storageId,
|
|
avatar: undefined // Remove avatar se colocar foto
|
|
});
|
|
|
|
// 5. Aguardar um pouco para garantir que o backend processou
|
|
await new Promise((resolve) => setTimeout(resolve, 300));
|
|
|
|
// 8. Limpar o input para permitir novo upload
|
|
input.value = '';
|
|
|
|
// 9. Fechar modal após sucesso
|
|
mostrarModalFoto = false;
|
|
|
|
// Toast de sucesso
|
|
const toast = document.createElement('div');
|
|
toast.className = 'toast toast-top toast-end';
|
|
toast.innerHTML = `
|
|
<div class="alert alert-success">
|
|
<svg xmlns="http://www.w3.org/2000/svg" class="stroke-current shrink-0 h-6 w-6" fill="none" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z" />
|
|
</svg>
|
|
<span>Foto de perfil atualizada com sucesso!</span>
|
|
</div>
|
|
`;
|
|
document.body.appendChild(toast);
|
|
setTimeout(() => toast.remove(), 3000);
|
|
} catch (e: unknown) {
|
|
const errorMessage = e instanceof Error ? e.message : String(e);
|
|
erroUpload = errorMessage || 'Erro ao fazer upload da foto';
|
|
}
|
|
uploadandoFoto = false;
|
|
}
|
|
|
|
async function handleSelecionarAvatar(avatarUrl: string) {
|
|
uploadandoFoto = true;
|
|
erroUpload = '';
|
|
|
|
try {
|
|
// 2. Salvar avatar selecionado no backend
|
|
await client.mutation(api.usuarios.atualizarPerfil, {
|
|
avatar: avatarUrl,
|
|
fotoPerfil: undefined // Remove foto se colocar avatar
|
|
});
|
|
|
|
// 6. Fechar modal após sucesso
|
|
mostrarModalFoto = false;
|
|
|
|
// Toast de sucesso
|
|
const toast = document.createElement('div');
|
|
toast.className = 'toast toast-top toast-end';
|
|
toast.innerHTML = `
|
|
<div class="alert alert-success">
|
|
<svg xmlns="http://www.w3.org/2000/svg" class="stroke-current shrink-0 h-6 w-6" fill="none" viewBox="0 0 24 24">
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z" />
|
|
</svg>
|
|
<span>Avatar atualizado com sucesso!</span>
|
|
</div>
|
|
`;
|
|
document.body.appendChild(toast);
|
|
setTimeout(() => toast.remove(), 3000);
|
|
} catch (e: unknown) {
|
|
const errorMessage = e instanceof Error ? e.message : String(e);
|
|
erroUpload = errorMessage || 'Erro ao salvar avatar';
|
|
} finally {
|
|
uploadandoFoto = false;
|
|
}
|
|
}
|
|
|
|
function abrirModalFoto() {
|
|
erroUpload = '';
|
|
modoFoto = 'avatar';
|
|
avatarSelecionado = '';
|
|
mostrarModalFoto = true;
|
|
}
|
|
</script>
|
|
|
|
<ProtectedRoute>
|
|
<main class="min-h-screen pb-12">
|
|
<!-- BANNER HERO PREMIUM -->
|
|
<div class="relative mb-8 overflow-hidden">
|
|
<!-- Background com gradiente animado -->
|
|
<div
|
|
class="animate-gradient absolute inset-0 bg-linear-to-br from-purple-600 via-blue-600 to-indigo-700"
|
|
></div>
|
|
|
|
<!-- Overlay pattern -->
|
|
<div
|
|
class="absolute inset-0 opacity-10"
|
|
style="background-image: url('data:image/svg+xml,%3Csvg width="60" height="60" viewBox="0 0 60 60" xmlns="http://www.w3.org/2000/svg"%3E%3Cg fill="none" fill-rule="evenodd"%3E%3Cg fill="%23ffffff" fill-opacity="1"%3E%3Cpath d="M36 34v-4h-2v4h-4v2h4v4h2v-4h4v-2h-4zm0-30V0h-2v4h-4v2h4v4h2V6h4V4h-4zM6 34v-4H4v4H0v2h4v4h2v-4h4v-2H6zM6 4V0H4v4H0v2h4v4h2V6h4V4H6z"/%3E%3C/g%3E%3C/g%3E%3C/svg%3E');"
|
|
></div>
|
|
|
|
<!-- Conteúdo do Banner -->
|
|
<div class="relative container mx-auto px-6 py-16">
|
|
<div class="flex flex-col items-center gap-8 md:flex-row">
|
|
<!-- Avatar PREMIUM -->
|
|
<div
|
|
class="group relative"
|
|
role="button"
|
|
tabindex="0"
|
|
onmouseenter={() => (mostrarBotaoCamera = true)}
|
|
onmouseleave={() => (mostrarBotaoCamera = false)}
|
|
>
|
|
s
|
|
<button
|
|
type="button"
|
|
class="avatar cursor-pointer border-0 bg-transparent p-0"
|
|
onclick={abrirModalFoto}
|
|
>
|
|
<div
|
|
class="animate-float h-40 w-40 rounded-full shadow-2xl ring-4 ring-white ring-offset-4 ring-offset-transparent transition-all duration-300 hover:scale-105 hover:ring-8"
|
|
>
|
|
{#if currentUser.data?.fotoPerfilUrl}
|
|
<img
|
|
src={currentUser.data.fotoPerfilUrl}
|
|
alt="Foto de perfil"
|
|
class="object-cover"
|
|
/>
|
|
{:else if currentUser.data?.avatar}
|
|
<img src={currentUser.data.avatar} alt="Avatar" class="object-cover" />
|
|
{:else}
|
|
<div class="flex items-center justify-center bg-white text-purple-700">
|
|
<span class="text-5xl font-black"
|
|
>{currentUser.data?.nome.substring(0, 2).toUpperCase()}</span
|
|
>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
</button>
|
|
|
|
<!-- Botão de editar MODERNO -->
|
|
<button
|
|
type="button"
|
|
class={`absolute right-2 bottom-2 flex h-12 w-12 items-center justify-center rounded-full bg-white text-purple-600 shadow-2xl transition-all duration-300 hover:scale-110 ${mostrarBotaoCamera ? 'scale-100 opacity-100' : 'scale-50 opacity-0'}`}
|
|
onclick={abrirModalFoto}
|
|
aria-label="Editar foto de perfil"
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-6 w-6"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
stroke-width="2.5"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
d="M3 9a2 2 0 012-2h.93a2 2 0 001.664-.89l.812-1.22A2 2 0 0110.07 4h3.86a2 2 0 011.664.89l.812 1.22A2 2 0 0018.07 7H19a2 2 0 012 2v9a2 2 0 01-2 2H5a2 2 0 01-2-2V9z"
|
|
/>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
d="M15 13a3 3 0 11-6 0 3 3 0 016 0z"
|
|
/>
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Informações do Usuário PREMIUM -->
|
|
<div class="flex-1 text-center text-white md:text-left">
|
|
<h1 class="mb-3 text-5xl font-black drop-shadow-lg">
|
|
{currentUser.data?.nome}
|
|
</h1>
|
|
|
|
{#if funcionario?.descricaoCargo}
|
|
<p
|
|
class="mb-3 flex items-center justify-center gap-2 text-2xl font-semibold text-white/90 md:justify-start"
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-6 w-6"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M21 13.255A23.931 23.931 0 0112 15c-3.183 0-6.22-.62-9-1.745M16 6V4a2 2 0 00-2-2h-4a2 2 0 00-2 2v2m4 6h.01M5 20h14a2 2 0 002-2V8a2 2 0 00-2-2H5a2 2 0 00-2 2v10a2 2 0 002 2z"
|
|
/>
|
|
</svg>
|
|
{funcionario.descricaoCargo}
|
|
</p>
|
|
{/if}
|
|
|
|
<p
|
|
class="mb-4 flex items-center justify-center gap-2 text-lg text-white/80 md:justify-start"
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-5 w-5"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M3 8l7.89 5.26a2 2 0 002.22 0L21 8M5 19h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v10a2 2 0 002 2z"
|
|
/>
|
|
</svg>
|
|
{currentUser.data?.email}
|
|
</p>
|
|
|
|
<div class="flex flex-wrap items-center justify-center gap-3 md:justify-start">
|
|
<div
|
|
class="badge badge-lg border-0 bg-white/90 px-4 font-bold text-purple-700 shadow-lg"
|
|
>
|
|
{currentUser.data?.role?.nome || 'Usuário'}
|
|
</div>
|
|
|
|
{#if meuTime}
|
|
<div
|
|
class="badge badge-lg border-0 bg-white/80 px-4 font-semibold shadow-lg"
|
|
style="color: {meuTime.cor}"
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="mr-2 h-4 w-4"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M17 20h5v-2a3 3 0 00-5.356-1.857M17 20H7m10 0v-2c0-.656-.126-1.283-.356-1.857M7 20H2v-2a3 3 0 015.356-1.857M7 20v-2c0-.656.126-1.283.356-1.857m0 0a5.002 5.002 0 019.288 0M15 7a3 3 0 11-6 0 3 3 0 016 0zm6 3a2 2 0 11-4 0 2 2 0 014 0zM7 10a2 2 0 11-4 0 2 2 0 014 0z"
|
|
/>
|
|
</svg>
|
|
{meuTime.nome}
|
|
</div>
|
|
{/if}
|
|
|
|
{#if funcionario?.statusFerias === 'em_ferias'}
|
|
<div
|
|
class="badge badge-lg animate-pulse border-0 bg-yellow-400 px-4 font-bold text-yellow-900 shadow-lg"
|
|
>
|
|
🏖️ Em Férias
|
|
</div>
|
|
{:else}
|
|
<div
|
|
class="badge badge-lg border-0 bg-green-400 px-4 font-bold text-green-900 shadow-lg"
|
|
>
|
|
✅ Ativo
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="container mx-auto max-w-7xl px-6">
|
|
<!-- Tabs PREMIUM -->
|
|
<div
|
|
role="tablist"
|
|
class="tabs tabs-boxed from-base-200 to-base-300 mb-8 bg-linear-to-r p-2 shadow-xl"
|
|
>
|
|
<button
|
|
type="button"
|
|
role="tab"
|
|
class={`tab tab-lg font-semibold transition-all duration-300 ${abaAtiva === 'meu-perfil' ? 'tab-active scale-105 bg-linear-to-r from-purple-600 to-blue-600 text-white shadow-lg' : 'hover:bg-base-100'}`}
|
|
onclick={() => (abaAtiva = 'meu-perfil')}
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="mr-2 h-5 w-5"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
stroke-width="2"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
d="M16 7a4 4 0 11-8 0 4 4 0 018 0zM12 14a7 7 0 00-7 7h14a7 7 0 00-7-7z"
|
|
/>
|
|
</svg>
|
|
Meu Perfil
|
|
</button>
|
|
|
|
<button
|
|
type="button"
|
|
role="tab"
|
|
class={`tab tab-lg font-semibold transition-all duration-300 ${abaAtiva === 'minhas-ferias' ? 'tab-active scale-105 bg-linear-to-r from-purple-600 to-blue-600 text-white shadow-lg' : 'hover:bg-base-100'}`}
|
|
onclick={() => (abaAtiva = 'minhas-ferias')}
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="mr-2 h-5 w-5"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
stroke-width="2"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z"
|
|
/>
|
|
</svg>
|
|
Minhas Férias
|
|
</button>
|
|
|
|
<button
|
|
type="button"
|
|
role="tab"
|
|
class={`tab tab-lg font-semibold transition-all duration-300 ${abaAtiva === 'minhas-ausencias' ? 'tab-active scale-105 bg-linear-to-r from-orange-600 to-amber-600 text-white shadow-lg' : 'hover:bg-base-100'}`}
|
|
onclick={() => (abaAtiva = 'minhas-ausencias')}
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="mr-2 h-5 w-5"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
stroke-width="2"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
Minhas Ausências
|
|
</button>
|
|
|
|
{#if ehGestor}
|
|
<button
|
|
type="button"
|
|
role="tab"
|
|
class={`tab tab-lg font-semibold transition-all duration-300 ${abaAtiva === 'aprovar-ferias' ? 'tab-active scale-105 bg-linear-to-r from-purple-600 to-blue-600 text-white shadow-lg' : 'hover:bg-base-100'}`}
|
|
onclick={() => (abaAtiva = 'aprovar-ferias')}
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="mr-2 h-5 w-5"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
stroke-width="2"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
Aprovar Férias
|
|
{#if (solicitacoesSubordinados || []).filter((s) => s.status === 'aguardando_aprovacao').length > 0}
|
|
<span class="badge badge-error badge-sm ml-2 animate-pulse">
|
|
{(solicitacoesSubordinados || []).filter((s) => s.status === 'aguardando_aprovacao')
|
|
.length}
|
|
</span>
|
|
{/if}
|
|
</button>
|
|
|
|
<button
|
|
type="button"
|
|
role="tab"
|
|
class={`tab tab-lg font-semibold transition-all duration-300 ${abaAtiva === 'aprovar-ausencias' ? 'tab-active scale-105 bg-linear-to-r from-orange-600 to-amber-600 text-white shadow-lg' : 'hover:bg-base-100'}`}
|
|
onclick={() => (abaAtiva = 'aprovar-ausencias')}
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="mr-2 h-5 w-5"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
stroke-width="2"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
Aprovar Ausências
|
|
{#if (ausenciasSubordinados || []).filter((a) => a.status === 'aguardando_aprovacao').length > 0}
|
|
<span class="badge badge-error badge-sm ml-2 animate-pulse">
|
|
{(ausenciasSubordinados || []).filter((a) => a.status === 'aguardando_aprovacao')
|
|
.length}
|
|
</span>
|
|
{/if}
|
|
</button>
|
|
{/if}
|
|
</div>
|
|
|
|
<!-- Conteúdo das Abas -->
|
|
{#if abaAtiva === 'meu-perfil'}
|
|
<!-- Meu Perfil PREMIUM -->
|
|
<div class="space-y-6">
|
|
<!-- STATS CARDS -->
|
|
<div class="grid grid-cols-1 gap-6 md:grid-cols-4">
|
|
<!-- Stat 1: Perfil -->
|
|
<div
|
|
class="card bg-linear-to-br from-purple-500 to-purple-700 text-white shadow-2xl transition-transform hover:scale-105"
|
|
>
|
|
<div class="card-body">
|
|
<div class="flex items-center justify-between">
|
|
<div>
|
|
<p class="text-sm font-medium text-white/80">Seu Perfil</p>
|
|
<p class="text-2xl font-black">
|
|
{currentUser.data?.role?.nome || 'Usuário'}
|
|
</p>
|
|
</div>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-12 w-12 opacity-80"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M5.121 17.804A13.937 13.937 0 0112 16c2.5 0 4.847.655 6.879 1.804M15 10a3 3 0 11-6 0 3 3 0 016 0zm6 2a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Stat 2: Time -->
|
|
<div
|
|
class="card bg-linear-to-br from-blue-500 to-blue-700 text-white shadow-2xl transition-transform hover:scale-105"
|
|
>
|
|
<div class="card-body">
|
|
<div class="flex items-center justify-between">
|
|
<div>
|
|
<p class="text-sm font-medium text-white/80">Seu Time</p>
|
|
<p class="truncate text-2xl font-black">
|
|
{meuTime?.nome || 'Sem time'}
|
|
</p>
|
|
</div>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-12 w-12 opacity-80"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M17 20h5v-2a3 3 0 00-5.356-1.857M17 20H7m10 0v-2c0-.656-.126-1.283-.356-1.857M7 20H2v-2a3 3 0 015.356-1.857M7 20v-2c0-.656.126-1.283.356-1.857m0 0a5.002 5.002 0 019.288 0M15 7a3 3 0 11-6 0 3 3 0 016 0zm6 3a2 2 0 11-4 0 2 2 0 014 0zM7 10a2 2 0 11-4 0 2 2 0 014 0z"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Stat 3: Status -->
|
|
<div
|
|
class="card bg-linear-to-br from-green-500 to-green-700 text-white shadow-2xl transition-transform hover:scale-105"
|
|
>
|
|
<div class="card-body">
|
|
<div class="flex items-center justify-between">
|
|
<div>
|
|
<p class="text-sm font-medium text-white/80">Status</p>
|
|
<p class="text-2xl font-black">
|
|
{funcionario?.statusFerias === 'em_ferias' ? 'Em Férias' : 'Ativo'}
|
|
</p>
|
|
</div>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-12 w-12 opacity-80"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Stat 4: Matrícula -->
|
|
<div
|
|
class="card bg-linear-to-br from-indigo-500 to-indigo-700 text-white shadow-2xl transition-transform hover:scale-105"
|
|
>
|
|
<div class="card-body">
|
|
<div class="flex items-center justify-between">
|
|
<div>
|
|
<p class="text-sm font-medium text-white/80">Matrícula</p>
|
|
<p class="text-2xl font-black">
|
|
{funcionario?.matricula || '---'}
|
|
</p>
|
|
</div>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-12 w-12 opacity-80"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M10 6H5a2 2 0 00-2 2v9a2 2 0 002 2h14a2 2 0 002-2V8a2 2 0 00-2-2h-5m-4 0V5a2 2 0 114 0v1m-4 0a2 2 0 104 0m-5 8a2 2 0 100-4 2 2 0 000 4zm0 0c1.306 0 2.417.835 2.83 2M9 14a3.001 3.001 0 00-2.83 2M15 11h3m-3 4h2"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- CARDS PRINCIPAIS -->
|
|
<div class="grid grid-cols-1 gap-6 lg:grid-cols-2">
|
|
<!-- Informações Pessoais PREMIUM -->
|
|
<div
|
|
class="card bg-base-100 hover:shadow-3xl border-t-4 border-purple-500 shadow-2xl transition-shadow"
|
|
>
|
|
<div class="card-body">
|
|
<h2 class="card-title mb-6 flex items-center gap-2 text-2xl">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-7 w-7 text-purple-600"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M16 7a4 4 0 11-8 0 4 4 0 018 0zM12 14a7 7 0 00-7 7h14a7 7 0 00-7-7z"
|
|
/>
|
|
</svg>
|
|
Informações Pessoais
|
|
</h2>
|
|
<div class="space-y-4">
|
|
<div
|
|
class="hover:bg-base-200 flex items-start gap-3 rounded-lg p-3 transition-colors"
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="text-primary mt-1 h-6 w-6"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M16 7a4 4 0 11-8 0 4 4 0 018 0zM12 14a7 7 0 00-7 7h14a7 7 0 00-7-7z"
|
|
/>
|
|
</svg>
|
|
<div class="flex-1">
|
|
<span class="label-text text-base-content/70 text-sm font-bold"
|
|
>Nome Completo</span
|
|
>
|
|
<p class="text-base-content text-lg font-semibold">
|
|
{currentUser.data?.nome}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="divider my-1"></div>
|
|
|
|
<div
|
|
class="hover:bg-base-200 flex items-start gap-3 rounded-lg p-3 transition-colors"
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="text-primary mt-1 h-6 w-6"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M3 8l7.89 5.26a2 2 0 002.22 0L21 8M5 19h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v10a2 2 0 002 2z"
|
|
/>
|
|
</svg>
|
|
<div class="flex-1">
|
|
<span class="label-text text-base-content/70 text-sm font-bold"
|
|
>E-mail Institucional</span
|
|
>
|
|
<p class="text-base-content text-lg font-semibold break-all">
|
|
{currentUser.data?.email}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="divider my-1"></div>
|
|
|
|
<div
|
|
class="hover:bg-base-200 flex items-start gap-3 rounded-lg p-3 transition-colors"
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="text-primary mt-1 h-6 w-6"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M9 12l2 2 4-4m5.618-4.016A11.955 11.955 0 0112 2.944a11.955 11.955 0 01-8.618 3.04A12.02 12.02 0 003 9c0 5.591 3.824 10.29 9 11.622 5.176-1.332 9-6.03 9-11.622 0-1.042-.133-2.052-.382-3.016z"
|
|
/>
|
|
</svg>
|
|
<div class="flex-1">
|
|
<span class="label-text text-base-content/70 text-sm font-bold"
|
|
>Perfil de Acesso</span
|
|
>
|
|
<div class="badge badge-primary badge-lg mt-1 font-bold">
|
|
{currentUser.data?.role?.nome || 'Usuário'}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Dados Funcionais PREMIUM -->
|
|
{#if funcionario}
|
|
<div
|
|
class="card bg-base-100 hover:shadow-3xl border-t-4 border-blue-500 shadow-2xl transition-shadow"
|
|
>
|
|
<div class="card-body">
|
|
<h2 class="card-title mb-6 flex items-center gap-2 text-2xl">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-7 w-7 text-blue-600"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M21 13.255A23.931 23.931 0 0112 15c-3.183 0-6.22-.62-9-1.745M16 6V4a2 2 0 00-2-2h-4a2 2 0 00-2 2v2m4 6h.01M5 20h14a2 2 0 002-2V8a2 2 0 00-2-2H5a2 2 0 00-2 2v10a2 2 0 002 2z"
|
|
/>
|
|
</svg>
|
|
Dados Funcionais
|
|
</h2>
|
|
<div class="space-y-4">
|
|
<div
|
|
class="hover:bg-base-200 flex items-start gap-3 rounded-lg p-3 transition-colors"
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="text-primary mt-1 h-6 w-6"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M10 6H5a2 2 0 00-2 2v9a2 2 0 002 2h14a2 2 0 002-2V8a2 2 0 00-2-2h-5m-4 0V5a2 2 0 114 0v1m-4 0a2 2 0 104 0m-5 8a2 2 0 100-4 2 2 0 000 4zm0 0c1.306 0 2.417.835 2.83 2M9 14a3.001 3.001 0 00-2.83 2M15 11h3m-3 4h2"
|
|
/>
|
|
</svg>
|
|
<div class="flex-1">
|
|
<span class="label-text text-base-content/70 text-sm font-bold"
|
|
>Matrícula</span
|
|
>
|
|
<p class="text-base-content text-lg font-semibold">
|
|
{funcionario.matricula || 'Não informada'}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="divider my-1"></div>
|
|
|
|
<div
|
|
class="hover:bg-base-200 flex items-start gap-3 rounded-lg p-3 transition-colors"
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="text-primary mt-1 h-6 w-6"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M7 21a4 4 0 01-4-4V5a2 2 0 012-2h4a2 2 0 012 2v12a4 4 0 01-4 4zm0 0h12a2 2 0 002-2v-4a2 2 0 00-2-2h-2.343M11 7.343l1.657-1.657a2 2 0 012.828 0l2.829 2.829a2 2 0 010 2.828l-8.486 8.485M7 17h.01"
|
|
/>
|
|
</svg>
|
|
<div class="flex-1">
|
|
<span class="label-text text-base-content/70 text-sm font-bold">CPF</span>
|
|
<p class="text-base-content text-lg font-semibold">
|
|
{funcionario.cpf}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="divider my-1"></div>
|
|
|
|
<div
|
|
class="hover:bg-base-200 flex items-start gap-3 rounded-lg p-3 transition-colors"
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="text-primary mt-1 h-6 w-6"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M17 20h5v-2a3 3 0 00-5.356-1.857M17 20H7m10 0v-2c0-.656-.126-1.283-.356-1.857M7 20H2v-2a3 3 0 015.356-1.857M7 20v-2c0-.656.126-1.283.356-1.857m0 0a5.002 5.002 0 019.288 0M15 7a3 3 0 11-6 0 3 3 0 016 0zm6 3a2 2 0 11-4 0 2 2 0 014 0zM7 10a2 2 0 11-4 0 2 2 0 014 0z"
|
|
/>
|
|
</svg>
|
|
<div class="flex-1">
|
|
<span class="label-text text-base-content/70 text-sm font-bold">Time</span>
|
|
{#if meuTime}
|
|
<div class="mt-1 flex items-center gap-2">
|
|
<div
|
|
class="badge badge-lg font-semibold"
|
|
style="background-color: {meuTime.cor}20; border-color: {meuTime.cor}; color: {meuTime.cor}"
|
|
>
|
|
{meuTime.nome}
|
|
</div>
|
|
</div>
|
|
<p class="text-base-content/60 mt-1 text-xs">
|
|
Gestor: <span class="font-semibold">{meuTime.gestor?.nome}</span>
|
|
</p>
|
|
{:else}
|
|
<p class="text-base-content/50 mt-1 text-sm">Não atribuído a um time</p>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
|
|
<div class="divider my-1"></div>
|
|
|
|
<div
|
|
class="hover:bg-base-200 flex items-start gap-3 rounded-lg p-3 transition-colors"
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="text-primary mt-1 h-6 w-6"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
<div class="flex-1">
|
|
<span class="label-text text-base-content/70 text-sm font-bold"
|
|
>Status Atual</span
|
|
>
|
|
{#if funcionario.statusFerias === 'em_ferias'}
|
|
<div class="badge badge-warning badge-lg mt-1 font-bold">
|
|
🏖️ Em Férias
|
|
</div>
|
|
{:else}
|
|
<div class="badge badge-success badge-lg mt-1 font-bold">✅ Ativo</div>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
|
|
<!-- Times Gerenciados PREMIUM -->
|
|
{#if ehGestor}
|
|
<div
|
|
class="card border-t-4 border-amber-500 bg-linear-to-br from-amber-50 to-orange-50 shadow-2xl dark:from-amber-950 dark:to-orange-950"
|
|
>
|
|
<div class="card-body">
|
|
<h2
|
|
class="card-title mb-6 flex items-center gap-2 text-2xl text-amber-700 dark:text-amber-400"
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-7 w-7"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M19 21V5a2 2 0 00-2-2H7a2 2 0 00-2 2v16m14 0h2m-2 0h-5m-9 0H3m2 0h5M9 7h1m-1 4h1m4-4h1m-1 4h1m-5 10v-5a1 1 0 011-1h2a1 1 0 011 1v5m-4 0h4"
|
|
/>
|
|
</svg>
|
|
Times que Você Gerencia
|
|
<div class="badge badge-warning badge-lg ml-2">
|
|
{meusTimesGestor.length}
|
|
</div>
|
|
</h2>
|
|
|
|
{#if meusTimesGestor.length === 0}
|
|
<div class="alert alert-info">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
class="h-6 w-6 shrink-0 stroke-current"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
></path>
|
|
</svg>
|
|
<span>Você não gerencia nenhum time no momento.</span>
|
|
</div>
|
|
{:else}
|
|
<div class="grid grid-cols-1 gap-6 md:grid-cols-2 lg:grid-cols-3">
|
|
{#each meusTimesGestor as time (time._id)}
|
|
<div
|
|
class="card dark:bg-base-100 border-l-4 bg-white shadow-xl transition-all hover:scale-105 hover:shadow-2xl"
|
|
style="border-color: {time.cor}"
|
|
>
|
|
<div class="card-body">
|
|
<div class="flex items-start gap-2">
|
|
<div class="flex-1">
|
|
<h3 class="mb-2 text-xl font-black" style="color: {time.cor}">
|
|
{time.nome}
|
|
</h3>
|
|
<p class="text-base-content/70 mb-4 line-clamp-2 text-sm">
|
|
{time.descricao || 'Sem descrição'}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="divider my-2"></div>
|
|
|
|
<div class="flex items-center justify-between">
|
|
<div class="flex items-center gap-2">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="text-primary h-5 w-5"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M12 4.354a4 4 0 110 5.292M15 21H3v-1a6 6 0 0112 0v1zm0 0h6v-1a6 6 0 00-9-5.197M13 7a4 4 0 11-8 0 4 4 0 018 0z"
|
|
/>
|
|
</svg>
|
|
<span class="text-lg font-bold">{time.membros?.length || 0}</span>
|
|
<span class="text-base-content/70 text-sm">membros</span>
|
|
</div>
|
|
<div
|
|
class="badge badge-lg font-semibold"
|
|
style="background-color: {time.cor}20; border-color: {time.cor}; color: {time.cor}"
|
|
>
|
|
Gestor
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{/each}
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
{:else if abaAtiva === 'minhas-ferias'}
|
|
<!-- Minhas Férias -->
|
|
<div class="space-y-6">
|
|
<!-- Estatísticas -->
|
|
<div class="grid grid-cols-1 gap-4 md:grid-cols-5">
|
|
<div class="stat bg-base-100 rounded-box border-base-300 border shadow-lg">
|
|
<div class="stat-figure text-primary">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-8 w-8"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
<div class="stat-title">Total</div>
|
|
<div class="stat-value text-primary">{statsMinhasFerias.total}</div>
|
|
<div class="stat-desc">Solicitações</div>
|
|
</div>
|
|
|
|
<div class="stat bg-base-100 rounded-box border-warning/30 border shadow-lg">
|
|
<div class="stat-figure text-warning">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-8 w-8"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
<div class="stat-title">Aguardando</div>
|
|
<div class="stat-value text-warning">
|
|
{statsMinhasFerias.aguardando}
|
|
</div>
|
|
<div class="stat-desc">Pendentes</div>
|
|
</div>
|
|
|
|
<div class="stat bg-base-100 rounded-box border-success/30 border shadow-lg">
|
|
<div class="stat-figure text-success">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-8 w-8"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
<div class="stat-title">Aprovadas</div>
|
|
<div class="stat-value text-success">
|
|
{statsMinhasFerias.aprovadas}
|
|
</div>
|
|
<div class="stat-desc">Deferidas</div>
|
|
</div>
|
|
|
|
<div class="stat bg-base-100 rounded-box border-error/30 border shadow-lg">
|
|
<div class="stat-figure text-error">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-8 w-8"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M10 14l2-2m0 0l2-2m-2 2l-2-2m2 2l2 2m7-2a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
<div class="stat-title">Reprovadas</div>
|
|
<div class="stat-value text-error">
|
|
{statsMinhasFerias.reprovadas}
|
|
</div>
|
|
<div class="stat-desc">Indeferidas</div>
|
|
</div>
|
|
|
|
<div
|
|
class="stat rounded-box border-2 border-purple-500/30 bg-linear-to-br from-purple-500/10 to-purple-600/20 shadow-lg"
|
|
>
|
|
<div class="stat-figure text-purple-600">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-8 w-8"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M14.828 14.828a4 4 0 01-5.656 0M9 10h.01M15 10h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
<div class="stat-title">Em Férias</div>
|
|
<div class="stat-value text-purple-600">
|
|
{statsMinhasFerias.emFerias}
|
|
</div>
|
|
<div class="stat-desc">Agora</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Filtros e Botão Nova Solicitação -->
|
|
<div class="card bg-base-100 shadow-lg">
|
|
<div class="card-body">
|
|
<div
|
|
class="flex flex-col items-start justify-between gap-4 md:flex-row md:items-center"
|
|
>
|
|
<h2 class="card-title text-lg">Filtros</h2>
|
|
<button
|
|
type="button"
|
|
class="btn btn-primary gap-2"
|
|
onclick={() => {
|
|
if (funcionarioIdDisponivel) {
|
|
mostrarWizard = true;
|
|
}
|
|
}}
|
|
disabled={!funcionarioIdDisponivel}
|
|
title={funcionarioIdDisponivel
|
|
? 'Clique para agendar suas férias'
|
|
: 'Estamos validando seus dados de funcionário...'}
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-5 w-5"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z"
|
|
/>
|
|
</svg>
|
|
Agendar Férias
|
|
</button>
|
|
</div>
|
|
<div class="mt-4 grid grid-cols-1 gap-4 md:grid-cols-1">
|
|
<div class="form-control">
|
|
<label class="label" for="status">
|
|
<span class="label-text">Status</span>
|
|
</label>
|
|
<select
|
|
id="status"
|
|
class="select select-bordered"
|
|
bind:value={filtroStatusFerias}
|
|
>
|
|
<option value="todos">Todos</option>
|
|
<option value="aguardando_aprovacao">Aguardando Aprovação</option>
|
|
<option value="aprovado">Aprovado</option>
|
|
<option value="reprovado">Reprovado</option>
|
|
<option value="data_ajustada_aprovada">Data Ajustada</option>
|
|
<option value="EmFérias">Em Férias</option>
|
|
</select>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Lista de Solicitações -->
|
|
<div class="card bg-base-100 shadow-lg">
|
|
<div class="card-body">
|
|
<h2 class="card-title mb-4 text-lg">
|
|
Minhas Solicitações ({solicitacoesFiltradas.length})
|
|
</h2>
|
|
|
|
{#if solicitacoesFiltradas.length === 0}
|
|
<div class="alert">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
class="stroke-info h-6 w-6 shrink-0"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
></path>
|
|
</svg>
|
|
<span>Nenhuma solicitação encontrada com os filtros aplicados.</span>
|
|
</div>
|
|
{:else}
|
|
<div class="overflow-x-auto">
|
|
<table class="table-zebra table">
|
|
<thead>
|
|
<tr>
|
|
<th>Ano</th>
|
|
<th>Período</th>
|
|
<th>Dias</th>
|
|
<th>Status</th>
|
|
<th>Solicitado em</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{#each solicitacoesFiltradas as periodo (periodo._id)}
|
|
<tr>
|
|
<td>{periodo.anoReferencia}</td>
|
|
<td>
|
|
{formatarDataString(periodo.dataInicio)} - {formatarDataString(
|
|
periodo.dataFim
|
|
)}
|
|
</td>
|
|
<td class="font-bold">{periodo.diasFerias} dias</td>
|
|
<td>
|
|
<div class={`badge ${getStatusBadge(periodo.status)}`}>
|
|
{getStatusTexto(periodo.status)}
|
|
</div>
|
|
</td>
|
|
<td class="text-xs"
|
|
>{new Date(periodo._creationTime).toLocaleDateString('pt-BR')}</td
|
|
>
|
|
</tr>
|
|
{/each}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{:else if abaAtiva === 'minhas-ausencias'}
|
|
<!-- Minhas Ausências -->
|
|
<div class="space-y-6">
|
|
<!-- Estatísticas -->
|
|
<div class="grid grid-cols-1 gap-4 md:grid-cols-4">
|
|
<div class="stat bg-base-100 rounded-box border-base-300 border shadow-lg">
|
|
<div class="stat-figure text-orange-500">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-8 w-8"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
<div class="stat-title">Total</div>
|
|
<div class="stat-value text-orange-500">
|
|
{statsMinhasAusencias.total}
|
|
</div>
|
|
<div class="stat-desc">Solicitações</div>
|
|
</div>
|
|
|
|
<div class="stat bg-base-100 rounded-box border-warning/30 border shadow-lg">
|
|
<div class="stat-figure text-warning">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-8 w-8"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
<div class="stat-title">Aguardando</div>
|
|
<div class="stat-value text-warning">
|
|
{statsMinhasAusencias.aguardando}
|
|
</div>
|
|
<div class="stat-desc">Pendentes</div>
|
|
</div>
|
|
|
|
<div class="stat bg-base-100 rounded-box border-success/30 border shadow-lg">
|
|
<div class="stat-figure text-success">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-8 w-8"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
<div class="stat-title">Aprovadas</div>
|
|
<div class="stat-value text-success">
|
|
{statsMinhasAusencias.aprovadas}
|
|
</div>
|
|
<div class="stat-desc">Deferidas</div>
|
|
</div>
|
|
|
|
<div class="stat bg-base-100 rounded-box border-error/30 border shadow-lg">
|
|
<div class="stat-figure text-error">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-8 w-8"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M10 14l2-2m0 0l2-2m-2 2l-2-2m2 2l2 2m7-2a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
</div>
|
|
<div class="stat-title">Reprovadas</div>
|
|
<div class="stat-value text-error">
|
|
{statsMinhasAusencias.reprovadas}
|
|
</div>
|
|
<div class="stat-desc">Indeferidas</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Calendário de Ausências -->
|
|
<div class="card bg-base-100 shadow-lg">
|
|
<div class="card-body">
|
|
<h2 class="card-title mb-4 text-lg">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-6 w-6 text-orange-500"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M8 7V3m8 4V3m-9 8h10M5 21h14a2 2 0 002-2V7a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z"
|
|
/>
|
|
</svg>
|
|
Calendário de Ausências
|
|
</h2>
|
|
<p class="text-base-content/70 mb-4 text-sm">
|
|
Visualize todas as suas solicitações de ausência no calendário
|
|
</p>
|
|
{#if ausenciasParaCalendario.length > 0}
|
|
<CalendarioAusencias
|
|
ausenciasExistentes={ausenciasParaCalendario}
|
|
readonly={true}
|
|
modoVisualizacao="month"
|
|
/>
|
|
{:else}
|
|
<div class="alert alert-info">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
class="stroke-info h-6 w-6 shrink-0"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
></path>
|
|
</svg>
|
|
<span
|
|
>Você ainda não possui solicitações de ausência. Clique em "Solicitar Ausência"
|
|
para criar uma nova.</span
|
|
>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Filtros e Botão Nova Solicitação -->
|
|
<div class="card bg-base-100 shadow-lg">
|
|
<div class="card-body">
|
|
<div
|
|
class="flex flex-col items-start justify-between gap-4 md:flex-row md:items-center"
|
|
>
|
|
<h2 class="card-title text-lg">Filtros</h2>
|
|
<button
|
|
type="button"
|
|
class="btn btn-warning gap-2"
|
|
onclick={() => {
|
|
const funcionarioId = currentUser?.data?.funcionarioId;
|
|
console.log('🔍 [Perfil] Click no botão - funcionarioId:', funcionarioId);
|
|
if (funcionarioId) {
|
|
mostrarWizardAusencia = true;
|
|
} else {
|
|
alert(
|
|
`Não foi possível identificar seu funcionário.\n\nVerifique no console (F12) o objeto usuario:\n${JSON.stringify(currentUser?.data, null, 2)}\n\nEntre em contato com o suporte se o problema persistir.`
|
|
);
|
|
}
|
|
}}
|
|
disabled={!funcionarioIdDisponivel}
|
|
title={funcionarioIdDisponivel
|
|
? 'Clique para solicitar uma ausência'
|
|
: 'Funcionário não identificado. Entre em contato com o suporte.'}
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-5 w-5"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
Solicitar Ausência
|
|
</button>
|
|
</div>
|
|
<div class="mt-4 grid grid-cols-1 gap-4 md:grid-cols-1">
|
|
<div class="form-control">
|
|
<label class="label" for="status-ausencias">
|
|
<span class="label-text">Status</span>
|
|
</label>
|
|
<select
|
|
id="status-ausencias"
|
|
class="select select-bordered"
|
|
bind:value={filtroStatusAusencias}
|
|
>
|
|
<option value="todos">Todos</option>
|
|
<option value="aguardando_aprovacao">Aguardando Aprovação</option>
|
|
<option value="aprovado">Aprovado</option>
|
|
<option value="reprovado">Reprovado</option>
|
|
</select>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Lista de Solicitações -->
|
|
<div class="card bg-base-100 shadow-lg">
|
|
<div class="card-body">
|
|
<h2 class="card-title mb-4 text-lg">
|
|
Minhas Solicitações ({ausenciasFiltradas.length})
|
|
</h2>
|
|
|
|
{#if ausenciasFiltradas.length === 0}
|
|
<div class="alert">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
class="stroke-info h-6 w-6 shrink-0"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
></path>
|
|
</svg>
|
|
<span>Nenhuma solicitação encontrada com os filtros aplicados.</span>
|
|
</div>
|
|
{:else}
|
|
<div class="overflow-x-auto">
|
|
<table class="table-zebra table">
|
|
<thead>
|
|
<tr>
|
|
<th>Período</th>
|
|
<th>Dias</th>
|
|
<th>Motivo</th>
|
|
<th>Status</th>
|
|
<th>Solicitado em</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{#each ausenciasFiltradas as ausencia (ausencia._id)}
|
|
<tr>
|
|
<td>
|
|
{new Date(ausencia.dataInicio).toLocaleDateString('pt-BR')} até {new Date(
|
|
ausencia.dataFim
|
|
).toLocaleDateString('pt-BR')}
|
|
</td>
|
|
<td class="font-bold">
|
|
{Math.ceil(
|
|
(new Date(ausencia.dataFim).getTime() -
|
|
new Date(ausencia.dataInicio).getTime()) /
|
|
(1000 * 60 * 60 * 24)
|
|
) + 1} dias
|
|
</td>
|
|
<td class="max-w-xs truncate" title={ausencia.motivo}>
|
|
{ausencia.motivo}
|
|
</td>
|
|
<td>
|
|
<div class={`badge ${getStatusBadge(ausencia.status)}`}>
|
|
{getStatusTexto(ausencia.status)}
|
|
</div>
|
|
</td>
|
|
<td class="text-xs"
|
|
>{new Date(ausencia.criadoEm).toLocaleDateString('pt-BR')}</td
|
|
>
|
|
</tr>
|
|
{/each}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{:else if abaAtiva === 'aprovar-ferias'}
|
|
<!-- Aprovar Férias (Gestores) PREMIUM -->
|
|
<div class="card bg-base-100 border-t-4 border-green-500 shadow-2xl">
|
|
<div class="card-body">
|
|
<h2 class="card-title mb-6 flex items-center gap-2 text-2xl">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-7 w-7 text-green-600"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M9 12l2 2 4-4M7.835 4.697a3.42 3.42 0 001.946-.806 3.42 3.42 0 014.438 0 3.42 3.42 0 001.946.806 3.42 3.42 0 013.138 3.138 3.42 3.42 0 00.806 1.946 3.42 3.42 0 010 4.438 3.42 3.42 0 00-.806 1.946 3.42 3.42 0 01-3.138 3.138 3.42 3.42 0 00-1.946.806 3.42 3.42 0 01-4.438 0 3.42 3.42 0 00-1.946-.806 3.42 3.42 0 01-3.138-3.138 3.42 3.42 0 00-.806-1.946 3.42 3.42 0 010-4.438 3.42 3.42 0 00.806-1.946 3.42 3.42 0 013.138-3.138z"
|
|
/>
|
|
</svg>
|
|
Solicitações da Equipe
|
|
<div class="badge badge-lg badge-primary ml-2">
|
|
{solicitacoesSubordinados.length}
|
|
</div>
|
|
</h2>
|
|
|
|
{#if solicitacoesSubordinados.length === 0}
|
|
<div class="alert alert-success">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
class="h-6 w-6 shrink-0 stroke-current"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
></path>
|
|
</svg>
|
|
<span class="font-semibold">Nenhuma solicitação pendente no momento.</span>
|
|
</div>
|
|
{:else}
|
|
<div class="overflow-x-auto">
|
|
<table class="table-zebra table-lg table">
|
|
<thead>
|
|
<tr class="bg-base-200">
|
|
<th class="font-bold">Funcionário</th>
|
|
<th class="font-bold">Time</th>
|
|
<th class="font-bold">Ano</th>
|
|
<th class="font-bold">Período</th>
|
|
<th class="font-bold">Dias</th>
|
|
<th class="font-bold">Status</th>
|
|
<th class="font-bold">Ações</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{#each solicitacoesSubordinados as periodo (periodo._id)}
|
|
<tr class="hover:bg-base-200 transition-colors">
|
|
<td>
|
|
<div class="font-bold">
|
|
{periodo.funcionario?.nome}
|
|
</div>
|
|
</td>
|
|
<td>
|
|
{#if periodo.time}
|
|
<div
|
|
class="badge badge-lg font-semibold"
|
|
style="background-color: {periodo.time.cor}20; border-color: {periodo
|
|
.time.cor}; color: {periodo.time.cor}"
|
|
>
|
|
{periodo.time.nome}
|
|
</div>
|
|
{/if}
|
|
</td>
|
|
<td class="font-semibold">{periodo.anoReferencia}</td>
|
|
<td class="font-semibold">
|
|
{formatarDataString(periodo.dataInicio)} - {formatarDataString(
|
|
periodo.dataFim
|
|
)}
|
|
</td>
|
|
<td class="text-lg font-bold">{periodo.diasFerias}</td>
|
|
<td>
|
|
<div
|
|
class={`badge badge-lg font-semibold ${getStatusBadge(periodo.status)}`}
|
|
>
|
|
{getStatusTexto(periodo.status)}
|
|
</div>
|
|
</td>
|
|
<td>
|
|
{#if periodo.status === 'aguardando_aprovacao'}
|
|
<button
|
|
type="button"
|
|
class="btn btn-primary btn-sm gap-2 shadow-lg transition-transform hover:scale-105"
|
|
onclick={() => selecionarPeriodo(periodo._id)}
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-4 w-4"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"
|
|
/>
|
|
</svg>
|
|
Analisar
|
|
</button>
|
|
{:else}
|
|
<button
|
|
type="button"
|
|
class="btn btn-sm gap-2"
|
|
onclick={() => selecionarPeriodo(periodo._id)}
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-4 w-4"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M15 12a3 3 0 11-6 0 3 3 0 016 0z"
|
|
/>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M2.458 12C3.732 7.943 7.523 5 12 5c4.478 0 8.268 2.943 9.542 7-1.274 4.057-5.064 7-9.542 7-4.477 0-8.268-2.943-9.542-7z"
|
|
/>
|
|
</svg>
|
|
Detalhes
|
|
</button>
|
|
{/if}
|
|
</td>
|
|
</tr>
|
|
{/each}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
{:else if abaAtiva === 'aprovar-ausencias'}
|
|
<!-- Aprovar Ausências (Gestores) -->
|
|
<div class="card bg-base-100 border-t-4 border-orange-500 shadow-2xl">
|
|
<div class="card-body">
|
|
<h2 class="card-title mb-6 flex items-center gap-2 text-2xl">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-7 w-7 text-orange-600"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
Solicitações de Ausências da Equipe
|
|
<div class="badge badge-lg badge-warning ml-2">
|
|
{ausenciasSubordinados.length}
|
|
</div>
|
|
</h2>
|
|
|
|
{#if ausenciasSubordinados.length === 0}
|
|
<div class="alert alert-success">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
class="h-6 w-6 shrink-0 stroke-current"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
></path>
|
|
</svg>
|
|
<span class="font-semibold">Nenhuma solicitação pendente no momento.</span>
|
|
</div>
|
|
{:else}
|
|
<div class="overflow-x-auto">
|
|
<table class="table-zebra table-lg table">
|
|
<thead>
|
|
<tr class="bg-base-200">
|
|
<th class="font-bold">Funcionário</th>
|
|
<th class="font-bold">Time</th>
|
|
<th class="font-bold">Período</th>
|
|
<th class="font-bold">Dias</th>
|
|
<th class="font-bold">Status</th>
|
|
<th class="font-bold">Ações</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{#each ausenciasSubordinados as ausencia (ausencia._id)}
|
|
<tr class="hover:bg-base-200 transition-colors">
|
|
<td>
|
|
<div class="font-bold">
|
|
{ausencia.funcionario?.nome || 'N/A'}
|
|
</div>
|
|
</td>
|
|
<td>
|
|
{#if ausencia.time}
|
|
<div
|
|
class="badge badge-lg font-semibold"
|
|
style="background-color: {ausencia.time
|
|
.cor}20; border-color: {ausencia.time.cor}; color: {ausencia.time
|
|
.cor}"
|
|
>
|
|
{ausencia.time.nome}
|
|
</div>
|
|
{/if}
|
|
</td>
|
|
<td class="font-semibold">
|
|
{new Date(ausencia.dataInicio).toLocaleDateString('pt-BR')} até
|
|
{new Date(ausencia.dataFim).toLocaleDateString('pt-BR')}
|
|
</td>
|
|
<td class="text-lg font-bold">
|
|
{Math.ceil(
|
|
(new Date(ausencia.dataFim).getTime() -
|
|
new Date(ausencia.dataInicio).getTime()) /
|
|
(1000 * 60 * 60 * 24)
|
|
) + 1} dias
|
|
</td>
|
|
<td>
|
|
<div
|
|
class={`badge badge-lg font-semibold ${getStatusBadge(ausencia.status)}`}
|
|
>
|
|
{getStatusTexto(ausencia.status)}
|
|
</div>
|
|
</td>
|
|
<td>
|
|
{#if ausencia.status === 'aguardando_aprovacao'}
|
|
<button
|
|
type="button"
|
|
class="btn btn-warning btn-sm gap-2 shadow-lg transition-transform hover:scale-105"
|
|
onclick={() => (solicitacaoAusenciaAprovar = ausencia._id)}
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-4 w-4"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M9 5H7a2 2 0 00-2 2v12a2 2 0 002 2h10a2 2 0 002-2V7a2 2 0 00-2-2h-2M9 5a2 2 0 002 2h2a2 2 0 002-2M9 5a2 2 0 012-2h2a2 2 0 012 2m-6 9l2 2 4-4"
|
|
/>
|
|
</svg>
|
|
Analisar
|
|
</button>
|
|
{:else}
|
|
<button
|
|
type="button"
|
|
class="btn btn-sm gap-2"
|
|
onclick={() => (solicitacaoAusenciaAprovar = ausencia._id)}
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-4 w-4"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M15 12a3 3 0 11-6 0 3 3 0 016 0z"
|
|
/>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M2.458 12C3.732 7.943 7.523 5 12 5c4.478 0 8.268 2.943 9.542 7-1.274 4.057-5.064 7-9.542 7-4.477 0-8.268-2.943-9.542-7z"
|
|
/>
|
|
</svg>
|
|
Detalhes
|
|
</button>
|
|
{/if}
|
|
</td>
|
|
</tr>
|
|
{/each}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
|
|
<!-- Modal de Aprovação de Férias -->
|
|
{#if periodoSelecionado && currentUser.data}
|
|
{#await client.query( api.ferias.obterDetalhes, { feriasId: periodoSelecionado } ) then detalhes}
|
|
{#if detalhes}
|
|
<dialog class="modal modal-open">
|
|
<div class="modal-box max-w-4xl">
|
|
<AprovarFerias
|
|
periodo={detalhes}
|
|
gestorId={currentUser.data._id}
|
|
onSucesso={recarregar}
|
|
onCancelar={() => (periodoSelecionado = null)}
|
|
/>
|
|
</div>
|
|
<form method="dialog" class="modal-backdrop">
|
|
<button
|
|
type="button"
|
|
onclick={() => (periodoSelecionado = null)}
|
|
aria-label="Fechar modal">Fechar</button
|
|
>
|
|
</form>
|
|
</dialog>
|
|
{/if}
|
|
{/await}
|
|
{/if}
|
|
|
|
<!-- Modal de Upload de Foto / Escolher Avatar -->
|
|
{#if mostrarModalFoto}
|
|
<dialog class="modal modal-open">
|
|
<div class="modal-box max-h-[90vh] max-w-4xl overflow-y-auto">
|
|
<h3
|
|
class="mb-8 bg-linear-to-r from-purple-600 to-blue-600 bg-clip-text text-center text-3xl font-black text-transparent"
|
|
>
|
|
Alterar Foto de Perfil
|
|
</h3>
|
|
|
|
<!-- Preview da foto atual -->
|
|
<div class="mb-8 flex justify-center">
|
|
<div class="avatar">
|
|
<div
|
|
class="ring-primary ring-offset-base-100 h-32 w-32 rounded-full shadow-2xl ring-4 ring-offset-4"
|
|
>
|
|
{#if currentUser.data?.fotoPerfilUrl}
|
|
<img src={currentUser.data.fotoPerfilUrl} alt="Foto atual" class="object-cover" />
|
|
{:else if currentUser.data?.avatar}
|
|
<img src={currentUser.data.avatar} alt="Avatar atual" class="object-cover" />
|
|
{:else}
|
|
<div class="bg-primary text-primary-content flex items-center justify-center">
|
|
<span class="text-4xl font-bold"
|
|
>{currentUser.data?.nome.substring(0, 2).toUpperCase()}</span
|
|
>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Tabs: Avatar ou Upload -->
|
|
<div
|
|
role="tablist"
|
|
class="tabs tabs-boxed from-base-200 to-base-300 mb-8 bg-linear-to-r p-2 shadow-xl"
|
|
>
|
|
<button
|
|
type="button"
|
|
role="tab"
|
|
class={`tab tab-lg font-semibold transition-all ${modoFoto === 'avatar' ? 'tab-active bg-linear-to-r from-purple-600 to-blue-600 text-white' : ''}`}
|
|
onclick={() => (modoFoto = 'avatar')}
|
|
disabled={uploadandoFoto}
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="mr-2 h-5 w-5"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M14.828 14.828a4 4 0 01-5.656 0M9 10h.01M15 10h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
Escolher Avatar
|
|
</button>
|
|
<button
|
|
type="button"
|
|
role="tab"
|
|
class={`tab tab-lg font-semibold transition-all ${modoFoto === 'upload' ? 'tab-active bg-linear-to-r from-purple-600 to-blue-600 text-white' : ''}`}
|
|
onclick={() => (modoFoto = 'upload')}
|
|
disabled={uploadandoFoto}
|
|
>
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="mr-2 h-5 w-5"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M4 16l4.586-4.586a2 2 0 012.828 0L16 16m-2-2l1.586-1.586a2 2 0 012.828 0L20 14m-6-6h.01M6 20h12a2 2 0 002-2V6a2 2 0 00-2-2H6a2 2 0 00-2 2v12a2 2 0 002 2z"
|
|
/>
|
|
</svg>
|
|
Enviar Foto
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Conteúdo das Tabs -->
|
|
{#if modoFoto === 'avatar'}
|
|
<!-- Galeria de Avatares -->
|
|
<div class="mb-4">
|
|
<p class="text-base-content/70 mb-6 text-center text-lg">
|
|
Escolha um dos <strong class="text-primary">30 avatares profissionais</strong> para seu
|
|
perfil
|
|
</p>
|
|
|
|
<div
|
|
class="bg-base-200 grid max-h-[500px] grid-cols-3 gap-4 overflow-y-auto rounded-2xl p-6 shadow-inner md:grid-cols-5 lg:grid-cols-6"
|
|
>
|
|
{#each avatarGallery as avatar (avatar.id)}
|
|
<button
|
|
type="button"
|
|
class={`flex cursor-pointer flex-col items-center rounded-xl p-2 transition-all hover:scale-110 ${avatarSelecionado === avatar.url ? 'ring-primary bg-primary/10 scale-105 ring-4' : 'hover:ring-primary/50 hover:bg-base-100 hover:ring-2'}`}
|
|
onclick={() => (avatarSelecionado = avatar.url)}
|
|
ondblclick={() => handleSelecionarAvatar(avatar.url)}
|
|
disabled={uploadandoFoto}
|
|
aria-label="Selecionar avatar {avatar.name}"
|
|
>
|
|
<div class="avatar">
|
|
<div class="h-20 w-20 rounded-full shadow-lg">
|
|
<img src={avatar.url} alt={avatar.name} loading="lazy" />
|
|
</div>
|
|
</div>
|
|
<div class="mt-2 w-full truncate text-center text-[10px] font-semibold">
|
|
{avatar.name}
|
|
</div>
|
|
</button>
|
|
{/each}
|
|
</div>
|
|
|
|
<div class="alert alert-info mt-6 shadow-lg">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
class="h-6 w-6 shrink-0 stroke-current"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
></path>
|
|
</svg>
|
|
<span>
|
|
<strong>Dica:</strong> Clique uma vez para selecionar, clique duas vezes para aplicar
|
|
imediatamente!
|
|
</span>
|
|
</div>
|
|
</div>
|
|
|
|
{#if avatarSelecionado}
|
|
<div class="mt-6 flex justify-center gap-3">
|
|
<button
|
|
type="button"
|
|
class="btn btn-lg gap-2 shadow-xl transition-all hover:scale-105"
|
|
style="background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); color: white; border: none;"
|
|
onclick={() => handleSelecionarAvatar(avatarSelecionado)}
|
|
disabled={uploadandoFoto}
|
|
>
|
|
{#if uploadandoFoto}
|
|
<span class="loading loading-spinner"></span>
|
|
{:else}
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-6 w-6"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
stroke="currentColor"
|
|
stroke-width="2.5"
|
|
>
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M5 13l4 4L19 7" />
|
|
</svg>
|
|
{/if}
|
|
{uploadandoFoto ? 'Salvando...' : 'Confirmar Avatar'}
|
|
</button>
|
|
</div>
|
|
{/if}
|
|
{:else}
|
|
<!-- Upload de nova foto -->
|
|
<div class="form-control">
|
|
<label class="label" for="foto-upload">
|
|
<span class="label-text text-lg font-bold">Selecionar nova foto</span>
|
|
</label>
|
|
<input
|
|
id="foto-upload"
|
|
type="file"
|
|
class="file-input file-input-bordered file-input-lg w-full shadow-lg"
|
|
accept="image/*"
|
|
onchange={handleUploadFoto}
|
|
disabled={uploadandoFoto}
|
|
/>
|
|
<div class="label">
|
|
<span class="label-text-alt text-base-content/70"
|
|
>Formatos aceitos: JPG, PNG, GIF. Tamanho máximo: 5MB</span
|
|
>
|
|
</div>
|
|
</div>
|
|
{/if}
|
|
|
|
{#if erroUpload}
|
|
<div class="alert alert-error mt-6 shadow-lg">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="h-6 w-6 shrink-0 stroke-current"
|
|
fill="none"
|
|
viewBox="0 0 24 24"
|
|
>
|
|
<path
|
|
stroke-linecap="round"
|
|
stroke-linejoin="round"
|
|
stroke-width="2"
|
|
d="M10 14l2-2m0 0l2-2m-2 2l-2-2m2 2l2 2m7-2a9 9 0 11-18 0 9 9 0 0118 0z"
|
|
/>
|
|
</svg>
|
|
<span class="font-semibold">{erroUpload}</span>
|
|
</div>
|
|
{/if}
|
|
|
|
{#if uploadandoFoto && modoFoto === 'upload'}
|
|
<div class="mt-6 flex items-center justify-center gap-3">
|
|
<span class="loading loading-spinner loading-lg text-primary"></span>
|
|
<span class="text-lg font-semibold">Enviando foto...</span>
|
|
</div>
|
|
{/if}
|
|
|
|
<div class="modal-action mt-8">
|
|
<button
|
|
type="button"
|
|
class="btn btn-lg"
|
|
onclick={() => (mostrarModalFoto = false)}
|
|
disabled={uploadandoFoto}
|
|
>
|
|
Cancelar
|
|
</button>
|
|
</div>
|
|
</div>
|
|
<form method="dialog" class="modal-backdrop">
|
|
<button
|
|
type="button"
|
|
onclick={() => (mostrarModalFoto = false)}
|
|
aria-label="Fechar modal"
|
|
disabled={uploadandoFoto}>Fechar</button
|
|
>
|
|
</form>
|
|
</dialog>
|
|
{/if}
|
|
</main>
|
|
</ProtectedRoute>
|
|
|
|
<!-- Modal Wizard Solicitação de Férias -->
|
|
{#if mostrarWizard && funcionarioIdDisponivel}
|
|
<dialog class="modal modal-open">
|
|
<div class="modal-box max-h-[90vh] max-w-4xl overflow-hidden">
|
|
<h3 class="mb-6 text-center text-2xl font-bold">Nova Solicitação de Férias</h3>
|
|
<div class="max-h-[80vh] overflow-y-auto">
|
|
<WizardSolicitacaoFerias
|
|
funcionarioId={funcionarioIdDisponivel}
|
|
onSucesso={() => {
|
|
mostrarWizard = false;
|
|
}}
|
|
onCancelar={() => (mostrarWizard = false)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div
|
|
class="modal-backdrop"
|
|
role="button"
|
|
tabindex="0"
|
|
onclick={() => (mostrarWizard = false)}
|
|
onkeydown={(event) => {
|
|
if (event.key === 'Enter' || event.key === ' ') {
|
|
event.preventDefault();
|
|
mostrarWizard = false;
|
|
}
|
|
}}
|
|
></div>
|
|
</dialog>
|
|
{/if}
|
|
|
|
<!-- Modal Wizard Solicitação de Ausência -->
|
|
{#if mostrarWizardAusencia && funcionarioIdDisponivel}
|
|
<dialog
|
|
class="modal modal-open"
|
|
onclick={(e) => e.target === e.currentTarget && (mostrarWizardAusencia = false)}
|
|
>
|
|
<div
|
|
class="modal-box flex max-h-[90vh] max-w-4xl flex-col p-0"
|
|
onclick={(e) => e.stopPropagation()}
|
|
>
|
|
<!-- Header -->
|
|
<div class="border-base-300 flex items-center justify-between border-b px-6 py-4">
|
|
<h2 id="modal-title" class="flex items-center gap-2 text-xl font-bold">
|
|
<Calendar class="text-primary h-5 w-5" />
|
|
Nova Solicitação de Ausência
|
|
</h2>
|
|
<button
|
|
type="button"
|
|
class="btn btn-sm btn-circle"
|
|
onclick={() => (mostrarWizardAusencia = false)}
|
|
aria-label="Fechar"
|
|
>
|
|
<X class="h-5 w-5" />
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Content -->
|
|
<div class="flex-1 overflow-y-auto p-6">
|
|
<WizardSolicitacaoAusencia
|
|
funcionarioId={funcionarioIdDisponivel}
|
|
onSucesso={async () => {
|
|
mostrarWizardAusencia = false;
|
|
// As queries do Convex são reativas e devem atualizar automaticamente
|
|
// Mas garantimos que o componente será re-renderizado
|
|
// Forçar recarregamento das queries mudando temporariamente o filtro
|
|
const filtroAnterior = filtroStatusAusencias;
|
|
if (filtroAnterior !== 'todos') {
|
|
filtroStatusAusencias = 'todos';
|
|
await new Promise((resolve) => setTimeout(resolve, 200));
|
|
}
|
|
filtroStatusAusencias = filtroAnterior;
|
|
}}
|
|
onCancelar={() => (mostrarWizardAusencia = false)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<form method="dialog" class="modal-backdrop">
|
|
<button type="button" onclick={() => (mostrarWizardAusencia = false)}>fechar</button>
|
|
</form>
|
|
</dialog>
|
|
{/if}
|
|
|
|
<!-- Modal de Aprovação de Ausências -->
|
|
{#if solicitacaoAusenciaAprovar && currentUser.data}
|
|
{#await client.query( api.ausencias.obterDetalhes, { solicitacaoId: solicitacaoAusenciaAprovar } ) then detalhes}
|
|
{#if detalhes}
|
|
<dialog class="modal modal-open">
|
|
<div class="modal-box max-w-4xl">
|
|
<AprovarAusencias
|
|
solicitacao={detalhes}
|
|
gestorId={currentUser.data._id}
|
|
onSucesso={() => {
|
|
solicitacaoAusenciaAprovar = null;
|
|
}}
|
|
onCancelar={() => (solicitacaoAusenciaAprovar = null)}
|
|
/>
|
|
</div>
|
|
<form method="dialog" class="modal-backdrop">
|
|
<button
|
|
type="button"
|
|
onclick={() => (solicitacaoAusenciaAprovar = null)}
|
|
aria-label="Fechar modal">Fechar</button
|
|
>
|
|
</form>
|
|
</dialog>
|
|
{/if}
|
|
{/await}
|
|
{/if}
|
|
|
|
<style>
|
|
@keyframes gradient-shift {
|
|
0%,
|
|
100% {
|
|
background-position: 0% 50%;
|
|
}
|
|
50% {
|
|
background-position: 100% 50%;
|
|
}
|
|
}
|
|
|
|
@keyframes float {
|
|
0%,
|
|
100% {
|
|
transform: translateY(0px);
|
|
}
|
|
50% {
|
|
transform: translateY(-10px);
|
|
}
|
|
}
|
|
|
|
.animate-gradient {
|
|
background-size: 200% 200%;
|
|
animation: gradient-shift 15s ease infinite;
|
|
}
|
|
|
|
.animate-float {
|
|
animation: float 6s ease-in-out infinite;
|
|
}
|
|
</style>
|