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:
@@ -0,0 +1,15 @@
|
||||
namespace GestionIntegral.Api.Dtos.Reportes
|
||||
{
|
||||
public class DistribucionSuscripcionDto
|
||||
{
|
||||
public string NombreEmpresa { get; set; } = string.Empty;
|
||||
public string NombrePublicacion { get; set; } = string.Empty;
|
||||
public string NombreSuscriptor { get; set; } = string.Empty;
|
||||
public string Direccion { get; set; } = string.Empty;
|
||||
public string? Telefono { get; set; }
|
||||
public DateTime FechaInicio { get; set; }
|
||||
public DateTime? FechaFin { get; set; }
|
||||
public string DiasEntrega { get; set; } = string.Empty;
|
||||
public string? Observaciones { get; set; }
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,42 @@
|
||||
namespace GestionIntegral.Api.Dtos.Reportes.ViewModels
|
||||
{
|
||||
// Clases internas para la agrupación
|
||||
public class GrupoPublicacion
|
||||
{
|
||||
public string NombrePublicacion { get; set; } = string.Empty;
|
||||
public IEnumerable<DistribucionSuscripcionDto> Suscripciones { get; set; } = Enumerable.Empty<DistribucionSuscripcionDto>();
|
||||
}
|
||||
|
||||
public class GrupoEmpresa
|
||||
{
|
||||
public string NombreEmpresa { get; set; } = string.Empty;
|
||||
public IEnumerable<GrupoPublicacion> Publicaciones { get; set; } = Enumerable.Empty<GrupoPublicacion>();
|
||||
}
|
||||
|
||||
public class DistribucionSuscripcionesViewModel
|
||||
{
|
||||
public IEnumerable<GrupoEmpresa> DatosAgrupados { get; }
|
||||
public string FechaDesde { get; set; } = string.Empty;
|
||||
public string FechaHasta { get; set; } = string.Empty;
|
||||
public string FechaGeneracion { get; set; } = string.Empty;
|
||||
|
||||
public DistribucionSuscripcionesViewModel(IEnumerable<DistribucionSuscripcionDto> suscripciones)
|
||||
{
|
||||
DatosAgrupados = suscripciones
|
||||
.GroupBy(s => s.NombreEmpresa)
|
||||
.Select(gEmpresa => new GrupoEmpresa
|
||||
{
|
||||
NombreEmpresa = gEmpresa.Key,
|
||||
Publicaciones = gEmpresa
|
||||
.GroupBy(s => s.NombrePublicacion)
|
||||
.Select(gPub => new GrupoPublicacion
|
||||
{
|
||||
NombrePublicacion = gPub.Key,
|
||||
Suscripciones = gPub.OrderBy(s => s.NombreSuscriptor).ToList()
|
||||
})
|
||||
.OrderBy(p => p.NombrePublicacion)
|
||||
})
|
||||
.OrderBy(e => e.NombreEmpresa);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -4,13 +4,15 @@ namespace GestionIntegral.Api.Dtos.Suscripciones
|
||||
{
|
||||
public int IdAjuste { get; set; }
|
||||
public int IdSuscriptor { get; set; }
|
||||
public int IdEmpresa { get; set; }
|
||||
public string? NombreEmpresa { get; set; }
|
||||
public string FechaAjuste { get; set; } = string.Empty;
|
||||
public string TipoAjuste { get; set; } = string.Empty;
|
||||
public decimal Monto { get; set; }
|
||||
public string Motivo { get; set; } = string.Empty;
|
||||
public string Estado { get; set; } = string.Empty;
|
||||
public int? IdFacturaAplicado { get; set; }
|
||||
public string FechaAlta { get; set; } = string.Empty; // yyyy-MM-dd
|
||||
public string FechaAlta { get; set; } = string.Empty;
|
||||
public string NombreUsuarioAlta { get; set; } = string.Empty;
|
||||
}
|
||||
}
|
||||
@@ -6,6 +6,9 @@ namespace GestionIntegral.Api.Dtos.Suscripciones
|
||||
{
|
||||
[Required]
|
||||
public int IdSuscriptor { get; set; }
|
||||
|
||||
[Required]
|
||||
public int IdEmpresa { get; set; }
|
||||
|
||||
[Required]
|
||||
public DateTime FechaAjuste { get; set; }
|
||||
|
||||
@@ -1,3 +1,5 @@
|
||||
// Archivo: GestionIntegral.Api/Dtos/Suscripciones/CreateSuscriptorDto.cs
|
||||
|
||||
using System.ComponentModel.DataAnnotations;
|
||||
|
||||
namespace GestionIntegral.Api.Dtos.Suscripciones
|
||||
@@ -13,6 +15,7 @@ namespace GestionIntegral.Api.Dtos.Suscripciones
|
||||
public string? Email { get; set; }
|
||||
|
||||
[StringLength(50)]
|
||||
[RegularExpression(@"^[0-9\s\+\-\(\)]*$", ErrorMessage = "El teléfono solo puede contener números y los símbolos +, -, () y espacios.")]
|
||||
public string? Telefono { get; set; }
|
||||
|
||||
[Required(ErrorMessage = "La dirección es obligatoria.")]
|
||||
@@ -25,9 +28,11 @@ namespace GestionIntegral.Api.Dtos.Suscripciones
|
||||
|
||||
[Required(ErrorMessage = "El número de documento es obligatorio.")]
|
||||
[StringLength(11)]
|
||||
[RegularExpression("^[0-9]*$", ErrorMessage = "El número de documento solo puede contener números.")]
|
||||
public string NroDocumento { get; set; } = string.Empty;
|
||||
|
||||
[StringLength(22, MinimumLength = 22, ErrorMessage = "El CBU debe tener 22 dígitos.")]
|
||||
[RegularExpression("^[0-9]*$", ErrorMessage = "El CBU solo puede contener números.")]
|
||||
public string? CBU { get; set; }
|
||||
|
||||
[Required(ErrorMessage = "La forma de pago es obligatoria.")]
|
||||
|
||||
@@ -3,8 +3,12 @@ using System.ComponentModel.DataAnnotations;
|
||||
namespace GestionIntegral.Api.Dtos.Suscripciones;
|
||||
public class UpdateAjusteDto
|
||||
{
|
||||
[Required]
|
||||
public int IdEmpresa { get; set; }
|
||||
|
||||
[Required]
|
||||
public DateTime FechaAjuste { get; set; }
|
||||
|
||||
[Required]
|
||||
[RegularExpression("^(Credito|Debito)$")]
|
||||
public string TipoAjuste { get; set; } = string.Empty;
|
||||
|
||||
@@ -2,7 +2,6 @@ using System.ComponentModel.DataAnnotations;
|
||||
|
||||
namespace GestionIntegral.Api.Dtos.Suscripciones
|
||||
{
|
||||
// Es idéntico al CreateDto, pero se mantiene separado por si las reglas de validación cambian.
|
||||
public class UpdateSuscriptorDto
|
||||
{
|
||||
[Required(ErrorMessage = "El nombre completo es obligatorio.")]
|
||||
@@ -14,6 +13,7 @@ namespace GestionIntegral.Api.Dtos.Suscripciones
|
||||
public string? Email { get; set; }
|
||||
|
||||
[StringLength(50)]
|
||||
[RegularExpression(@"^[0-9\s\+\-\(\)]*$", ErrorMessage = "El teléfono solo puede contener números y los símbolos +, -, () y espacios.")]
|
||||
public string? Telefono { get; set; }
|
||||
|
||||
[Required(ErrorMessage = "La dirección es obligatoria.")]
|
||||
@@ -26,9 +26,11 @@ namespace GestionIntegral.Api.Dtos.Suscripciones
|
||||
|
||||
[Required(ErrorMessage = "El número de documento es obligatorio.")]
|
||||
[StringLength(11)]
|
||||
[RegularExpression("^[0-9]*$", ErrorMessage = "El número de documento solo puede contener números.")]
|
||||
public string NroDocumento { get; set; } = string.Empty;
|
||||
|
||||
[StringLength(22, MinimumLength = 22, ErrorMessage = "El CBU debe tener 22 dígitos.")]
|
||||
[RegularExpression("^[0-9]*$", ErrorMessage = "El CBU solo puede contener números.")]
|
||||
public string? CBU { get; set; }
|
||||
|
||||
[Required(ErrorMessage = "La forma de pago es obligatoria.")]
|
||||
|
||||
@@ -4,6 +4,7 @@ namespace GestionIntegral.Api.Models.Suscripciones
|
||||
{
|
||||
public int IdAjuste { get; set; }
|
||||
public int IdSuscriptor { get; set; }
|
||||
public int IdEmpresa { get; set; }
|
||||
public DateTime FechaAjuste { get; set; }
|
||||
public string TipoAjuste { get; set; } = string.Empty;
|
||||
public decimal Monto { get; set; }
|
||||
|
||||
Reference in New Issue
Block a user