feat: enhance vacation approval process by adding notification system for employees, including email alerts and in-app notifications; improve error handling and user feedback during vacation management

This commit is contained in:
2025-12-10 06:27:25 -03:00
parent 73da995109
commit d27c0b6f91
22 changed files with 1572 additions and 215 deletions

View File

@@ -4,6 +4,8 @@ import { internal } from './_generated/api';
import { Id, Doc } from './_generated/dataModel';
import { verificarLicencaAtiva } from './atestadosLicencas';
import { getCurrentUserFunction } from './auth';
import { formatarDataBR } from './utils/datas';
import { api } from './_generated/api';
// Validador para períodos
const periodoValidator = v.object({
@@ -433,13 +435,58 @@ export const aprovar = mutation({
.first();
if (usuario) {
// Criar notificação in-app para funcionário
await ctx.db.insert('notificacoesFerias', {
destinatarioId: usuario._id,
feriasId: registro._id,
tipo: 'aprovado',
lida: false,
mensagem: `Período de férias de ${registro.diasFerias} dias foi aprovado!`
mensagem: `Período de férias de ${formatarDataBR(registro.dataInicio)} até ${formatarDataBR(registro.dataFim)} (${registro.diasFerias} dias) foi aprovado por ${nomeGestor}!`
});
// Enviar email ao funcionário usando template (agendado via scheduler)
if (gestorUsuario) {
// Obter URL do sistema
let urlSistema = process.env.FRONTEND_URL || 'http://localhost:5173';
if (!urlSistema.match(/^https?:\/\//i)) {
urlSistema = `http://${urlSistema}`;
}
try {
await ctx.scheduler.runAfter(0, api.email.enviarEmailComTemplate, {
destinatario: usuario.email,
destinatarioId: usuario._id,
templateCodigo: 'ferias_aprovada',
variaveis: {
funcionarioNome: usuario.nome,
gestorNome: gestorUsuario.nome,
dataInicio: formatarDataBR(registro.dataInicio),
dataFim: formatarDataBR(registro.dataFim),
diasFerias: registro.diasFerias.toString(),
urlSistema
},
enviadoPor: args.gestorId
});
} catch (error) {
// Fallback para envio direto se houver erro ao agendar ou processar o template
console.warn(
'Erro ao agendar envio de email com template ferias_aprovada, usando envio direto:',
error
);
await ctx.runMutation(api.email.enfileirarEmail, {
destinatario: usuario.email,
destinatarioId: usuario._id,
assunto: 'Solicitação de Férias Aprovada',
corpo: `<p>Olá ${usuario.nome},</p>
<p>Sua solicitação de férias foi <strong>aprovada</strong> pelo gestor ${gestorUsuario.nome}:</p>
<ul>
<li><strong>Período:</strong> ${formatarDataBR(registro.dataInicio)} até ${formatarDataBR(registro.dataFim)}</li>
<li><strong>Dias:</strong> ${registro.diasFerias} dias</li>
</ul>`,
enviadoPor: args.gestorId
});
}
}
}
}
@@ -553,6 +600,10 @@ export const ajustarEAprovar = mutation({
]
});
// Buscar nome do gestor
const gestorUsuario = await ctx.db.get(args.gestorId);
const nomeGestor = gestorUsuario?.nome || 'Gestor';
// Notificar funcionário
if (funcionario) {
const usuario = await ctx.db
@@ -561,13 +612,58 @@ export const ajustarEAprovar = mutation({
.first();
if (usuario) {
// Criar notificação in-app para funcionário
await ctx.db.insert('notificacoesFerias', {
destinatarioId: usuario._id,
feriasId: registroAntigo._id,
tipo: 'data_ajustada',
lida: false,
mensagem: `Período de férias foi aprovado com ajuste de datas: ${args.novaDataInicio} a ${args.novaDataFim}`
mensagem: `Período de férias foi aprovado com ajuste de datas: ${formatarDataBR(args.novaDataInicio)} a ${formatarDataBR(args.novaDataFim)} (${novosDias} dias) por ${nomeGestor}`
});
// Enviar email ao funcionário usando template (agendado via scheduler)
if (gestorUsuario) {
// Obter URL do sistema
let urlSistema = process.env.FRONTEND_URL || 'http://localhost:5173';
if (!urlSistema.match(/^https?:\/\//i)) {
urlSistema = `http://${urlSistema}`;
}
try {
await ctx.scheduler.runAfter(0, api.email.enviarEmailComTemplate, {
destinatario: usuario.email,
destinatarioId: usuario._id,
templateCodigo: 'ferias_aprovada',
variaveis: {
funcionarioNome: usuario.nome,
gestorNome: gestorUsuario.nome,
dataInicio: formatarDataBR(args.novaDataInicio),
dataFim: formatarDataBR(args.novaDataFim),
diasFerias: novosDias.toString(),
urlSistema
},
enviadoPor: args.gestorId
});
} catch (error) {
// Fallback para envio direto se houver erro ao agendar ou processar o template
console.warn(
'Erro ao agendar envio de email com template ferias_aprovada, usando envio direto:',
error
);
await ctx.runMutation(api.email.enfileirarEmail, {
destinatario: usuario.email,
destinatarioId: usuario._id,
assunto: 'Solicitação de Férias Aprovada (com Ajuste de Datas)',
corpo: `<p>Olá ${usuario.nome},</p>
<p>Sua solicitação de férias foi <strong>aprovada com ajuste de datas</strong> pelo gestor ${gestorUsuario.nome}:</p>
<ul>
<li><strong>Período:</strong> ${formatarDataBR(args.novaDataInicio)} até ${formatarDataBR(args.novaDataFim)}</li>
<li><strong>Dias:</strong> ${novosDias} dias</li>
</ul>`,
enviadoPor: args.gestorId
});
}
}
}
}
@@ -718,6 +814,111 @@ export const atualizarStatus = mutation({
);
}
// Se o status foi alterado para Cancelado_RH, notificar o funcionário
if (args.novoStatus === 'Cancelado_RH') {
const funcionario = await ctx.db.get(registro.funcionarioId);
if (funcionario) {
const funcionarioUsuario = await ctx.db
.query('usuarios')
.withIndex('by_funcionarioId', (q) => q.eq('funcionarioId', funcionario._id))
.first();
if (funcionarioUsuario) {
// Buscar usuário do RH que está cancelando
const usuarioRH = await ctx.db.get(args.usuarioId);
const nomeRH = usuarioRH?.nome || 'Recursos Humanos';
// Criar notificação in-app para funcionário
await ctx.db.insert('notificacoesFerias', {
destinatarioId: funcionarioUsuario._id,
feriasId: registro._id,
tipo: 'cancelado',
lida: false,
mensagem: `Sua solicitação de férias de ${formatarDataBR(registro.dataInicio)} até ${formatarDataBR(registro.dataFim)} (${registro.diasFerias} dias) foi cancelada pelo setor de Recursos Humanos.`
});
// Obter URL do sistema
let urlSistema = process.env.FRONTEND_URL || 'http://localhost:5173';
if (!urlSistema.match(/^https?:\/\//i)) {
urlSistema = `http://${urlSistema}`;
}
// Enviar email ao funcionário usando template (agendado via scheduler)
try {
await ctx.scheduler.runAfter(0, api.email.enviarEmailComTemplate, {
destinatario: funcionarioUsuario.email,
destinatarioId: funcionarioUsuario._id,
templateCodigo: 'ferias_cancelada_rh',
variaveis: {
funcionarioNome: funcionarioUsuario.nome,
dataInicio: formatarDataBR(registro.dataInicio),
dataFim: formatarDataBR(registro.dataFim),
diasFerias: registro.diasFerias.toString(),
urlSistema
},
enviadoPor: args.usuarioId
});
} catch (error) {
// Fallback para envio direto se houver erro ao agendar ou processar o template
console.warn(
'Erro ao agendar envio de email com template ferias_cancelada_rh, usando envio direto:',
error
);
await ctx.runMutation(api.email.enfileirarEmail, {
destinatario: funcionarioUsuario.email,
destinatarioId: funcionarioUsuario._id,
assunto: 'Solicitação de Férias Cancelada',
corpo: `<p>Olá ${funcionarioUsuario.nome},</p>
<p>Sua solicitação de férias foi <strong>cancelada</strong> pelo setor de Recursos Humanos:</p>
<ul>
<li><strong>Período:</strong> ${formatarDataBR(registro.dataInicio)} até ${formatarDataBR(registro.dataFim)}</li>
<li><strong>Dias:</strong> ${registro.diasFerias} dias</li>
</ul>
<p>Para mais informações, entre em contato com o setor de Recursos Humanos.</p>`,
enviadoPor: args.usuarioId
});
}
// Criar ou obter conversa entre RH e funcionário
const conversasExistentes = await ctx.db
.query('conversas')
.filter((q) => q.eq(q.field('tipo'), 'individual'))
.collect();
let conversaId: Id<'conversas'> | null = null;
for (const conversa of conversasExistentes) {
if (
conversa.participantes.length === 2 &&
conversa.participantes.includes(args.usuarioId) &&
conversa.participantes.includes(funcionarioUsuario._id)
) {
conversaId = conversa._id;
break;
}
}
if (!conversaId) {
conversaId = await ctx.db.insert('conversas', {
tipo: 'individual',
participantes: [args.usuarioId, funcionarioUsuario._id],
criadoPor: args.usuarioId,
criadoEm: Date.now()
});
}
// Criar mensagem de chat (texto simples)
await ctx.db.insert('mensagens', {
conversaId,
remetenteId: args.usuarioId,
tipo: 'texto',
conteudo: `Sua solicitação de férias de ${formatarDataBR(registro.dataInicio)} até ${formatarDataBR(registro.dataFim)} (${registro.diasFerias} dias) foi cancelada pelo setor de Recursos Humanos. Para mais informações, entre em contato conosco.`,
enviadaEm: Date.now()
});
}
}
}
return null;
}
});