feat: Implementación de Secciones, Recargos, Porc. Pago Dist. y backend E/S Dist.

Backend API:
- Recargos por Zona (`dist_RecargoZona`):
  - CRUD completo (Modelos, DTOs, Repositorio, Servicio, Controlador).
  - Endpoints anidados bajo `/publicaciones/{idPublicacion}/recargos`.
  - Lógica de negocio para vigencias (cierre/reapertura de períodos).
  - Auditoría en `dist_RecargoZona_H`.
- Porcentajes de Pago Distribuidores (`dist_PorcPago`):
  - CRUD completo (Modelos, DTOs, Repositorio, Servicio, Controlador).
  - Endpoints anidados bajo `/publicaciones/{idPublicacion}/porcentajespago`.
  - Lógica de negocio para vigencias.
  - Auditoría en `dist_PorcPago_H`.
- Porcentajes/Montos Pago Canillitas (`dist_PorcMonPagoCanilla`):
  - CRUD completo (Modelos, DTOs, Repositorio, Servicio, Controlador).
  - Endpoints anidados bajo `/publicaciones/{idPublicacion}/porcentajesmoncanilla`.
  - Lógica de negocio para vigencias.
  - Auditoría en `dist_PorcMonPagoCanilla_H`.
- Secciones de Publicación (`dist_dtPubliSecciones`):
  - CRUD completo (Modelos, DTOs, Repositorio, Servicio, Controlador).
  - Endpoints anidados bajo `/publicaciones/{idPublicacion}/secciones`.
  - Auditoría en `dist_dtPubliSecciones_H`.
- Entradas/Salidas Distribuidores (`dist_EntradasSalidas`):
  - Implementado backend (Modelos, DTOs, Repositorio, Servicio, Controlador).
  - Lógica para determinar precios/recargos/porcentajes aplicables.
  - Cálculo de monto y afectación de saldos de distribuidores en `cue_Saldos`.
  - Auditoría en `dist_EntradasSalidas_H`.
- Correcciones de Mapeo Dapper:
  - Aplicados alias explícitos en repositorios de RecargoZona, PorcPago, PorcMonCanilla, PubliSeccion,
    Canilla, Distribuidor y Precio para asegurar mapeo correcto de IDs y columnas.

Frontend React:
- Recargos por Zona:
  - `recargoZonaService.ts`.
  - `RecargoZonaFormModal.tsx` para crear/editar períodos de recargos.
  - `GestionarRecargosPublicacionPage.tsx` para listar y gestionar recargos por publicación.
- Porcentajes de Pago Distribuidores:
  - `porcPagoService.ts`.
  - `PorcPagoFormModal.tsx`.
  - `GestionarPorcentajesPagoPage.tsx`.
- Porcentajes/Montos Pago Canillitas:
  - `porcMonCanillaService.ts`.
  - `PorcMonCanillaFormModal.tsx`.
  - `GestionarPorcMonCanillaPage.tsx`.
- Secciones de Publicación:
  - `publiSeccionService.ts`.
  - `PubliSeccionFormModal.tsx`.
  - `GestionarSeccionesPublicacionPage.tsx`.
- Navegación:
  - Actualizadas rutas y menús para acceder a la gestión de recargos, porcentajes (dist. y canillita) y secciones desde la vista de una publicación.
- Layout:
  - Uso consistente de `Box` con Flexbox en lugar de `Grid` en nuevos modales y páginas para evitar errores de tipo.
This commit is contained in:
2025-05-21 14:58:52 -03:00
parent b6ba52f074
commit e7e185a9cb
140 changed files with 10465 additions and 394 deletions

View File

@@ -0,0 +1,25 @@
using System;
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class CreateEntradaSalidaDistDto
{
[Required]
public int IdPublicacion { get; set; }
[Required]
public int IdDistribuidor { get; set; }
[Required]
public DateTime Fecha { get; set; }
[Required]
[RegularExpression("^(Salida|Entrada)$", ErrorMessage = "Tipo de movimiento debe ser 'Salida' o 'Entrada'.")]
public string TipoMovimiento { get; set; } = string.Empty;
[Required, Range(1, int.MaxValue)]
public int Cantidad { get; set; }
[Required, Range(1, int.MaxValue)] // Asumiendo que el remito es un número > 0
public int Remito { get; set; }
[StringLength(150)]
public string? Observacion { get; set; }
// IdPrecio, IdRecargo, IdPorcentaje se determinarán en el backend.
}
}

View File

@@ -0,0 +1,25 @@
using System;
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class CreatePorcMonCanillaDto
{
[Required]
public int IdPublicacion { get; set; }
[Required(ErrorMessage = "El canillita es obligatorio.")]
[Range(1, int.MaxValue, ErrorMessage = "Debe seleccionar un canillita válido.")]
public int IdCanilla { get; set; }
[Required(ErrorMessage = "La fecha de Vigencia Desde es obligatoria.")]
public DateTime VigenciaD { get; set; }
[Required(ErrorMessage = "El valor (porcentaje o monto) es obligatorio.")]
[Range(0, double.MaxValue, ErrorMessage = "El valor debe ser un monto positivo.")] // double.MaxValue para permitir porcentajes > 100 si fuera necesario, ajustar
public decimal PorcMon { get; set; }
[Required]
public bool EsPorcentaje { get; set; }
}
}

View File

@@ -0,0 +1,22 @@
using System;
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class CreatePorcPagoDto
{
[Required]
public int IdPublicacion { get; set; }
[Required(ErrorMessage = "El distribuidor es obligatorio.")]
[Range(1, int.MaxValue, ErrorMessage = "Debe seleccionar un distribuidor válido.")]
public int IdDistribuidor { get; set; }
[Required(ErrorMessage = "La fecha de Vigencia Desde es obligatoria.")]
public DateTime VigenciaD { get; set; }
[Required(ErrorMessage = "El porcentaje es obligatorio.")]
[Range(0, 100, ErrorMessage = "El porcentaje debe estar entre 0 y 100.")] // Asumiendo que es un porcentaje
public decimal Porcentaje { get; set; }
}
}

View File

@@ -0,0 +1,16 @@
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class CreatePubliSeccionDto
{
[Required]
public int IdPublicacion { get; set; }
[Required(ErrorMessage = "El nombre de la sección es obligatorio.")]
[StringLength(100)]
public string Nombre { get; set; } = string.Empty;
public bool Estado { get; set; } = true; // Por defecto activa al crear
}
}

View File

@@ -0,0 +1,22 @@
using System;
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class CreateRecargoZonaDto
{
[Required(ErrorMessage = "El ID de la publicación es obligatorio.")]
public int IdPublicacion { get; set; }
[Required(ErrorMessage = "La zona es obligatoria.")]
[Range(1, int.MaxValue, ErrorMessage = "Debe seleccionar una zona válida.")]
public int IdZona { get; set; }
[Required(ErrorMessage = "La fecha de Vigencia Desde es obligatoria.")]
public DateTime VigenciaD { get; set; }
[Required(ErrorMessage = "El valor del recargo es obligatorio.")]
[Range(0, 999999.99, ErrorMessage = "El valor del recargo debe ser un monto positivo.")]
public decimal Valor { get; set; }
}
}

View File

@@ -0,0 +1,19 @@
using System;
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class CreateSalidaOtroDestinoDto
{
[Required]
public int IdPublicacion { get; set; }
[Required]
public int IdDestino { get; set; }
[Required]
public DateTime Fecha { get; set; }
[Required, Range(1, int.MaxValue)]
public int Cantidad { get; set; }
[StringLength(150)]
public string? Observacion { get; set; }
}
}

View File

@@ -0,0 +1,20 @@
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class EntradaSalidaDistDto
{
public int IdParte { get; set; }
public int IdPublicacion { get; set; }
public string NombrePublicacion { get; set; } = string.Empty;
public string NombreEmpresaPublicacion { get; set; } = string.Empty; // Para identificar la empresa dueña del saldo
public int IdEmpresaPublicacion { get; set; } // Para afectar el saldo correcto
public int IdDistribuidor { get; set; }
public string NombreDistribuidor { get; set; } = string.Empty;
public string Fecha { get; set; } = string.Empty; // yyyy-MM-dd
public string TipoMovimiento { get; set; } = string.Empty; // "Salida" o "Entrada"
public int Cantidad { get; set; }
public int Remito { get; set; }
public string? Observacion { get; set; }
public decimal MontoCalculado { get; set; } // Calculado por el backend
// No exponemos Ids de Precio, Recargo, Porcentaje directamente, solo el resultado.
}
}

View File

@@ -0,0 +1,14 @@
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class PorcMonCanillaDto // Para Porcentaje/Monto de Pago de Canillitas
{
public int IdPorcMon { get; set; }
public int IdPublicacion { get; set; }
public int IdCanilla { get; set; }
public string NomApeCanilla { get; set; } = string.Empty; // Para mostrar en UI
public string VigenciaD { get; set; } = string.Empty; // yyyy-MM-dd
public string? VigenciaH { get; set; } // yyyy-MM-dd
public decimal PorcMon { get; set; } // Valor (puede ser % o monto)
public bool EsPorcentaje { get; set; } // True si PorcMon es un %, False si es un monto fijo
}
}

View File

@@ -0,0 +1,14 @@
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class PorcPagoDto // Para Porcentaje de Pago de Distribuidores
{
public int IdPorcentaje { get; set; }
public int IdPublicacion { get; set; }
// public string NombrePublicacion { get; set; } = string.Empty; // Opcional, si se necesita en una vista general de porcentajes
public int IdDistribuidor { get; set; }
public string NombreDistribuidor { get; set; } = string.Empty; // Para mostrar en UI
public string VigenciaD { get; set; } = string.Empty; // yyyy-MM-dd
public string? VigenciaH { get; set; } // yyyy-MM-dd
public decimal Porcentaje { get; set; }
}
}

View File

@@ -0,0 +1,11 @@
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class PubliSeccionDto
{
public int IdSeccion { get; set; }
public int IdPublicacion { get; set; }
// public string NombrePublicacion { get; set; } = string.Empty; // Opcional si se muestra en una lista global
public string Nombre { get; set; } = string.Empty; // Nombre de la sección
public bool Estado { get; set; } // Habilitada o no
}
}

View File

@@ -0,0 +1,13 @@
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class RecargoZonaDto
{
public int IdRecargo { get; set; }
public int IdPublicacion { get; set; }
public int IdZona { get; set; }
public string NombreZona { get; set; } = string.Empty; // Para mostrar en UI
public string VigenciaD { get; set; } = string.Empty; // yyyy-MM-dd
public string? VigenciaH { get; set; } // yyyy-MM-dd
public decimal Valor { get; set; }
}
}

View File

@@ -0,0 +1,14 @@
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class SalidaOtroDestinoDto
{
public int IdParte { get; set; }
public int IdPublicacion { get; set; }
public string NombrePublicacion { get; set; } = string.Empty;
public int IdDestino { get; set; }
public string NombreDestino { get; set; } = string.Empty;
public string Fecha { get; set; } = string.Empty; // yyyy-MM-dd
public int Cantidad { get; set; }
public string? Observacion { get; set; }
}
}

View File

@@ -0,0 +1,15 @@
// La edición de una entrada/salida puede ser compleja por la afectación de saldos.
// Por ahora, podríamos permitir editar Cantidad y Observacion si el TipoMovimiento no cambia.
// Cambiar Publicacion, Distribuidor, Fecha o TipoMovimiento podría requerir anular y recrear.
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class UpdateEntradaSalidaDistDto
{
[Required, Range(1, int.MaxValue)]
public int Cantidad { get; set; }
[StringLength(150)]
public string? Observacion { get; set; }
// No se permite cambiar TipoMovimiento, Fecha, Publicacion, Distribuidor, Remito aquí.
}
}

View File

@@ -0,0 +1,20 @@
using System;
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class UpdatePorcMonCanillaDto
{
// IdPublicacion, IdCanilla y VigenciaD no deberían cambiar.
// Solo se actualiza PorcMon, EsPorcentaje y opcionalmente se cierra con VigenciaH.
[Required(ErrorMessage = "El valor (porcentaje o monto) es obligatorio.")]
[Range(0, double.MaxValue, ErrorMessage = "El valor debe ser un monto positivo.")]
public decimal PorcMon { get; set; }
[Required]
public bool EsPorcentaje { get; set; }
public DateTime? VigenciaH { get; set; } // Para cerrar el período
}
}

View File

@@ -0,0 +1,18 @@
using System;
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class UpdatePorcPagoDto
{
// IdPublicacion, IdDistribuidor y VigenciaD no deberían cambiar.
// Si cambian, se considera un nuevo registro.
// Solo se actualiza el Porcentaje y opcionalmente se cierra con VigenciaH.
[Required(ErrorMessage = "El porcentaje es obligatorio.")]
[Range(0, 100, ErrorMessage = "El porcentaje debe estar entre 0 y 100.")]
public decimal Porcentaje { get; set; }
public DateTime? VigenciaH { get; set; } // Para cerrar el período
}
}

View File

@@ -0,0 +1,14 @@
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class UpdatePubliSeccionDto
{
[Required(ErrorMessage = "El nombre de la sección es obligatorio.")]
[StringLength(100)]
public string Nombre { get; set; } = string.Empty;
[Required]
public bool Estado { get; set; }
}
}

View File

@@ -0,0 +1,17 @@
using System;
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class UpdateRecargoZonaDto // Para actualizar un IdRecargo específico
{
// IdPublicacion, IdZona y VigenciaD no deberían cambiar. Si cambian, es un nuevo registro.
// Solo se actualiza Valor y opcionalmente se cierra con VigenciaH.
[Required(ErrorMessage = "El valor del recargo es obligatorio.")]
[Range(0, 999999.99, ErrorMessage = "El valor del recargo debe ser un monto positivo.")]
public decimal Valor { get; set; }
public DateTime? VigenciaH { get; set; } // Para cerrar el período
}
}

View File

@@ -0,0 +1,14 @@
using System;
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Distribucion
{
public class UpdateSalidaOtroDestinoDto
{
// IdPublicacion, IdDestino y Fecha usualmente no se cambian en una "salida" ya registrada.
// Se podría permitir cambiar cantidad y observación.
[Required, Range(1, int.MaxValue)]
public int Cantidad { get; set; }
[StringLength(150)]
public string? Observacion { get; set; }
}
}

View File

@@ -0,0 +1,19 @@
using System;
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Impresion
{
public class CambiarEstadoBobinaDto
{
[Required]
public int NuevoEstadoId { get; set; } // ID del nuevo estado (ej. 2 para "En Uso", 3 para "Dañada")
public int? IdPublicacion { get; set; } // Requerido si NuevoEstadoId es "En Uso"
public int? IdSeccion { get; set; } // Requerido si NuevoEstadoId es "En Uso"
[StringLength(250)]
public string? Obs { get; set; }
[Required]
public DateTime FechaCambioEstado { get; set; }
}
}

View File

@@ -0,0 +1,22 @@
using System;
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Impresion
{
public class CreateStockBobinaDto
{
[Required]
public int IdTipoBobina { get; set; }
[Required, StringLength(15)]
public string NroBobina { get; set; } = string.Empty;
[Required, Range(1, int.MaxValue)]
public int Peso { get; set; }
[Required]
public int IdPlanta { get; set; }
[Required, StringLength(15)]
public string Remito { get; set; } = string.Empty;
[Required]
public DateTime FechaRemito { get; set; }
// IdEstadoBobina se setea a "Disponible" (1) en el backend
// FechaEstado se setea a FechaRemito en el backend
}
}

View File

@@ -0,0 +1,21 @@
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Impresion
{
public class CreateTiradaRequestDto
{
[Required]
public int IdPublicacion { get; set; }
[Required]
public DateTime Fecha { get; set; }
[Required]
public int IdPlanta { get; set; }
[Required, Range(1, int.MaxValue)]
public int Ejemplares { get; set; } // Para bob_RegTiradas
[Required]
[MinLength(1, ErrorMessage = "Debe especificar al menos una sección para la tirada.")]
public List<DetalleSeccionTiradaDto> Secciones { get; set; } = new List<DetalleSeccionTiradaDto>();
}
}

View File

@@ -0,0 +1,11 @@
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Impresion
{
public class DetalleSeccionTiradaDto
{
public int IdSeccion { get; set; } // ID de la PubliSeccion
// public string NombreSeccion { get; set; } // Opcional, para mostrar en UI de creación/edición
[Required, Range(1, 1000)] // Ajustar rango según necesidad
public int CantPag { get; set; }
}
}

View File

@@ -0,0 +1,23 @@
namespace GestionIntegral.Api.Dtos.Impresion
{
public class StockBobinaDto
{
public int IdBobina { get; set; }
public int IdTipoBobina { get; set; }
public string NombreTipoBobina { get; set; } = string.Empty;
public string NroBobina { get; set; } = string.Empty;
public int Peso { get; set; }
public int IdPlanta { get; set; }
public string NombrePlanta { get; set; } = string.Empty;
public int IdEstadoBobina { get; set; }
public string NombreEstadoBobina { get; set; } = string.Empty;
public string Remito { get; set; } = string.Empty;
public string FechaRemito { get; set; } = string.Empty; // yyyy-MM-dd
public string? FechaEstado { get; set; } // yyyy-MM-dd
public int? IdPublicacion { get; set; }
public string? NombrePublicacion { get; set; }
public int? IdSeccion { get; set; }
public string? NombreSeccion { get; set; }
public string? Obs { get; set; }
}
}

View File

@@ -0,0 +1,23 @@
// Este DTO es más complejo para mostrar la información de forma útil.
namespace GestionIntegral.Api.Dtos.Impresion
{
public class DetalleSeccionEnListadoDto
{
public int IdSeccion { get; set; }
public string NombreSeccion { get; set; } = string.Empty;
public int CantPag { get; set; }
public int IdRegPublicacionSeccion {get;set;} // Id_Tirada de bob_RegPublicaciones
}
public class TiradaDto
{
public int IdRegistroTirada { get; set; } // Id de bob_RegTiradas
public int IdPublicacion { get; set; }
public string NombrePublicacion { get; set; } = string.Empty;
public string Fecha { get; set; } = string.Empty; // yyyy-MM-dd
public int IdPlanta { get; set; }
public string NombrePlanta { get; set; } = string.Empty;
public int Ejemplares { get; set; }
public List<DetalleSeccionEnListadoDto> SeccionesImpresas { get; set; } = new List<DetalleSeccionEnListadoDto>();
public int TotalPaginasSumadas { get; set; } // Suma de CantPag de las secciones impresas
}
}

View File

@@ -0,0 +1,23 @@
// (Para editar datos básicos de una bobina *Disponible*)
// Los cambios de estado tendrán DTOs dedicados.
using System.ComponentModel.DataAnnotations;
namespace GestionIntegral.Api.Dtos.Impresion
{
public class UpdateStockBobinaDto
{
[Required]
public int IdTipoBobina { get; set; }
[Required, StringLength(15)]
public string NroBobina { get; set; } = string.Empty;
[Required, Range(1, int.MaxValue)]
public int Peso { get; set; }
[Required]
public int IdPlanta { get; set; }
[Required, StringLength(15)]
public string Remito { get; set; } = string.Empty;
[Required]
public DateTime FechaRemito { get; set; }
// No se puede cambiar el estado desde aquí directamente
}
}