feat(reportes): rediseño Estado de Resultados vertical con drill-down, análisis horizontal/vertical y export Excel

- Nuevo endpoint GET /reportes/estado-resultados-detallado con cálculo contable:
  * Ventas, Devoluciones, Ventas netas, Costo de ventas, Utilidad bruta,
    Gastos operativos, Utilidad de la operación
  * Fórmula: subtotal_mxn - descuento_mxn (sin impuestos), nómina usa total_mxn
  * Excluye anticipos (uso_cfdi=P01 o clave_prod_serv=84111506)
  * Filtro por régimen fiscal opcional
  * Año anterior calculado automáticamente

- Nuevo endpoint GET /reportes/estado-resultados/drill-down:
  * Nivel 1: resumen agrupado por RFC
  * Nivel 2: CFDIs individuales filtrados por categoría
  * Categorías: ventas, devoluciones, costo-ventas, gastos-operativos

- Nuevo endpoint GET /reportes/estado-resultados/export:
  * Genera Excel con formato condicional (verde/rojo, negritas)

- Frontend:
  * Tabla vertical con % vertical, año anterior y variación %
  * Filas clickeables para drill-down modal de 2 niveles
  * Top 10 Clientes/Proveedores mantenidos debajo
  * Selector de régimen conectado al reporte

- Fix: NaN en total de drill-down nivel 2 por numeric como string en pg
  * Agregado ::float en queries SQL de CFDIs individuales
This commit is contained in:
Horux Dev
2026-05-15 22:53:10 +00:00
parent 69bf7417a8
commit 7b1f60cbf2
10 changed files with 1160 additions and 66 deletions

View File

@@ -1,5 +1,6 @@
import type { Request, Response, NextFunction } from 'express';
import * as reportesService from '../services/reportes.service.js';
import { exportEstadoResultadosToExcel } from '../services/reportes.service.js';
export async function getEstadoResultados(req: Request, res: Response, next: NextFunction) {
try {
@@ -83,3 +84,73 @@ export async function getConcentradoRfc(req: Request, res: Response, next: NextF
next(error);
}
}
export async function getEstadoResultadosDetallado(req: Request, res: Response, next: NextFunction) {
try {
const { fechaInicio, fechaFin, contribuyenteId, regimen } = req.query;
const now = new Date();
const inicio = (fechaInicio as string) || `${now.getFullYear()}-01-01`;
const fin = (fechaFin as string) || now.toISOString().split('T')[0];
const data = await reportesService.getEstadoResultadosDetallado(
req.tenantPool!,
inicio,
fin,
req.user!.tenantId,
contribuyenteId as string | undefined || null,
regimen as string | undefined || null,
);
res.json(data);
} catch (error) {
console.error('[reportes] Error en getEstadoResultadosDetallado:', error);
next(error);
}
}
export async function getEstadoResultadosDrillDown(req: Request, res: Response, next: NextFunction) {
try {
const { categoria, fechaInicio, fechaFin, contribuyenteId, regimen, rfc } = req.query;
const now = new Date();
const inicio = (fechaInicio as string) || `${now.getFullYear()}-01-01`;
const fin = (fechaFin as string) || now.toISOString().split('T')[0];
const data = await reportesService.getEstadoResultadosDrillDown(
req.tenantPool!,
categoria as string,
inicio,
fin,
contribuyenteId as string | undefined || null,
regimen as string | undefined || null,
rfc as string | undefined || null,
);
res.json(data);
} catch (error) {
console.error('[reportes] Error en getEstadoResultadosDrillDown:', error);
next(error);
}
}
export async function exportEstadoResultados(req: Request, res: Response, next: NextFunction) {
try {
const { fechaInicio, fechaFin, contribuyenteId, regimen } = req.query;
const now = new Date();
const inicio = (fechaInicio as string) || `${now.getFullYear()}-01-01`;
const fin = (fechaFin as string) || now.toISOString().split('T')[0];
const buffer = await exportEstadoResultadosToExcel(
req.tenantPool!,
inicio,
fin,
req.user!.tenantId,
contribuyenteId as string | undefined || null,
regimen as string | undefined || null,
);
res.setHeader('Content-Type', 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet');
res.setHeader('Content-Disposition', `attachment; filename="estado-resultados-${inicio}-${fin}.xlsx"`);
res.send(buffer);
} catch (error) {
console.error('[reportes] Error en exportEstadoResultados:', error);
next(error);
}
}