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:
		| @@ -35,9 +35,9 @@ interface GetNovedadesCanillasParams { | ||||
| } | ||||
|  | ||||
| interface GetListadoDistMensualParams { | ||||
|     fechaDesde: string; // yyyy-MM-dd | ||||
|     fechaHasta: string; // yyyy-MM-dd | ||||
|     esAccionista: boolean; | ||||
|   fechaDesde: string; // yyyy-MM-dd | ||||
|   fechaHasta: string; // yyyy-MM-dd | ||||
|   esAccionista: boolean; | ||||
| } | ||||
|  | ||||
| const getExistenciaPapelPdf = async (params: GetExistenciaPapelParams): Promise<Blob> => { | ||||
| @@ -420,40 +420,63 @@ const getCanillasGananciasReporte = async (params: GetNovedadesCanillasParams): | ||||
| }; | ||||
|  | ||||
| const getListadoDistMensualDiarios = async (params: GetListadoDistMensualParams): Promise<ListadoDistCanMensualDiariosDto[]> => { | ||||
|     const response = await apiClient.get<ListadoDistCanMensualDiariosDto[]>('/reportes/listado-distribucion-mensual/diarios', { params }); | ||||
|     return response.data; | ||||
|   const response = await apiClient.get<ListadoDistCanMensualDiariosDto[]>('/reportes/listado-distribucion-mensual/diarios', { params }); | ||||
|   return response.data; | ||||
| }; | ||||
|  | ||||
| const getListadoDistMensualDiariosPdf = async (params: GetListadoDistMensualParams): Promise<Blob> => { | ||||
|     const response = await apiClient.get('/reportes/listado-distribucion-mensual/diarios/pdf', { | ||||
|         params, | ||||
|         responseType: 'blob', | ||||
|     }); | ||||
|     return response.data; | ||||
|   const response = await apiClient.get('/reportes/listado-distribucion-mensual/diarios/pdf', { | ||||
|     params, | ||||
|     responseType: 'blob', | ||||
|   }); | ||||
|   return response.data; | ||||
| }; | ||||
|  | ||||
| const getListadoDistMensualPorPublicacion = async (params: GetListadoDistMensualParams): Promise<ListadoDistCanMensualPubDto[]> => { | ||||
|     const response = await apiClient.get<ListadoDistCanMensualPubDto[]>('/reportes/listado-distribucion-mensual/publicaciones', { params }); | ||||
|     return response.data; | ||||
|   const response = await apiClient.get<ListadoDistCanMensualPubDto[]>('/reportes/listado-distribucion-mensual/publicaciones', { params }); | ||||
|   return response.data; | ||||
| }; | ||||
|  | ||||
| const getListadoDistMensualPorPublicacionPdf = async (params: GetListadoDistMensualParams): Promise<Blob> => { | ||||
|     const response = await apiClient.get('/reportes/listado-distribucion-mensual/publicaciones/pdf', { | ||||
|         params, | ||||
|         responseType: 'blob', | ||||
|     }); | ||||
|     return response.data; | ||||
|   const response = await apiClient.get('/reportes/listado-distribucion-mensual/publicaciones/pdf', { | ||||
|     params, | ||||
|     responseType: 'blob', | ||||
|   }); | ||||
|   return response.data; | ||||
| }; | ||||
|  | ||||
| const getReporteFacturasPublicidadPdf = async (anio: number, mes: number): Promise<{ fileContent: Blob, fileName: string }> => { | ||||
|     const params = new URLSearchParams({ anio: String(anio), mes: String(mes) }); | ||||
|         const url = `/reportes/suscripciones/facturas-para-publicidad/pdf?${params.toString()}`; | ||||
|   const params = new URLSearchParams({ anio: String(anio), mes: String(mes) }); | ||||
|   const url = `/reportes/suscripciones/facturas-para-publicidad/pdf?${params.toString()}`; | ||||
|   const response = await apiClient.get(url, { | ||||
|     responseType: 'blob', | ||||
|   }); | ||||
|  | ||||
|   const contentDisposition = response.headers['content-disposition']; | ||||
|   let fileName = `ReportePublicidad_Suscripciones_${anio}-${String(mes).padStart(2, '0')}.pdf`; // Fallback | ||||
|   if (contentDisposition) { | ||||
|     const fileNameMatch = contentDisposition.match(/filename="(.+)"/); | ||||
|     if (fileNameMatch && fileNameMatch.length > 1) { | ||||
|       fileName = fileNameMatch[1]; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   return { fileContent: response.data, fileName: fileName }; | ||||
| }; | ||||
|  | ||||
| const getReporteDistribucionSuscripcionesPdf = async (fechaDesde: string, fechaHasta: string): Promise<{ fileContent: Blob, fileName: string }> => { | ||||
|     const params = new URLSearchParams({  | ||||
|         fechaDesde: fechaDesde,  | ||||
|         fechaHasta: fechaHasta  | ||||
|     }); | ||||
|     const url = `/reportes/suscripciones/distribucion/pdf?${params.toString()}`; | ||||
|      | ||||
|     const response = await apiClient.get(url, { | ||||
|         responseType: 'blob', | ||||
|     }); | ||||
|  | ||||
|     const contentDisposition = response.headers['content-disposition']; | ||||
|     let fileName = `ReportePublicidad_Suscripciones_${anio}-${String(mes).padStart(2, '0')}.pdf`; // Fallback | ||||
|     let fileName = `ReporteDistribucion_Suscripciones_${fechaDesde}_al_${fechaHasta}.pdf`; // Fallback | ||||
|     if (contentDisposition) { | ||||
|         const fileNameMatch = contentDisposition.match(/filename="(.+)"/); | ||||
|         if (fileNameMatch && fileNameMatch.length > 1) { | ||||
| @@ -508,6 +531,7 @@ const reportesService = { | ||||
|   getListadoDistMensualPorPublicacion, | ||||
|   getListadoDistMensualPorPublicacionPdf, | ||||
|   getReporteFacturasPublicidadPdf, | ||||
|   getReporteDistribucionSuscripcionesPdf, | ||||
| }; | ||||
|  | ||||
| export default reportesService; | ||||
		Reference in New Issue
	
	Block a user