Ajustes gerais - Etapa 1 #52

Merged
deyvisonwanderley merged 30 commits from ajustes_gerais into master 2025-12-01 18:00:39 +00:00
39 changed files with 9588 additions and 3562 deletions
Showing only changes of commit a149c5ead6 - Show all commits

View File

@@ -2,8 +2,13 @@ import { api } from "@sgse-app/backend/convex/_generated/api";
import { createConvexHttpClient } from "@mmailaender/convex-better-auth-svelte/sveltekit"; import { createConvexHttpClient } from "@mmailaender/convex-better-auth-svelte/sveltekit";
export const load = async ({ locals }) => { export const load = async ({ locals }) => {
try {
const client = createConvexHttpClient({ token: locals.token }); const client = createConvexHttpClient({ token: locals.token });
const currentUser = await client.query(api.auth.getCurrentUser, {}); const currentUser = await client.query(api.auth.getCurrentUser, {});
return { currentUser }; return { currentUser };
} catch (error) {
console.error("Erro ao carregar usuário atual no layout do dashboard:", error);
// Evita quebrar toda a área logada em caso de falha transitória na API/auth
return { currentUser: null };
}
}; };

View File

@@ -96,14 +96,17 @@
} }
</script> </script>
<div class="container mx-auto max-w-7xl px-4 py-6"> <div class="container mx-auto max-w-7xl px-4 py-8">
<div
class="rounded-2xl bg-base-100/80 shadow-xl border border-base-200/60 p-6 lg:p-8 space-y-6 backdrop-blur"
>
<!-- Header --> <!-- Header -->
<div class="mb-6 flex items-center justify-between"> <div class="flex flex-col gap-4 lg:flex-row lg:items-center lg:justify-between">
<div class="flex items-center gap-4"> <div class="flex items-center gap-4">
<div class="bg-info/10 rounded-xl p-3"> <div class="bg-gradient-to-br from-info/15 via-primary/10 to-secondary/10 rounded-2xl p-3">
<svg <svg
xmlns="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg"
class="text-info h-8 w-8" class="text-info h-9 w-9"
fill="none" fill="none"
viewBox="0 0 24 24" viewBox="0 0 24 24"
stroke="currentColor" stroke="currentColor"
@@ -118,10 +121,13 @@
</div> </div>
<div> <div>
<h1 class="text-base-content text-3xl font-bold">Gerenciar Templates</h1> <h1 class="text-base-content text-3xl font-bold">Gerenciar Templates</h1>
<p class="text-base-content/60 mt-1">Criar, editar e excluir templates de emails e mensagens</p> <p class="text-base-content/60 mt-1 text-sm lg:text-base">
Crie, edite e organize templates de <span class="font-semibold">chat</span> (texto puro) e
<span class="font-semibold">email HTML padronizado</span> usados em todas as notificações do SGSE.
</p>
</div> </div>
</div> </div>
<a href="/ti/notificacoes" class="btn btn-primary"> <a href="/ti/notificacoes" class="btn btn-outline gap-2">
<svg <svg
xmlns="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg"
class="h-5 w-5" class="h-5 w-5"
@@ -131,14 +137,14 @@
> >
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10 19l-7-7m0 0l7-7m-7 7h18" /> <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10 19l-7-7m0 0l7-7m-7 7h18" />
</svg> </svg>
Voltar Voltar para Notificações
</a> </a>
</div> </div>
<!-- Mensagens de Feedback --> <!-- Mensagens de Feedback -->
{#if mensagem} {#if mensagem}
<div <div
class="alert mb-6 shadow-lg" class="alert shadow-lg"
class:alert-success={mensagem.tipo === 'success'} class:alert-success={mensagem.tipo === 'success'}
class:alert-error={mensagem.tipo === 'error'} class:alert-error={mensagem.tipo === 'error'}
class:alert-info={mensagem.tipo === 'info'} class:alert-info={mensagem.tipo === 'info'}
@@ -151,7 +157,7 @@
{/if} {/if}
<!-- Filtros e Busca --> <!-- Filtros e Busca -->
<div class="card bg-base-100 mb-6 shadow-xl"> <div class="card bg-base-100 shadow-sm border border-base-200">
<div class="card-body"> <div class="card-body">
<div class="grid grid-cols-1 gap-4 md:grid-cols-2"> <div class="grid grid-cols-1 gap-4 md:grid-cols-2">
<div class="form-control"> <div class="form-control">
@@ -181,11 +187,11 @@
</div> </div>
<!-- Lista de Templates --> <!-- Lista de Templates -->
<div class="card bg-base-100 shadow-xl"> <div class="card bg-base-100 shadow-sm border border-base-200">
<div class="card-body"> <div class="card-body">
<div class="mb-4 flex items-center justify-between"> <div class="mb-4 flex items-center justify-between">
<h2 class="card-title">Templates ({templatesFiltrados.length})</h2> <h2 class="card-title">Templates ({templatesFiltrados.length})</h2>
<a href="/ti/notificacoes/templates/novo" class="btn btn-primary"> <a href="/ti/notificacoes/templates/novo" class="btn btn-primary gap-2">
<svg <svg
xmlns="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg"
class="h-5 w-5" class="h-5 w-5"
@@ -311,4 +317,5 @@
</div> </div>
</div> </div>
</div> </div>
</div>

View File

@@ -110,13 +110,16 @@
} }
</script> </script>
<div class="container mx-auto max-w-4xl px-4 py-6"> <div class="container mx-auto max-w-4xl px-4 py-8">
<div class="mb-6 flex items-center justify-between"> <div
class="rounded-2xl bg-base-100/80 shadow-xl border border-base-200/60 p-6 lg:p-8 space-y-6 backdrop-blur"
>
<div class="mb-4 flex items-center justify-between">
<div class="flex items-center gap-4"> <div class="flex items-center gap-4">
<div class="bg-info/10 rounded-xl p-3"> <div class="bg-gradient-to-br from-info/15 via-primary/10 to-secondary/10 rounded-2xl p-3">
<svg <svg
xmlns="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg"
class="text-info h-8 w-8" class="text-info h-9 w-9"
fill="none" fill="none"
viewBox="0 0 24 24" viewBox="0 0 24 24"
stroke="currentColor" stroke="currentColor"
@@ -131,13 +134,13 @@
</div> </div>
<div> <div>
<h1 class="text-base-content text-3xl font-bold">Editar Template</h1> <h1 class="text-base-content text-3xl font-bold">Editar Template</h1>
<p class="text-base-content/60 mt-1"> <p class="text-base-content/60 mt-1 text-sm lg:text-base">
Atualize as informações do template selecionado. Templates de sistema não podem ser Ajuste o texto base usado em <span class="font-semibold">chat</span> e na versão HTML de
editados. <span class="font-semibold">email</span>. Templates de sistema podem ter restrições de edição.
</p> </p>
</div> </div>
</div> </div>
<a href={resolve('/ti/notificacoes/templates')} class="btn btn-ghost"> Voltar </a> <a href={resolve('/ti/notificacoes/templates')} class="btn btn-outline"> Voltar </a>
</div> </div>
{#if !template} {#if !template}
@@ -147,7 +150,7 @@
{:else} {:else}
{#if mensagem} {#if mensagem}
<div <div
class="alert mb-6 shadow-lg" class="alert mb-4 shadow-lg"
class:alert-success={mensagem.tipo === 'success'} class:alert-success={mensagem.tipo === 'success'}
class:alert-error={mensagem.tipo === 'error'} class:alert-error={mensagem.tipo === 'error'}
class:alert-info={mensagem.tipo === 'info'} class:alert-info={mensagem.tipo === 'info'}
@@ -165,7 +168,7 @@
</div> </div>
{/if} {/if}
<div class="card bg-base-100 shadow-xl"> <div class="card bg-base-100 shadow-sm border border-base-200">
<div class="card-body space-y-4"> <div class="card-body space-y-4">
<div class="grid grid-cols-1 gap-4 md:grid-cols-2"> <div class="grid grid-cols-1 gap-4 md:grid-cols-2">
<div class="form-control"> <div class="form-control">
@@ -277,6 +280,6 @@
</button> </button>
</div> </div>
</div> </div>
</div>
{/if} {/if}
</div> </div>
</div>

View File

@@ -79,13 +79,16 @@
} }
</script> </script>
<div class="container mx-auto max-w-4xl px-4 py-6"> <div class="container mx-auto max-w-4xl px-4 py-8">
<div class="mb-6 flex items-center justify-between"> <div
class="rounded-2xl bg-base-100/80 shadow-xl border border-base-200/60 p-6 lg:p-8 space-y-6 backdrop-blur"
>
<div class="mb-4 flex items-center justify-between">
<div class="flex items-center gap-4"> <div class="flex items-center gap-4">
<div class="bg-info/10 rounded-xl p-3"> <div class="bg-gradient-to-br from-info/15 via-primary/10 to-secondary/10 rounded-2xl p-3">
<svg <svg
xmlns="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg"
class="text-info h-8 w-8" class="text-info h-9 w-9"
fill="none" fill="none"
viewBox="0 0 24 24" viewBox="0 0 24 24"
stroke="currentColor" stroke="currentColor"
@@ -100,17 +103,18 @@
</div> </div>
<div> <div>
<h1 class="text-base-content text-3xl font-bold">Novo Template</h1> <h1 class="text-base-content text-3xl font-bold">Novo Template</h1>
<p class="text-base-content/60 mt-1"> <p class="text-base-content/60 mt-1 text-sm lg:text-base">
Crie um template de email ou mensagem para reutilizar no sistema. Defina o texto base que será usado em <span class="font-semibold">chat</span> e na versão
HTML de <span class="font-semibold">email</span> com o estilo padrão do SGSE.
</p> </p>
</div> </div>
</div> </div>
<a href={resolve('/ti/notificacoes/templates')} class="btn btn-ghost"> Voltar </a> <a href={resolve('/ti/notificacoes/templates')} class="btn btn-outline"> Voltar </a>
</div> </div>
{#if mensagem} {#if mensagem}
<div <div
class="alert mb-6 shadow-lg" class="alert mb-4 shadow-lg"
class:alert-success={mensagem.tipo === 'success'} class:alert-success={mensagem.tipo === 'success'}
class:alert-error={mensagem.tipo === 'error'} class:alert-error={mensagem.tipo === 'error'}
class:alert-info={mensagem.tipo === 'info'} class:alert-info={mensagem.tipo === 'info'}
@@ -128,7 +132,7 @@
</div> </div>
{/if} {/if}
<div class="card bg-base-100 shadow-xl"> <div class="card bg-base-100 shadow-sm border border-base-200">
<div class="card-body space-y-4"> <div class="card-body space-y-4">
<div class="grid grid-cols-1 gap-4 md:grid-cols-2"> <div class="grid grid-cols-1 gap-4 md:grid-cols-2">
<div class="form-control"> <div class="form-control">

View File

@@ -1,6 +1,6 @@
import { v } from 'convex/values'; import { v } from 'convex/values';
import { mutation, query, internalMutation } from './_generated/server'; import { mutation, query, internalMutation } from './_generated/server';
import { internal } from './_generated/api'; import { internal, api } from './_generated/api';
import { Id } from './_generated/dataModel'; import { Id } from './_generated/dataModel';
import type { QueryCtx } from './_generated/server'; import type { QueryCtx } from './_generated/server';
@@ -387,7 +387,9 @@ export const verificarAlertasInternal = internalMutation({
threshold: alerta.threshold.toString() threshold: alerta.threshold.toString()
}; };
await ctx.scheduler.runAfter(0, internal.email.enviarEmailComTemplate, { // Importante: usar api.email.enviarEmailComTemplate (action pública),
// e não internal.email, para corresponder à tipagem gerada em ./_generated/api.
await ctx.scheduler.runAfter(0, api.email.enviarEmailComTemplate, {
destinatario: email, destinatario: email,
destinatarioId: usuario._id, destinatarioId: usuario._id,
templateCodigo: 'monitoramento_alerta_sistema', templateCodigo: 'monitoramento_alerta_sistema',