Feat: Implementa Reporte de Distribución de Suscripciones y Refactoriza Gestión de Ajustes
Se introduce una nueva funcionalidad de reporte crucial para la logística y se realiza una refactorización mayor del sistema de ajustes para garantizar la correcta imputación contable. ### ✨ Nuevas Características - **Nuevo Reporte de Distribución de Suscripciones (RR011):** - Se añade un nuevo reporte en PDF que genera un listado de todas las suscripciones activas en un rango de fechas. - El reporte está diseñado para el equipo de reparto, incluyendo datos clave como nombre del suscriptor, dirección, teléfono, días de entrega y observaciones. - Se implementa el endpoint, la lógica de servicio, la consulta a la base de datos y el template de QuestPDF en el backend. - Se crea la página correspondiente en el frontend (React) con su selector de fechas y se añade la ruta y el enlace en el menú de reportes. ### 🔄 Refactorización Mayor - **Asociación de Ajustes a Empresas:** - Se refactoriza por completo la entidad `Ajuste` para incluir una referencia obligatoria a una `IdEmpresa`. - **Motivo:** Corregir un error crítico en la lógica de negocio donde los ajustes de un suscriptor se aplicaban a la primera factura generada, sin importar a qué empresa correspondía el ajuste. - Se provee un script de migración SQL para actualizar el esquema de la base de datos (`susc_Ajustes`). - Se actualizan todos los DTOs, repositorios y servicios (backend) para manejar la nueva relación. - Se modifica el `FacturacionService` para que ahora aplique los ajustes pendientes correspondientes a cada empresa dentro de su bucle de facturación. - Se actualiza el formulario de creación/edición de ajustes en el frontend (React) para incluir un selector de empresa obligatorio. ### ⚡️ Optimizaciones de Rendimiento - **Solución de N+1 Queries:** - Se optimiza el método `ObtenerTodos` en `SuscriptorService` para obtener todas las formas de pago en una única consulta en lugar de una por cada suscriptor. - Se optimiza el método `ObtenerAjustesPorSuscriptor` en `AjusteService` para obtener todos los nombres de usuarios y empresas en dos consultas masivas, en lugar de N consultas individuales. - Se añade el método `GetByIdsAsync` al `IUsuarioRepository` y su implementación para soportar esta optimización. ### 🐛 Corrección de Errores - Se corrigen múltiples errores en el script de migración de base de datos (uso de `GO` dentro de transacciones, error de "columna inválida"). - Se soluciona un error de SQL (`INSERT` statement) en `AjusteRepository` que impedía la creación de nuevos ajustes. - Se corrige un bug en el `AjusteService` que causaba que el nombre de la empresa apareciera como "N/A" en la UI debido a un mapeo incompleto en el método optimizado. - Se corrige la lógica de generación de emails en `FacturacionService` para mostrar correctamente el nombre de la empresa en cada sección del resumen de cuenta.
This commit is contained in:
		| @@ -105,10 +105,7 @@ namespace GestionIntegral.Api.Services.Suscripciones | ||||
|                 foreach (var grupo in gruposParaFacturar) | ||||
|                 { | ||||
|                     int idSuscriptor = grupo.Key.IdSuscriptor; | ||||
|                     int idEmpresa = grupo.Key.IdEmpresa; | ||||
|  | ||||
|                     // La verificación de existencia ahora debe ser más específica, pero por ahora la omitimos | ||||
|                     // para no añadir otro método al repositorio. Asumimos que no se corre dos veces. | ||||
|                     int idEmpresa = grupo.Key.IdEmpresa; // <-- Ya tenemos la empresa del grupo | ||||
|  | ||||
|                     decimal importeBrutoTotal = 0; | ||||
|                     decimal descuentoPromocionesTotal = 0; | ||||
| @@ -136,10 +133,10 @@ namespace GestionIntegral.Api.Services.Suscripciones | ||||
|                         }); | ||||
|                     } | ||||
|  | ||||
|                     // 4. Aplicar ajustes. Se aplican a la PRIMERA factura que se genere para el cliente. | ||||
|                     // 4. Aplicar ajustes. Ahora se buscan por Suscriptor Y por Empresa. | ||||
|                     var ultimoDiaDelMes = periodoActual.AddMonths(1).AddDays(-1); | ||||
|                     var ajustesPendientes = await _ajusteRepository.GetAjustesPendientesHastaFechaAsync(idSuscriptor, ultimoDiaDelMes, transaction); | ||||
|                     decimal totalAjustes = 0; | ||||
|                     var ajustesPendientes = await _ajusteRepository.GetAjustesPendientesHastaFechaAsync(idSuscriptor, idEmpresa, ultimoDiaDelMes, transaction); | ||||
|                     decimal totalAjustes = ajustesPendientes.Sum(a => a.TipoAjuste == "Credito" ? -a.Monto : a.Monto); | ||||
|  | ||||
|                     // Verificamos si este grupo es el "primero" para este cliente para no aplicar ajustes varias veces | ||||
|                     bool esPrimerGrupoParaCliente = !facturasCreadas.Any(f => f.IdSuscriptor == idSuscriptor); | ||||
| @@ -177,7 +174,7 @@ namespace GestionIntegral.Api.Services.Suscripciones | ||||
|                         await _facturaDetalleRepository.CreateAsync(detalle, transaction); | ||||
|                     } | ||||
|  | ||||
|                     if (esPrimerGrupoParaCliente && ajustesPendientes.Any()) | ||||
|                     if (ajustesPendientes.Any()) | ||||
|                     { | ||||
|                         await _ajusteRepository.MarcarAjustesComoAplicadosAsync(ajustesPendientes.Select(a => a.IdAjuste), facturaCreada.IdFactura, transaction); | ||||
|                     } | ||||
| @@ -317,8 +314,9 @@ namespace GestionIntegral.Api.Services.Suscripciones | ||||
|             var periodo = $"{anio}-{mes:D2}"; | ||||
|             try | ||||
|             { | ||||
|                 var facturas = await _facturaRepository.GetListBySuscriptorYPeriodoAsync(idSuscriptor, periodo); | ||||
|                 if (!facturas.Any()) return (false, "No se encontraron facturas para este suscriptor en el período."); | ||||
|                 // 1. Reemplazamos la llamada original por la nueva, que ya trae toda la información necesaria. | ||||
|                 var facturasConEmpresa = await _facturaRepository.GetFacturasConEmpresaAsync(idSuscriptor, periodo); | ||||
|                 if (!facturasConEmpresa.Any()) return (false, "No se encontraron facturas para este suscriptor en el período."); | ||||
|  | ||||
|                 var suscriptor = await _suscriptorRepository.GetByIdAsync(idSuscriptor); | ||||
|                 if (suscriptor == null || string.IsNullOrEmpty(suscriptor.Email)) return (false, "El suscriptor no es válido o no tiene email."); | ||||
| @@ -326,21 +324,36 @@ namespace GestionIntegral.Api.Services.Suscripciones | ||||
|                 var resumenHtml = new StringBuilder(); | ||||
|                 var adjuntos = new List<(byte[] content, string name)>(); | ||||
|  | ||||
|                 foreach (var factura in facturas.Where(f => f.EstadoPago != "Anulada")) | ||||
|                 // 2. Iteramos sobre la nueva lista de tuplas. | ||||
|                 foreach (var item in facturasConEmpresa.Where(f => f.Factura.EstadoPago != "Anulada")) | ||||
|                 { | ||||
|                     var factura = item.Factura; | ||||
|                     var nombreEmpresa = item.NombreEmpresa; | ||||
|  | ||||
|                     // 3. Eliminamos la lógica compleja y propensa a errores para obtener la empresa. | ||||
|                     //    La llamada a GetDetallesPorFacturaIdAsync sigue siendo necesaria para el cuerpo del email. | ||||
|                     var detalles = await _facturaDetalleRepository.GetDetallesPorFacturaIdAsync(factura.IdFactura); | ||||
|                     if (!detalles.Any()) continue; | ||||
|  | ||||
|                     var primeraSuscripcionId = detalles.First().IdSuscripcion; | ||||
|                     var publicacion = await _publicacionRepository.GetByIdSimpleAsync(primeraSuscripcionId); | ||||
|                     var empresa = await _empresaRepository.GetByIdAsync(publicacion?.IdEmpresa ?? 0); | ||||
|  | ||||
|                     resumenHtml.Append($"<h4 style='margin-top: 20px; margin-bottom: 10px; color: #34515e;'>Resumen de Suscripción</h4>"); | ||||
|                     // Título mejorado para claridad | ||||
|                     resumenHtml.Append($"<h4 style='margin-top: 20px; margin-bottom: 10px; color: #34515e;'>Resumen para {nombreEmpresa}</h4>"); | ||||
|                     resumenHtml.Append("<table style='width: 100%; border-collapse: collapse; font-size: 0.9em;'>"); | ||||
|  | ||||
|                     // 1. Mostrar Detalles de Suscripciones | ||||
|                     foreach (var detalle in detalles) | ||||
|                     { | ||||
|                         resumenHtml.Append($"<tr><td style='padding: 5px; border-bottom: 1px solid #eee;'>{detalle.Descripcion}</td><td style='padding: 5px; border-bottom: 1px solid #eee; text-align: right;'>${detalle.ImporteNeto:N2}</td></tr>"); | ||||
|                     } | ||||
|                     var ajustes = await _ajusteRepository.GetAjustesPorIdFacturaAsync(factura.IdFactura); | ||||
|                     if (ajustes.Any()) | ||||
|                     { | ||||
|                         foreach (var ajuste in ajustes) | ||||
|                         { | ||||
|                             bool esCredito = ajuste.TipoAjuste == "Credito"; | ||||
|                             string colorMonto = esCredito ? "#d9534f" : "#5cb85c"; | ||||
|                             string signo = esCredito ? "-" : "+"; | ||||
|                             resumenHtml.Append($"<tr><td style='padding: 5px; border-bottom: 1px solid #eee; font-style: italic;'>Ajuste: {ajuste.Motivo}</td><td style='padding: 5px; border-bottom: 1px solid #eee; text-align: right; color: {colorMonto}; font-style: italic;'>{signo} ${ajuste.Monto:N2}</td></tr>"); | ||||
|                         } | ||||
|                     } | ||||
|                     resumenHtml.Append($"<tr style='font-weight: bold;'><td style='padding: 5px;'>Subtotal</td><td style='padding: 5px; text-align: right;'>${factura.ImporteFinal:N2}</td></tr>"); | ||||
|                     resumenHtml.Append("</table>"); | ||||
|  | ||||
| @@ -350,7 +363,8 @@ namespace GestionIntegral.Api.Services.Suscripciones | ||||
|                         if (File.Exists(rutaCompleta)) | ||||
|                         { | ||||
|                             byte[] pdfBytes = await File.ReadAllBytesAsync(rutaCompleta); | ||||
|                             string pdfFileName = $"Factura_{empresa?.Nombre?.Replace(" ", "")}_{factura.NumeroFactura}.pdf"; | ||||
|                             // Usamos el nombre de la empresa para un nombre de archivo más claro | ||||
|                             string pdfFileName = $"Factura_{nombreEmpresa.Replace(" ", "")}_{factura.NumeroFactura}.pdf"; | ||||
|                             adjuntos.Add((pdfBytes, pdfFileName)); | ||||
|                             _logger.LogInformation("PDF adjuntado: {FileName}", pdfFileName); | ||||
|                         } | ||||
| @@ -361,7 +375,7 @@ namespace GestionIntegral.Api.Services.Suscripciones | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 var totalGeneral = facturas.Where(f => f.EstadoPago != "Anulada").Sum(f => f.ImporteFinal); | ||||
|                 var totalGeneral = facturasConEmpresa.Where(f => f.Factura.EstadoPago != "Anulada").Sum(f => f.Factura.ImporteFinal); | ||||
|                 string asunto = $"Resumen de Cuenta - Diario El Día - Período {periodo}"; | ||||
|                 string cuerpoHtml = ConstruirCuerpoEmailConsolidado(suscriptor, periodo, resumenHtml.ToString(), totalGeneral); | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user