Add page UI imporvements

This commit is contained in:
Urtzi Alfaro
2025-12-18 22:23:16 +01:00
parent 033cdf84f0
commit a6ae730ef0
8 changed files with 636 additions and 542 deletions

View File

@@ -108,23 +108,23 @@ const ProcurementAnalyticsPage: React.FC = () => {
showMobileNotice={true}
>
{activeTab === 'overview' && (
<>
<div className="space-y-6">
{/* Overview Tab */}
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
{/* Plan Status Distribution */}
<AnalyticsCard title="Distribución de Estados de Planes">
<div className="space-y-3">
{dashboard?.plan_status_distribution?.map((status: any) => (
<div key={status.status} className="flex items-center justify-between">
<span className="text-[var(--text-secondary)]">{status.status}</span>
<div className="flex items-center gap-2">
<div key={status.status} className="flex items-center justify-between p-3 bg-[var(--bg-secondary)] rounded-lg hover:bg-[var(--bg-tertiary)] transition-colors">
<span className="text-sm font-medium text-[var(--text-primary)] capitalize">{status.status.replace('_', ' ')}</span>
<div className="flex items-center gap-3">
<div className="w-32 h-2 bg-[var(--bg-tertiary)] rounded-full overflow-hidden">
<div
className="h-full bg-[var(--color-primary)]"
className="h-full bg-[var(--color-primary)] rounded-full transition-all"
style={{ width: `${(status.count / (dashboard?.summary?.total_plans || 1)) * 100}%` }}
/>
</div>
<span className="text-sm font-medium text-[var(--text-primary)] w-8 text-right">
<span className="text-sm font-bold text-[var(--text-primary)] w-10 text-right">
{status.count}
</span>
</div>
@@ -139,20 +139,29 @@ const ProcurementAnalyticsPage: React.FC = () => {
actions={<AlertCircle className="h-5 w-5 text-[var(--color-error)]" />}
>
<div className="space-y-3">
<div className="flex justify-between items-center">
<span className="text-[var(--text-secondary)]">Stock Crítico</span>
<div className="flex justify-between items-center p-4 bg-red-50 dark:bg-red-900/20 rounded-lg">
<div className="flex items-center gap-2">
<div className="w-2 h-2 rounded-full bg-[var(--color-error)]"></div>
<span className="text-sm font-medium text-[var(--text-secondary)]">Stock Crítico</span>
</div>
<span className="text-2xl font-bold text-[var(--color-error)]">
{dashboard?.critical_requirements?.low_stock || 0}
</span>
</div>
<div className="flex justify-between items-center">
<span className="text-[var(--text-secondary)]">Entregas Atrasadas</span>
<div className="flex justify-between items-center p-4 bg-orange-50 dark:bg-orange-900/20 rounded-lg">
<div className="flex items-center gap-2">
<div className="w-2 h-2 rounded-full bg-[var(--color-warning)]"></div>
<span className="text-sm font-medium text-[var(--text-secondary)]">Entregas Atrasadas</span>
</div>
<span className="text-2xl font-bold text-[var(--color-warning)]">
{dashboard?.critical_requirements?.overdue || 0}
</span>
</div>
<div className="flex justify-between items-center">
<span className="text-[var(--text-secondary)]">Alta Prioridad</span>
<div className="flex justify-between items-center p-4 bg-blue-50 dark:bg-blue-900/20 rounded-lg">
<div className="flex items-center gap-2">
<div className="w-2 h-2 rounded-full bg-[var(--color-info)]"></div>
<span className="text-sm font-medium text-[var(--text-secondary)]">Alta Prioridad</span>
</div>
<span className="text-2xl font-bold text-[var(--color-info)]">
{dashboard?.critical_requirements?.high_priority || 0}
</span>
@@ -163,33 +172,33 @@ const ProcurementAnalyticsPage: React.FC = () => {
{/* Recent Plans */}
<AnalyticsCard title="Planes Recientes">
<div className="overflow-x-auto">
<div className="overflow-x-auto -mx-6">
<table className="w-full">
<thead>
<tr className="border-b border-[var(--border-primary)]">
<th className="text-left py-3 px-4 text-[var(--text-secondary)] font-medium">Plan</th>
<th className="text-left py-3 px-4 text-[var(--text-secondary)] font-medium">Fecha</th>
<th className="text-left py-3 px-4 text-[var(--text-secondary)] font-medium">Estado</th>
<th className="text-right py-3 px-4 text-[var(--text-secondary)] font-medium">Requerimientos</th>
<th className="text-right py-3 px-4 text-[var(--text-secondary)] font-medium">Costo Total</th>
<tr className="border-b border-[var(--border-primary)] bg-[var(--bg-secondary)]">
<th className="text-left py-3 px-6 text-xs font-semibold text-[var(--text-secondary)] uppercase tracking-wider">Plan</th>
<th className="text-left py-3 px-6 text-xs font-semibold text-[var(--text-secondary)] uppercase tracking-wider">Fecha</th>
<th className="text-left py-3 px-6 text-xs font-semibold text-[var(--text-secondary)] uppercase tracking-wider">Estado</th>
<th className="text-right py-3 px-6 text-xs font-semibold text-[var(--text-secondary)] uppercase tracking-wider">Requerimientos</th>
<th className="text-right py-3 px-6 text-xs font-semibold text-[var(--text-secondary)] uppercase tracking-wider">Costo Total</th>
</tr>
</thead>
<tbody>
{dashboard?.recent_plans?.map((plan: any) => (
<tr key={plan.id} className="border-b border-[var(--border-primary)] hover:bg-[var(--bg-secondary)]">
<td className="py-3 px-4 text-[var(--text-primary)]">{plan.plan_number}</td>
<td className="py-3 px-4 text-[var(--text-secondary)]">
<tr key={plan.id} className="border-b border-[var(--border-primary)] hover:bg-[var(--bg-secondary)] transition-colors">
<td className="py-4 px-6 text-sm font-medium text-[var(--text-primary)]">{plan.plan_number}</td>
<td className="py-4 px-6 text-sm text-[var(--text-secondary)]">
{new Date(plan.plan_date).toLocaleDateString()}
</td>
<td className="py-3 px-4">
<span className={`px-2 py-1 rounded-full text-xs ${getStatusColor(plan.status)}`}>
<td className="py-4 px-6">
<span className={`px-3 py-1 rounded-full text-xs font-medium ${getStatusColor(plan.status)}`}>
{plan.status}
</span>
</td>
<td className="py-3 px-4 text-right text-[var(--text-primary)]">
<td className="py-4 px-6 text-sm text-right font-medium text-[var(--text-primary)]">
{plan.total_requirements}
</td>
<td className="py-3 px-4 text-right text-[var(--text-primary)]">
<td className="py-4 px-6 text-sm text-right font-bold text-[var(--text-primary)]">
{formatters.currency(plan.total_estimated_cost)}
</td>
</tr>
@@ -198,40 +207,46 @@ const ProcurementAnalyticsPage: React.FC = () => {
</table>
</div>
</AnalyticsCard>
</>
</div>
)}
{activeTab === 'performance' && (
<>
<div className="space-y-6">
{/* Performance Tab */}
<div className="grid grid-cols-1 md:grid-cols-3 gap-6">
<AnalyticsCard>
<div className="text-center">
<Target className="mx-auto h-8 w-8 text-[var(--color-success)] mb-3" />
<div className="text-3xl font-bold text-[var(--text-primary)] mb-1">
<div className="text-center p-4">
<div className="inline-flex items-center justify-center w-16 h-16 rounded-full bg-green-100 dark:bg-green-900/30 mb-4">
<Target className="h-8 w-8 text-[var(--color-success)]" />
</div>
<div className="text-4xl font-bold text-[var(--text-primary)] mb-2">
{formatters.percentage(dashboard?.performance_metrics?.average_fulfillment_rate || 0)}
</div>
<div className="text-sm text-[var(--text-secondary)]">Tasa de Cumplimiento</div>
<div className="text-sm font-medium text-[var(--text-secondary)]">Tasa de Cumplimiento</div>
</div>
</AnalyticsCard>
<AnalyticsCard>
<div className="text-center">
<Calendar className="mx-auto h-8 w-8 text-[var(--color-info)] mb-3" />
<div className="text-3xl font-bold text-[var(--text-primary)] mb-1">
<div className="text-center p-4">
<div className="inline-flex items-center justify-center w-16 h-16 rounded-full bg-blue-100 dark:bg-blue-900/30 mb-4">
<Calendar className="h-8 w-8 text-[var(--color-info)]" />
</div>
<div className="text-4xl font-bold text-[var(--text-primary)] mb-2">
{formatters.percentage(dashboard?.performance_metrics?.average_on_time_delivery || 0)}
</div>
<div className="text-sm text-[var(--text-secondary)]">Entregas a Tiempo</div>
<div className="text-sm font-medium text-[var(--text-secondary)]">Entregas a Tiempo</div>
</div>
</AnalyticsCard>
<AnalyticsCard>
<div className="text-center">
<Award className="mx-auto h-8 w-8 text-[var(--color-warning)] mb-3" />
<div className="text-3xl font-bold text-[var(--text-primary)] mb-1">
<div className="text-center p-4">
<div className="inline-flex items-center justify-center w-16 h-16 rounded-full bg-orange-100 dark:bg-orange-900/30 mb-4">
<Award className="h-8 w-8 text-[var(--color-warning)]" />
</div>
<div className="text-4xl font-bold text-[var(--text-primary)] mb-2">
{dashboard?.performance_metrics?.supplier_performance?.toFixed(1) || '0.0'}
</div>
<div className="text-sm text-[var(--text-secondary)]">Puntuación de Calidad</div>
<div className="text-sm font-medium text-[var(--text-secondary)]">Puntuación de Calidad</div>
</div>
</AnalyticsCard>
</div>
@@ -239,84 +254,110 @@ const ProcurementAnalyticsPage: React.FC = () => {
{/* Performance Trend Chart */}
<AnalyticsCard title="Tendencias de Rendimiento (Últimos 7 días)" loading={trendsLoading}>
{trends && trends.performance_trend && trends.performance_trend.length > 0 ? (
<ResponsiveContainer width="100%" height={300}>
<div className="pt-4">
<ResponsiveContainer width="100%" height={350}>
<LineChart data={trends.performance_trend}>
<CartesianGrid strokeDasharray="3 3" stroke="var(--border-primary)" />
<CartesianGrid strokeDasharray="3 3" stroke="var(--border-primary)" opacity={0.3} />
<XAxis
dataKey="date"
stroke="var(--text-tertiary)"
tick={{ fill: 'var(--text-secondary)' }}
tick={{ fill: 'var(--text-secondary)', fontSize: 12 }}
tickLine={{ stroke: 'var(--border-primary)' }}
/>
<YAxis
stroke="var(--text-tertiary)"
tick={{ fill: 'var(--text-secondary)' }}
tick={{ fill: 'var(--text-secondary)', fontSize: 12 }}
tickFormatter={(value) => `${(value * 100).toFixed(0)}%`}
tickLine={{ stroke: 'var(--border-primary)' }}
/>
<Tooltip
contentStyle={{
backgroundColor: 'var(--bg-primary)',
border: '1px solid var(--border-primary)',
borderRadius: '8px'
borderRadius: '8px',
boxShadow: 'var(--shadow-lg)'
}}
formatter={(value: any) => `${(value * 100).toFixed(1)}%`}
labelStyle={{ color: 'var(--text-primary)' }}
labelStyle={{ color: 'var(--text-primary)', fontWeight: 600 }}
/>
<Legend />
<Legend wrapperStyle={{ paddingTop: '20px' }} />
<Line
type="monotone"
dataKey="fulfillment_rate"
stroke="var(--color-success)"
strokeWidth={2}
strokeWidth={3}
name="Tasa de Cumplimiento"
dot={{ fill: 'var(--color-success)' }}
dot={{ fill: 'var(--color-success)', r: 4 }}
activeDot={{ r: 6 }}
/>
<Line
type="monotone"
dataKey="on_time_rate"
stroke="var(--color-info)"
strokeWidth={2}
strokeWidth={3}
name="Entregas a Tiempo"
dot={{ fill: 'var(--color-info)' }}
dot={{ fill: 'var(--color-info)', r: 4 }}
activeDot={{ r: 6 }}
/>
</LineChart>
</ResponsiveContainer>
</div>
) : (
<div className="h-64 flex items-center justify-center text-[var(--text-tertiary)]">
No hay datos de tendencias disponibles
<div className="h-80 flex flex-col items-center justify-center text-[var(--text-tertiary)]">
<BarChart3 className="h-16 w-16 mb-4 opacity-20" />
<p className="text-sm">No hay datos de tendencias disponibles</p>
</div>
)}
</AnalyticsCard>
</>
</div>
)}
{activeTab === 'suppliers' && (
<>
<div className="space-y-6">
{/* Suppliers Tab */}
<AnalyticsCard title="Rendimiento de Proveedores">
<div className="overflow-x-auto">
<div className="overflow-x-auto -mx-6">
<table className="w-full">
<thead>
<tr className="border-b border-[var(--border-primary)]">
<th className="text-left py-3 px-4 text-[var(--text-secondary)] font-medium">Proveedor</th>
<th className="text-right py-3 px-4 text-[var(--text-secondary)] font-medium">Órdenes</th>
<th className="text-right py-3 px-4 text-[var(--text-secondary)] font-medium">Tasa Cumplimiento</th>
<th className="text-right py-3 px-4 text-[var(--text-secondary)] font-medium">Entregas a Tiempo</th>
<th className="text-right py-3 px-4 text-[var(--text-secondary)] font-medium">Calidad</th>
<tr className="border-b border-[var(--border-primary)] bg-[var(--bg-secondary)]">
<th className="text-left py-3 px-6 text-xs font-semibold text-[var(--text-secondary)] uppercase tracking-wider">Proveedor</th>
<th className="text-right py-3 px-6 text-xs font-semibold text-[var(--text-secondary)] uppercase tracking-wider">Órdenes</th>
<th className="text-right py-3 px-6 text-xs font-semibold text-[var(--text-secondary)] uppercase tracking-wider">Cumplimiento</th>
<th className="text-right py-3 px-6 text-xs font-semibold text-[var(--text-secondary)] uppercase tracking-wider">Puntualidad</th>
<th className="text-right py-3 px-6 text-xs font-semibold text-[var(--text-secondary)] uppercase tracking-wider">Calidad</th>
</tr>
</thead>
<tbody>
{dashboard?.supplier_performance?.map((supplier: any) => (
<tr key={supplier.id} className="border-b border-[var(--border-primary)] hover:bg-[var(--bg-secondary)]">
<td className="py-3 px-4 text-[var(--text-primary)]">{supplier.name}</td>
<td className="py-3 px-4 text-right text-[var(--text-primary)]">{supplier.total_orders}</td>
<td className="py-3 px-4 text-right text-[var(--text-primary)]">
<tr key={supplier.id} className="border-b border-[var(--border-primary)] hover:bg-[var(--bg-secondary)] transition-colors">
<td className="py-4 px-6 text-sm font-semibold text-[var(--text-primary)]">{supplier.name}</td>
<td className="py-4 px-6 text-sm text-right font-medium text-[var(--text-primary)]">
<span className="inline-flex items-center justify-center px-3 py-1 rounded-full bg-[var(--bg-tertiary)]">
{supplier.total_orders}
</span>
</td>
<td className="py-4 px-6 text-sm text-right">
<span className={`inline-flex items-center px-3 py-1 rounded-full font-semibold ${
supplier.fulfillment_rate >= 0.9 ? 'bg-green-100 dark:bg-green-900/30 text-green-700 dark:text-green-300' :
supplier.fulfillment_rate >= 0.7 ? 'bg-yellow-100 dark:bg-yellow-900/30 text-yellow-700 dark:text-yellow-300' :
'bg-red-100 dark:bg-red-900/30 text-red-700 dark:text-red-300'
}`}>
{formatters.percentage(supplier.fulfillment_rate)}
</span>
</td>
<td className="py-3 px-4 text-right text-[var(--text-primary)]">
<td className="py-4 px-6 text-sm text-right">
<span className={`inline-flex items-center px-3 py-1 rounded-full font-semibold ${
supplier.on_time_rate >= 0.9 ? 'bg-green-100 dark:bg-green-900/30 text-green-700 dark:text-green-300' :
supplier.on_time_rate >= 0.7 ? 'bg-yellow-100 dark:bg-yellow-900/30 text-yellow-700 dark:text-yellow-300' :
'bg-red-100 dark:bg-red-900/30 text-red-700 dark:text-red-300'
}`}>
{formatters.percentage(supplier.on_time_rate)}
</span>
</td>
<td className="py-3 px-4 text-right text-[var(--text-primary)]">
<td className="py-4 px-6 text-sm text-right">
<span className="font-bold text-[var(--text-primary)]">
{supplier.quality_score?.toFixed(1) || 'N/A'}
</span>
</td>
</tr>
))}
@@ -324,54 +365,72 @@ const ProcurementAnalyticsPage: React.FC = () => {
</table>
</div>
</AnalyticsCard>
</>
</div>
)}
{activeTab === 'costs' && (
<>
<div className="space-y-6">
{/* Costs Tab */}
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
<AnalyticsCard title="Análisis de Costos">
<div className="space-y-4">
<div className="flex justify-between items-center">
<span className="text-[var(--text-secondary)]">Costo Total Estimado</span>
<span className="text-2xl font-bold text-[var(--text-primary)]">
<div className="flex justify-between items-center p-4 bg-[var(--bg-secondary)] rounded-lg">
<div>
<span className="text-xs text-[var(--text-tertiary)] uppercase tracking-wide">Costo Total Estimado</span>
<div className="text-3xl font-bold text-[var(--text-primary)] mt-1">
{formatters.currency(dashboard?.summary?.total_estimated_cost || 0)}
</span>
</div>
<div className="flex justify-between items-center">
<span className="text-[var(--text-secondary)]">Costo Total Aprobado</span>
<span className="text-2xl font-bold text-[var(--text-primary)]">
</div>
<DollarSign className="h-12 w-12 text-[var(--color-info)] opacity-20" />
</div>
<div className="flex justify-between items-center p-4 bg-[var(--bg-secondary)] rounded-lg">
<div>
<span className="text-xs text-[var(--text-tertiary)] uppercase tracking-wide">Costo Total Aprobado</span>
<div className="text-3xl font-bold text-[var(--text-primary)] mt-1">
{formatters.currency(dashboard?.summary?.total_approved_cost || 0)}
</span>
</div>
<div className="flex justify-between items-center">
<span className="text-[var(--text-secondary)]">Variación Promedio</span>
<span className={`text-2xl font-bold ${
</div>
<DollarSign className="h-12 w-12 text-[var(--color-success)] opacity-20" />
</div>
<div className="flex justify-between items-center p-4 bg-[var(--bg-secondary)] rounded-lg">
<div>
<span className="text-xs text-[var(--text-tertiary)] uppercase tracking-wide">Variación Promedio</span>
<div className={`text-3xl font-bold mt-1 ${
(dashboard?.summary?.cost_variance || 0) > 0
? 'text-[var(--color-error)]'
: 'text-[var(--color-success)]'
}`}>
{formatters.currency(Math.abs(dashboard?.summary?.cost_variance || 0))}
</span>
{(dashboard?.summary?.cost_variance || 0) > 0 ? '+' : ''}{formatters.currency(dashboard?.summary?.cost_variance || 0)}
</div>
</div>
<TrendingUp className={`h-12 w-12 opacity-20 ${
(dashboard?.summary?.cost_variance || 0) > 0
? 'text-[var(--color-error)]'
: 'text-[var(--color-success)] rotate-180'
}`} />
</div>
</div>
</AnalyticsCard>
<AnalyticsCard title="Distribución de Costos por Categoría">
<div className="space-y-3">
<div className="space-y-4">
{dashboard?.cost_by_category?.map((category: any) => (
<div key={category.name} className="flex items-center justify-between">
<span className="text-[var(--text-secondary)]">{category.name}</span>
<div className="flex items-center gap-2">
<div className="w-32 h-2 bg-[var(--bg-tertiary)] rounded-full overflow-hidden">
<div key={category.name} className="space-y-2">
<div className="flex items-center justify-between">
<span className="text-sm font-medium text-[var(--text-primary)]">{category.name}</span>
<span className="text-sm font-bold text-[var(--text-primary)]">
{formatters.currency(category.amount)}
</span>
</div>
<div className="relative">
<div className="w-full h-3 bg-[var(--bg-tertiary)] rounded-full overflow-hidden">
<div
className="h-full bg-[var(--color-primary)]"
className="h-full bg-gradient-to-r from-[var(--color-primary)] to-[var(--color-primary-light)] rounded-full transition-all"
style={{ width: `${(category.amount / (dashboard?.summary?.total_estimated_cost || 1)) * 100}%` }}
/>
</div>
<span className="text-sm font-medium text-[var(--text-primary)] w-20 text-right">
{formatters.currency(category.amount)}
<span className="absolute -top-1 right-0 text-xs font-medium text-[var(--text-tertiary)]">
{((category.amount / (dashboard?.summary?.total_estimated_cost || 1)) * 100).toFixed(1)}%
</span>
</div>
</div>
@@ -379,79 +438,93 @@ const ProcurementAnalyticsPage: React.FC = () => {
</div>
</AnalyticsCard>
</div>
</>
</div>
)}
{activeTab === 'quality' && (
<>
<div className="space-y-6">
{/* Quality Tab */}
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
<AnalyticsCard title="Métricas de Calidad">
<div className="space-y-4">
<div className="flex justify-between items-center">
<span className="text-[var(--text-secondary)]">Puntuación Promedio</span>
<span className="text-3xl font-bold text-[var(--text-primary)]">
{dashboard?.quality_metrics?.avg_score?.toFixed(1) || '0.0'} / 10
<div className="flex flex-col items-center p-6 bg-gradient-to-br from-[var(--color-primary)]/10 to-[var(--color-primary-light)]/5 rounded-lg">
<span className="text-sm text-[var(--text-secondary)] mb-2">Puntuación Promedio</span>
<div className="flex items-baseline gap-1">
<span className="text-5xl font-bold text-[var(--text-primary)]">
{dashboard?.quality_metrics?.avg_score?.toFixed(1) || '0.0'}
</span>
<span className="text-2xl font-medium text-[var(--text-secondary)]">/ 10</span>
</div>
<div className="flex justify-between items-center">
<span className="text-[var(--text-secondary)]">Productos con Calidad Alta</span>
</div>
<div className="grid grid-cols-2 gap-4">
<div className="flex flex-col items-center p-4 bg-green-50 dark:bg-green-900/20 rounded-lg">
<Award className="h-8 w-8 text-[var(--color-success)] mb-2" />
<span className="text-2xl font-bold text-[var(--color-success)]">
{dashboard?.quality_metrics?.high_quality_count || 0}
</span>
<span className="text-xs text-[var(--text-secondary)] text-center mt-1">Calidad Alta</span>
</div>
<div className="flex justify-between items-center">
<span className="text-[var(--text-secondary)]">Productos con Calidad Baja</span>
<div className="flex flex-col items-center p-4 bg-red-50 dark:bg-red-900/20 rounded-lg">
<AlertTriangle className="h-8 w-8 text-[var(--color-error)] mb-2" />
<span className="text-2xl font-bold text-[var(--color-error)]">
{dashboard?.quality_metrics?.low_quality_count || 0}
</span>
<span className="text-xs text-[var(--text-secondary)] text-center mt-1">Calidad Baja</span>
</div>
</div>
</div>
</AnalyticsCard>
<AnalyticsCard title="Tendencia de Calidad (Últimos 7 días)" loading={trendsLoading}>
{trends && trends.quality_trend && trends.quality_trend.length > 0 ? (
<ResponsiveContainer width="100%" height={200}>
<div className="pt-4">
<ResponsiveContainer width="100%" height={280}>
<LineChart data={trends.quality_trend}>
<CartesianGrid strokeDasharray="3 3" stroke="var(--border-primary)" />
<CartesianGrid strokeDasharray="3 3" stroke="var(--border-primary)" opacity={0.3} />
<XAxis
dataKey="date"
stroke="var(--text-tertiary)"
tick={{ fill: 'var(--text-secondary)' }}
tick={{ fill: 'var(--text-secondary)', fontSize: 12 }}
tickLine={{ stroke: 'var(--border-primary)' }}
/>
<YAxis
stroke="var(--text-tertiary)"
tick={{ fill: 'var(--text-secondary)' }}
tick={{ fill: 'var(--text-secondary)', fontSize: 12 }}
domain={[0, 10]}
ticks={[0, 2, 4, 6, 8, 10]}
tickLine={{ stroke: 'var(--border-primary)' }}
/>
<Tooltip
contentStyle={{
backgroundColor: 'var(--bg-primary)',
border: '1px solid var(--border-primary)',
borderRadius: '8px'
borderRadius: '8px',
boxShadow: 'var(--shadow-lg)'
}}
formatter={(value: any) => `${value.toFixed(1)} / 10`}
labelStyle={{ color: 'var(--text-primary)' }}
labelStyle={{ color: 'var(--text-primary)', fontWeight: 600 }}
/>
<Line
type="monotone"
dataKey="quality_score"
stroke="var(--color-warning)"
strokeWidth={2}
strokeWidth={3}
name="Puntuación de Calidad"
dot={{ fill: 'var(--color-warning)' }}
dot={{ fill: 'var(--color-warning)', r: 4 }}
activeDot={{ r: 6 }}
/>
</LineChart>
</ResponsiveContainer>
</div>
) : (
<div className="h-48 flex items-center justify-center text-[var(--text-tertiary)]">
No hay datos de calidad disponibles
<div className="h-80 flex flex-col items-center justify-center text-[var(--text-tertiary)]">
<Award className="h-16 w-16 mb-4 opacity-20" />
<p className="text-sm">No hay datos de calidad disponibles</p>
</div>
)}
</AnalyticsCard>
</div>
</>
</div>
)}
</AnalyticsPageLayout>
);

View File

@@ -144,58 +144,68 @@ const ProductionAnalyticsPage: React.FC = () => {
mobileNoticeText={t('mobile.swipe_scroll_interact')}
>
{/* Tab Content */}
<div className="min-h-screen">
<div className="space-y-6">
{/* Overview Tab - Mixed Dashboard */}
{activeTab === 'overview' && (
<div className="grid gap-6">
<div className="grid grid-cols-1 gap-6">
<LiveBatchTrackerWidget />
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
<OnTimeCompletionWidget />
<CapacityUtilizationWidget />
</div>
<QualityScoreTrendsWidget />
<WasteDefectTrackerWidget />
<CapacityUtilizationWidget />
</div>
)}
{/* Bakery Operations Tab */}
{activeTab === 'operations' && (
<div className="grid gap-6">
<div className="grid grid-cols-1 gap-6">
<TodaysScheduleSummaryWidget />
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
<OnTimeCompletionWidget />
<LiveBatchTrackerWidget />
<CapacityUtilizationWidget />
</div>
<LiveBatchTrackerWidget />
</div>
)}
{/* Cost & Efficiency Tab */}
{activeTab === 'cost-efficiency' && (
<div className="grid gap-6">
<div className="grid grid-cols-1 gap-6">
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
<CostPerUnitWidget />
<WasteDefectTrackerWidget />
<YieldPerformanceWidget />
</div>
<WasteDefectTrackerWidget />
</div>
)}
{/* Quality Assurance Tab */}
{activeTab === 'quality' && (
<div className="grid gap-6">
<div className="grid grid-cols-1 gap-6">
<QualityScoreTrendsWidget />
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
<WasteDefectTrackerWidget />
<TopDefectTypesWidget />
</div>
</div>
)}
{/* Equipment & Maintenance Tab */}
{activeTab === 'equipment' && (
<div className="grid gap-6">
<div className="grid grid-cols-1 gap-6">
<EquipmentStatusWidget />
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
<MaintenanceScheduleWidget />
<EquipmentEfficiencyWidget />
</div>
</div>
)}
{/* AI Insights Tab */}
{activeTab === 'ai-insights' && (
<div className="grid gap-6">
<div className="grid grid-cols-1 gap-6">
<AIInsightsWidget />
<PredictiveMaintenanceWidget />
</div>

View File

@@ -218,19 +218,20 @@ const AIInsightsPage: React.FC = () => {
showMobileNotice={true}
>
{/* Category Filter */}
<Card className="p-6">
<div className="flex flex-wrap gap-2">
<Card className="p-6 bg-gradient-to-br from-[var(--bg-primary)] to-[var(--bg-secondary)]">
<h3 className="text-sm font-semibold text-[var(--text-secondary)] uppercase tracking-wider mb-4">Filtrar por Categoría</h3>
<div className="flex flex-wrap gap-3">
{categories.map((category) => (
<button
key={category.value}
onClick={() => setSelectedCategory(category.value)}
className={`px-4 py-2 rounded-full text-sm font-medium transition-colors ${
className={`px-5 py-2.5 rounded-full text-sm font-semibold transition-all transform hover:scale-105 ${
selectedCategory === category.value
? 'bg-blue-600 text-white'
: 'bg-[var(--bg-tertiary)] text-[var(--text-secondary)] hover:bg-[var(--bg-quaternary)]'
? 'bg-gradient-to-r from-[var(--color-primary)] to-[var(--color-primary-light)] text-white shadow-lg'
: 'bg-[var(--bg-tertiary)] text-[var(--text-secondary)] hover:bg-[var(--bg-quaternary)] border border-[var(--border-primary)]'
}`}
>
{category.label} ({category.count})
{category.label} <span className="ml-1 opacity-75">({category.count})</span>
</button>
))}
</div>
@@ -239,26 +240,32 @@ const AIInsightsPage: React.FC = () => {
{/* Insights List */}
<div className="space-y-4">
{filteredInsights.map((insight) => (
<Card key={insight.id} className="p-6">
<Card key={insight.id} className="p-6 hover:shadow-lg transition-shadow border-l-4" style={{borderLeftColor: insight.priority === 'high' ? 'var(--color-error)' : insight.priority === 'medium' ? 'var(--color-warning)' : 'var(--color-success)'}}>
<div className="flex items-start justify-between">
<div className="flex-1">
<div className="flex items-center space-x-3 mb-3">
<div className={`p-2 rounded-lg ${getTypeColor(insight.type)}`}>
<div className="flex items-center space-x-3 mb-4">
<div className={`p-3 rounded-xl ${getTypeColor(insight.type)} shadow-sm`}>
{getTypeIcon(insight.type)}
</div>
<div className="flex items-center space-x-2">
<div className="flex flex-wrap items-center gap-2">
<Badge variant={getPriorityColor(insight.priority)}>
{insight.priority === 'high' ? 'Alta' : insight.priority === 'medium' ? 'Media' : 'Baja'} Prioridad
{insight.priority === 'high' ? '🔴 Alta' : insight.priority === 'medium' ? '🟡 Media' : '🟢 Baja'} Prioridad
</Badge>
<Badge variant="secondary">
<Target className="w-3 h-3 mr-1" />
{insight.confidence}% confianza
</Badge>
<Badge variant="secondary">{insight.confidence}% confianza</Badge>
{insight.actionable && (
<Badge variant="primary">Accionable</Badge>
<Badge variant="primary">
<Zap className="w-3 h-3 mr-1" />
Accionable
</Badge>
)}
</div>
</div>
<h3 className="text-lg font-semibold text-[var(--text-primary)] mb-2">{insight.title}</h3>
<p className="text-[var(--text-secondary)] mb-3">{getInsightDescription(insight)}</p>
<h3 className="text-xl font-bold text-[var(--text-primary)] mb-2">{insight.title}</h3>
<p className="text-[var(--text-secondary)] leading-relaxed mb-4">{getInsightDescription(insight)}</p>
{/* Impact */}
{insight.impact_value && insight.impact_type && (

View File

@@ -119,7 +119,8 @@ const EventRegistryPage: React.FC = () => {
)}
{/* Controls Bar */}
<div className="mb-6 flex flex-wrap items-center justify-between gap-4">
<Card className="p-4 mb-6 bg-gradient-to-r from-[var(--bg-primary)] to-[var(--bg-secondary)]">
<div className="flex flex-wrap items-center justify-between gap-4">
<div className="flex items-center gap-3">
<Button
variant={showFilters ? 'primary' : 'secondary'}
@@ -166,6 +167,7 @@ const EventRegistryPage: React.FC = () => {
</Button>
</div>
</div>
</Card>
{/* Main Content */}
<div className="flex gap-6">
@@ -213,38 +215,38 @@ const EventRegistryPage: React.FC = () => {
) : (
<>
{/* Table */}
<div className="overflow-x-auto">
<div className="overflow-x-auto -mx-6">
<table className="w-full">
<thead className="border-b border-gray-200 bg-gray-50">
<thead className="border-b border-[var(--border-primary)] bg-[var(--bg-secondary)]">
<tr>
<th className="px-6 py-3 text-left text-xs font-medium uppercase tracking-wider text-gray-500">
<th className="px-6 py-3 text-left text-xs font-semibold uppercase tracking-wider text-[var(--text-secondary)]">
Timestamp
</th>
<th className="px-6 py-3 text-left text-xs font-medium uppercase tracking-wider text-gray-500">
<th className="px-6 py-3 text-left text-xs font-semibold uppercase tracking-wider text-[var(--text-secondary)]">
Servicio
</th>
<th className="px-6 py-3 text-left text-xs font-medium uppercase tracking-wider text-gray-500">
<th className="px-6 py-3 text-left text-xs font-semibold uppercase tracking-wider text-[var(--text-secondary)]">
Acción
</th>
<th className="px-6 py-3 text-left text-xs font-medium uppercase tracking-wider text-gray-500">
<th className="px-6 py-3 text-left text-xs font-semibold uppercase tracking-wider text-[var(--text-secondary)]">
Recurso
</th>
<th className="px-6 py-3 text-left text-xs font-medium uppercase tracking-wider text-gray-500">
<th className="px-6 py-3 text-left text-xs font-semibold uppercase tracking-wider text-[var(--text-secondary)]">
Severidad
</th>
<th className="px-6 py-3 text-left text-xs font-medium uppercase tracking-wider text-gray-500">
<th className="px-6 py-3 text-left text-xs font-semibold uppercase tracking-wider text-[var(--text-secondary)]">
Descripción
</th>
<th className="px-6 py-3 text-left text-xs font-medium uppercase tracking-wider text-gray-500">
<th className="px-6 py-3 text-left text-xs font-semibold uppercase tracking-wider text-[var(--text-secondary)]">
Acciones
</th>
</tr>
</thead>
<tbody className="divide-y divide-gray-200 bg-white">
<tbody className="divide-y divide-[var(--border-primary)] bg-[var(--bg-primary)]">
{auditLogs.map((event) => (
<tr
key={event.id}
className="hover:bg-gray-50 transition-colors"
className="hover:bg-[var(--bg-secondary)] transition-colors"
>
<td className="whitespace-nowrap px-6 py-4 text-sm text-gray-900">
<div className="flex flex-col">
@@ -299,31 +301,31 @@ const EventRegistryPage: React.FC = () => {
</div>
{/* Pagination */}
<div className="border-t border-gray-200 bg-gray-50 px-6 py-4">
<div className="border-t border-[var(--border-primary)] bg-[var(--bg-secondary)] px-6 py-4">
<div className="flex items-center justify-between">
<div className="text-sm text-gray-700">
<div className="text-sm text-[var(--text-secondary)]">
Mostrando{' '}
<span className="font-medium">
<span className="font-semibold text-[var(--text-primary)]">
{(currentPage - 1) * pageSize + 1}
</span>{' '}
a{' '}
<span className="font-medium">
<span className="font-semibold text-[var(--text-primary)]">
{Math.min(currentPage * pageSize, auditLogs.length)}
</span>{' '}
de{' '}
<span className="font-medium">{auditLogs.length}</span>{' '}
<span className="font-semibold text-[var(--text-primary)]">{auditLogs.length}</span>{' '}
eventos
</div>
<div className="flex items-center gap-2">
<div className="flex items-center gap-3">
<Button
variant="secondary"
size="sm"
onClick={() => setCurrentPage(p => Math.max(1, p - 1))}
disabled={currentPage === 1}
>
Anterior
Anterior
</Button>
<span className="text-sm text-gray-700">
<span className="text-sm font-medium text-[var(--text-primary)] px-3 py-1 bg-[var(--bg-primary)] rounded-lg">
Página {currentPage} de {totalPages}
</span>
<Button
@@ -332,7 +334,7 @@ const EventRegistryPage: React.FC = () => {
onClick={() => setCurrentPage(p => Math.min(totalPages, p + 1))}
disabled={currentPage === totalPages}
>
Siguiente
Siguiente
</Button>
</div>
</div>

View File

@@ -254,9 +254,9 @@ const ForecastingPage: React.FC = () => {
{/* Ingredient Selection Section */}
<div className="lg:col-span-2 space-y-6">
{/* Ingredients Grid - Similar to POSPage products */}
<Card className="p-4">
<h3 className="text-lg font-semibold mb-4 flex items-center">
<Brain className="w-5 h-5 mr-2" />
<Card className="p-6">
<h3 className="text-lg font-semibold mb-4 flex items-center text-[var(--text-primary)]">
<Brain className="w-5 h-5 mr-2 text-[var(--color-primary)]" />
Ingredientes Disponibles ({products.length})
</h3>
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
@@ -333,8 +333,8 @@ const ForecastingPage: React.FC = () => {
<div className="space-y-6">
{/* Period Selection */}
<Card className="p-6">
<h3 className="text-lg font-semibold mb-4 flex items-center">
<Calendar className="w-5 h-5 mr-2" />
<h3 className="text-lg font-semibold mb-4 flex items-center text-[var(--text-primary)]">
<Calendar className="w-5 h-5 mr-2 text-[var(--color-info)]" />
Configuración
</h3>
<div className="space-y-4">
@@ -345,7 +345,7 @@ const ForecastingPage: React.FC = () => {
<select
value={forecastPeriod}
onChange={(e) => setForecastPeriod(e.target.value)}
className="w-full px-3 py-2 border border-[var(--border-secondary)] rounded-md"
className="w-full px-4 py-3 border border-[var(--border-secondary)] rounded-lg bg-[var(--bg-primary)] text-[var(--text-primary)] focus:ring-2 focus:ring-[var(--color-primary)] focus:border-transparent transition-all"
disabled={isGenerating}
>
{periods.map(period => (
@@ -355,11 +355,14 @@ const ForecastingPage: React.FC = () => {
</div>
{selectedProduct && (
<div className="p-3 bg-[var(--bg-secondary)] rounded-lg">
<p className="text-sm font-medium text-[var(--text-primary)]">
<div className="p-4 bg-gradient-to-br from-[var(--color-primary)]/10 to-[var(--color-primary-light)]/5 rounded-lg border border-[var(--color-primary)]/20">
<div className="flex items-center gap-2 mb-1">
<Package className="w-4 h-4 text-[var(--color-primary)]" />
<p className="text-sm font-semibold text-[var(--text-primary)]">
{products.find(p => p.id === selectedProduct)?.name}
</p>
<p className="text-xs text-[var(--text-secondary)]">
</div>
<p className="text-xs text-[var(--text-secondary)] ml-6">
Predicción para {forecastPeriod} días
</p>
</div>
@@ -368,9 +371,9 @@ const ForecastingPage: React.FC = () => {
</Card>
{/* Generate Forecast */}
<Card className="p-6">
<h3 className="text-lg font-semibold mb-4 flex items-center">
<Zap className="w-5 h-5 mr-2" />
<Card className="p-6 bg-gradient-to-br from-[var(--bg-primary)] to-[var(--bg-secondary)]">
<h3 className="text-lg font-semibold mb-4 flex items-center text-[var(--text-primary)]">
<Zap className="w-5 h-5 mr-2 text-[var(--color-warning)]" />
Generar Predicción
</h3>
<Button
@@ -393,9 +396,12 @@ const ForecastingPage: React.FC = () => {
</Button>
{!selectedProduct && (
<p className="text-xs text-[var(--text-tertiary)] mt-2 text-center">
<div className="mt-3 p-3 bg-blue-50 dark:bg-blue-900/20 rounded-lg">
<p className="text-xs text-[var(--color-info)] text-center flex items-center justify-center gap-2">
<AlertTriangle className="w-4 h-4" />
Selecciona un ingrediente para continuar
</p>
</div>
)}
</Card>
</div>
@@ -403,27 +409,30 @@ const ForecastingPage: React.FC = () => {
{/* Results Section */}
{hasGeneratedForecast && forecasts.length > 0 && (
<>
<div className="space-y-6">
{/* Results Header */}
<Card className="p-6">
<div className="flex items-center justify-between mb-4">
<Card className="p-6 bg-gradient-to-br from-[var(--bg-primary)] to-[var(--bg-secondary)]">
<div className="flex items-center justify-between mb-6">
<div>
<h3 className="text-xl font-bold text-[var(--text-primary)]">Resultados de Predicción</h3>
<p className="text-[var(--text-secondary)]">
<h3 className="text-2xl font-bold text-[var(--text-primary)] flex items-center gap-2">
<TrendingUp className="w-6 h-6 text-[var(--color-success)]" />
Resultados de Predicción
</h3>
<p className="text-sm text-[var(--text-secondary)] mt-1">
{products.find(p => p.id === selectedProduct)?.name} {forecastPeriod} días
</p>
</div>
</div>
{/* Chart View */}
<div className="min-h-96">
<div className="min-h-96 p-4 bg-[var(--bg-primary)] rounded-lg">
<DemandChart
data={forecasts}
product={selectedProduct}
period={forecastPeriod}
loading={isLoading}
error={hasError ? 'Error al cargar las predicciones' : null}
height={400}
height={450}
title=""
/>
</div>
@@ -433,25 +442,25 @@ const ForecastingPage: React.FC = () => {
{currentInsights.length > 0 && (
<Card className="p-6">
<h3 className="text-lg font-semibold text-[var(--text-primary)] mb-4 flex items-center">
<Activity className="w-5 h-5 mr-2" />
<Activity className="w-5 h-5 mr-2 text-[var(--color-info)]" />
Factores de Influencia
</h3>
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
{currentInsights.map((insight, index) => {
const IconComponent = insight.icon;
return (
<div key={index} className="flex items-start space-x-3 p-4 bg-[var(--bg-secondary)] rounded-lg">
<div key={index} className="flex items-start space-x-3 p-4 bg-[var(--bg-secondary)] rounded-lg hover:bg-[var(--bg-tertiary)] transition-all border border-transparent hover:border-[var(--border-primary)]">
<div className={`p-2 rounded-lg ${
insight.impact === 'positive' ? 'bg-[var(--color-success-100)] text-[var(--color-success-600)]' :
insight.impact === 'high' ? 'bg-[var(--color-error-100)] text-[var(--color-error-600)]' :
insight.impact === 'moderate' ? 'bg-[var(--color-warning-100)] text-[var(--color-warning-600)]' :
'bg-[var(--color-info-100)] text-[var(--color-info-600)]'
insight.impact === 'positive' ? 'bg-green-100 dark:bg-green-900/30 text-green-700 dark:text-green-300' :
insight.impact === 'high' ? 'bg-red-100 dark:bg-red-900/30 text-red-700 dark:text-red-300' :
insight.impact === 'moderate' ? 'bg-orange-100 dark:bg-orange-900/30 text-orange-700 dark:text-orange-300' :
'bg-blue-100 dark:bg-blue-900/30 text-blue-700 dark:text-blue-300'
}`}>
<IconComponent className="w-4 h-4" />
<IconComponent className="w-5 h-5" />
</div>
<div className="flex-1">
<p className="text-sm font-medium text-[var(--text-primary)]">{insight.title}</p>
<p className="text-xs text-[var(--text-secondary)]">{insight.description}</p>
<p className="text-sm font-semibold text-[var(--text-primary)]">{insight.title}</p>
<p className="text-xs text-[var(--text-secondary)] mt-1">{insight.description}</p>
</div>
</div>
);
@@ -459,7 +468,7 @@ const ForecastingPage: React.FC = () => {
</div>
</Card>
)}
</>
</div>
)}
{/* Help Section - Only when no models available */}

View File

@@ -226,48 +226,48 @@ const PerformanceAnalyticsPage: React.FC = () => {
>
{/* Vista General Tab */}
{activeTab === 'overview' && !isLoading && (
<>
<div className="space-y-6">
{/* Department Comparison Matrix */}
{departments && departments.length > 0 && (
<AnalyticsCard title="Comparación de Departamentos">
<div className="space-y-4">
{departments.map((dept) => (
<div key={dept.department_id} className="border rounded-lg p-4">
<div className="flex items-center justify-between mb-3">
<h4 className="font-medium text-[var(--text-primary)]">
<div key={dept.department_id} className="border border-[var(--border-primary)] rounded-lg p-5 bg-[var(--bg-secondary)] hover:bg-[var(--bg-tertiary)] transition-all">
<div className="flex items-center justify-between mb-4">
<h4 className="font-semibold text-lg text-[var(--text-primary)]">
{dept.department_name}
</h4>
<div className="flex items-center space-x-2">
<span className="text-lg font-semibold text-[var(--text-primary)]">
<div className="flex items-center space-x-3">
<span className="text-2xl font-bold text-[var(--text-primary)]">
{dept.efficiency.toFixed(1)}%
</span>
{getTrendIcon(dept.trend)}
</div>
</div>
<div className="grid grid-cols-3 gap-4 text-sm">
<div>
<p className="text-[var(--text-tertiary)] text-xs">
<div className="grid grid-cols-1 md:grid-cols-3 gap-4">
<div className="p-3 bg-[var(--bg-primary)] rounded-lg">
<p className="text-[var(--text-tertiary)] text-xs uppercase tracking-wide mb-1">
{dept.metrics.primary_metric.label}
</p>
<p className="font-medium">
<p className="font-bold text-lg text-[var(--text-primary)]">
{dept.metrics.primary_metric.value.toFixed(1)}
{dept.metrics.primary_metric.unit}
</p>
</div>
<div>
<p className="text-[var(--text-tertiary)] text-xs">
<div className="p-3 bg-[var(--bg-primary)] rounded-lg">
<p className="text-[var(--text-tertiary)] text-xs uppercase tracking-wide mb-1">
{dept.metrics.secondary_metric.label}
</p>
<p className="font-medium">
<p className="font-bold text-lg text-[var(--text-primary)]">
{dept.metrics.secondary_metric.value.toFixed(1)}
{dept.metrics.secondary_metric.unit}
</p>
</div>
<div>
<p className="text-[var(--text-tertiary)] text-xs">
<div className="p-3 bg-[var(--bg-primary)] rounded-lg">
<p className="text-[var(--text-tertiary)] text-xs uppercase tracking-wide mb-1">
{dept.metrics.tertiary_metric.label}
</p>
<p className="font-medium">
<p className="font-bold text-lg text-[var(--text-primary)]">
{dept.metrics.tertiary_metric.value.toFixed(1)}
{dept.metrics.tertiary_metric.unit}
</p>
@@ -282,7 +282,8 @@ const PerformanceAnalyticsPage: React.FC = () => {
{/* Process Efficiency Breakdown */}
{processScore && (
<AnalyticsCard title="Desglose de Eficiencia por Procesos">
<ResponsiveContainer width="100%" height={300}>
<div className="pt-4">
<ResponsiveContainer width="100%" height={350}>
<BarChart
data={[
{ name: 'Producción', value: processScore.production_efficiency, weight: processScore.breakdown.production.weight },
@@ -291,18 +292,34 @@ const PerformanceAnalyticsPage: React.FC = () => {
{ name: 'Pedidos', value: processScore.order_efficiency, weight: processScore.breakdown.orders.weight },
]}
>
<CartesianGrid strokeDasharray="3 3" />
<XAxis dataKey="name" />
<YAxis />
<Tooltip />
<Legend />
<Bar dataKey="value" fill="var(--color-primary)" name="Eficiencia (%)" />
<Bar dataKey="weight" fill="var(--color-secondary)" name="Peso (%)" />
<CartesianGrid strokeDasharray="3 3" stroke="var(--border-primary)" opacity={0.3} />
<XAxis
dataKey="name"
tick={{ fill: 'var(--text-secondary)', fontSize: 12 }}
tickLine={{ stroke: 'var(--border-primary)' }}
/>
<YAxis
tick={{ fill: 'var(--text-secondary)', fontSize: 12 }}
tickLine={{ stroke: 'var(--border-primary)' }}
/>
<Tooltip
contentStyle={{
backgroundColor: 'var(--bg-primary)',
border: '1px solid var(--border-primary)',
borderRadius: '8px',
boxShadow: 'var(--shadow-lg)'
}}
labelStyle={{ color: 'var(--text-primary)', fontWeight: 600 }}
/>
<Legend wrapperStyle={{ paddingTop: '20px' }} />
<Bar dataKey="value" fill="var(--color-primary)" name="Eficiencia (%)" radius={[8, 8, 0, 0]} />
<Bar dataKey="weight" fill="var(--color-secondary)" name="Peso (%)" radius={[8, 8, 0, 0]} />
</BarChart>
</ResponsiveContainer>
</div>
</AnalyticsCard>
)}
</>
</div>
)}
{/* Eficiencia Operativa Tab */}

View File

@@ -357,61 +357,32 @@ const BakerySettingsPage: React.FC = () => {
description={t('bakery.description')}
/>
{/* Bakery Header Card */}
<Card className="p-4 sm:p-6">
<div className="flex flex-col sm:flex-row items-start sm:items-center gap-4 sm:gap-6">
<div className="w-16 h-16 bg-gradient-to-br from-[var(--color-primary)] to-[var(--color-primary-dark)] rounded-lg flex items-center justify-center text-white font-bold text-2xl flex-shrink-0 shadow-lg">
{config.name.charAt(0) || 'B'}
</div>
<div className="flex-1 min-w-0">
<h1 className="text-xl sm:text-2xl font-bold text-text-primary mb-1 truncate">
{config.name || t('bakery.information.fields.name')}
</h1>
<p className="text-sm sm:text-base text-text-secondary truncate flex items-center gap-2">
<Mail className="w-4 h-4" />
{config.email}
</p>
<p className="text-xs sm:text-sm text-text-tertiary truncate flex items-center gap-2 mt-1">
<MapPin className="w-4 h-4" />
{config.address}, {config.city}
</p>
</div>
{hasUnsavedChanges && (
<div className="flex items-center gap-2 text-sm text-yellow-600 dark:text-yellow-500 bg-yellow-50 dark:bg-yellow-900/20 px-3 py-2 rounded-lg w-full sm:w-auto">
<AlertCircle className="w-4 h-4" />
<span className="hidden sm:inline">{t('bakery.unsaved_changes')}</span>
</div>
)}
</div>
</Card>
{/* Search Bar */}
<SettingsSearch
placeholder={t('common.search') || 'Search settings...'}
onSearch={setSearchQuery}
className="max-w-md"
/>
{/* Tabs Navigation */}
<Tabs value={activeTab} onValueChange={setActiveTab}>
<TabsList className="w-full sm:w-auto overflow-x-auto">
<TabsTrigger value="information" className="flex-1 sm:flex-none whitespace-nowrap">
<Store className="w-4 h-4 mr-2" />
{t('bakery.tabs.information')}
<Card className="p-2 bg-[var(--bg-secondary)]">
<TabsList className="w-full grid grid-cols-2 sm:grid-cols-4 gap-2">
<TabsTrigger value="information" className="flex items-center justify-center gap-2 data-[state=active]:bg-[var(--color-primary)] data-[state=active]:text-white">
<Store className="w-4 h-4" />
<span className="hidden sm:inline">{t('bakery.tabs.information')}</span>
<span className="sm:hidden">Info</span>
</TabsTrigger>
<TabsTrigger value="hours" className="flex-1 sm:flex-none whitespace-nowrap">
<Clock className="w-4 h-4 mr-2" />
{t('bakery.tabs.hours')}
<TabsTrigger value="hours" className="flex items-center justify-center gap-2 data-[state=active]:bg-[var(--color-primary)] data-[state=active]:text-white">
<Clock className="w-4 h-4" />
<span className="hidden sm:inline">{t('bakery.tabs.hours')}</span>
<span className="sm:hidden">Hours</span>
</TabsTrigger>
<TabsTrigger value="operations" className="flex-1 sm:flex-none whitespace-nowrap">
<SettingsIcon className="w-4 h-4 mr-2" />
{t('bakery.tabs.operations')}
<TabsTrigger value="operations" className="flex items-center justify-center gap-2 data-[state=active]:bg-[var(--color-primary)] data-[state=active]:text-white">
<SettingsIcon className="w-4 h-4" />
<span className="hidden sm:inline">{t('bakery.tabs.operations')}</span>
<span className="sm:hidden">Ops</span>
</TabsTrigger>
<TabsTrigger value="notifications" className="flex-1 sm:flex-none whitespace-nowrap">
<Bell className="w-4 h-4 mr-2" />
{t('bakery.tabs.notifications')}
<TabsTrigger value="notifications" className="flex items-center justify-center gap-2 data-[state=active]:bg-[var(--color-primary)] data-[state=active]:text-white">
<Bell className="w-4 h-4" />
<span className="hidden sm:inline">{t('bakery.tabs.notifications')}</span>
<span className="sm:hidden">Notif</span>
</TabsTrigger>
</TabsList>
</Card>
{/* Tab 1: Information */}
<TabsContent value="information">
@@ -420,10 +391,10 @@ const BakerySettingsPage: React.FC = () => {
<SettingSection
title={t('bakery.information.general_section')}
description="Basic information about your bakery"
icon={<Store className="w-5 h-5" />}
icon={<Store className="w-5 h-5 text-[var(--color-primary)]" />}
>
<div className="p-4 sm:p-6">
<div className="grid grid-cols-1 md:grid-cols-2 gap-4 sm:gap-6">
<div className="p-6 bg-gradient-to-br from-[var(--bg-primary)] to-[var(--bg-secondary)]">
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
<Input
label={t('bakery.information.fields.name')}
value={config.name}
@@ -467,16 +438,16 @@ const BakerySettingsPage: React.FC = () => {
/>
</div>
<div className="mt-4 sm:mt-6">
<label className="block text-sm font-medium text-text-secondary mb-2">
<div className="mt-6 md:col-span-2">
<label className="block text-sm font-semibold text-[var(--text-primary)] mb-3">
{t('bakery.information.fields.description')}
</label>
<textarea
value={config.description}
onChange={handleInputChange('description')}
disabled={isLoading}
rows={3}
className="w-full px-3 py-2 border border-[var(--border-primary)] rounded-lg resize-none bg-[var(--bg-primary)] text-[var(--text-primary)] disabled:bg-[var(--bg-secondary)] disabled:text-[var(--text-secondary)] text-sm sm:text-base focus:ring-2 focus:ring-[var(--color-primary)] focus:border-transparent"
rows={4}
className="w-full px-4 py-3 border-2 border-[var(--border-primary)] rounded-lg resize-none bg-[var(--bg-primary)] text-[var(--text-primary)] disabled:bg-[var(--bg-secondary)] disabled:text-[var(--text-secondary)] focus:ring-2 focus:ring-[var(--color-primary)] focus:border-[var(--color-primary)] transition-all"
placeholder={t('bakery.information.placeholders.description')}
/>
</div>
@@ -487,10 +458,10 @@ const BakerySettingsPage: React.FC = () => {
<SettingSection
title={t('bakery.information.location_section')}
description="Where your bakery is located"
icon={<MapPin className="w-5 h-5" />}
icon={<MapPin className="w-5 h-5 text-[var(--color-success)]" />}
>
<div className="p-4 sm:p-6">
<div className="grid grid-cols-1 md:grid-cols-2 gap-4 sm:gap-6">
<div className="p-6 bg-gradient-to-br from-[var(--bg-primary)] to-[var(--bg-secondary)]">
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
<Input
label={t('bakery.information.fields.address')}
value={config.address}
@@ -536,10 +507,10 @@ const BakerySettingsPage: React.FC = () => {
<SettingSection
title={t('bakery.information.business_section')}
description="Tax and business configuration"
icon={<Building2 className="w-5 h-5" />}
icon={<Building2 className="w-5 h-5 text-[var(--color-info)]" />}
>
<div className="p-4 sm:p-6">
<div className="grid grid-cols-1 md:grid-cols-2 gap-4 sm:gap-6">
<div className="p-6 bg-gradient-to-br from-[var(--bg-primary)] to-[var(--bg-secondary)]">
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
<Input
label={t('bakery.information.fields.tax_id')}
value={config.taxId}
@@ -583,7 +554,7 @@ const BakerySettingsPage: React.FC = () => {
<SettingSection
title={t('bakery.hours.title')}
description="Configure your opening hours for each day of the week"
icon={<Clock className="w-5 h-5" />}
icon={<Clock className="w-5 h-5 text-[var(--color-warning)]" />}
>
{daysOfWeek.map((day) => {
const hours = businessHours[day.key];
@@ -725,33 +696,38 @@ const BakerySettingsPage: React.FC = () => {
{/* Floating Save Button */}
{hasUnsavedChanges && (
<div className="fixed bottom-6 right-4 sm:right-6 left-4 sm:left-auto z-50">
<Card className="p-3 sm:p-4 shadow-xl border-2 border-[var(--color-primary)]">
<div className="flex flex-col sm:flex-row items-stretch sm:items-center gap-3">
<div className="flex items-center gap-2 text-sm text-text-secondary">
<AlertCircle className="w-4 h-4 text-yellow-500 flex-shrink-0" />
<span>{t('bakery.unsaved_changes')}</span>
<div className="fixed bottom-6 right-4 sm:right-6 left-4 sm:left-auto z-50 animate-fade-in">
<Card className="p-4 sm:p-5 shadow-2xl border-2 border-[var(--color-primary)] bg-[var(--bg-primary)]">
<div className="flex flex-col sm:flex-row items-stretch sm:items-center gap-4">
<div className="flex items-center gap-3">
<div className="p-2 bg-yellow-100 dark:bg-yellow-900/30 rounded-lg">
<AlertCircle className="w-5 h-5 text-yellow-600 dark:text-yellow-400 animate-pulse" />
</div>
<div className="flex gap-2">
<div>
<p className="text-sm font-semibold text-[var(--text-primary)]">{t('bakery.unsaved_changes')}</p>
<p className="text-xs text-[var(--text-secondary)]">Save or discard your changes</p>
</div>
</div>
<div className="flex gap-3 sm:ml-4">
<Button
variant="outline"
size="sm"
size="md"
onClick={handleDiscard}
disabled={isLoading}
className="flex-1 sm:flex-none"
className="flex-1 sm:flex-none border-2"
>
<X className="w-4 h-4 mr-1" />
<X className="w-4 h-4 mr-2" />
{t('common.discard')}
</Button>
<Button
variant="primary"
size="sm"
size="md"
onClick={activeTab === 'operations' || activeTab === 'notifications' ? handleSaveOperationalSettings : handleSaveConfig}
isLoading={isLoading}
loadingText={t('common.saving')}
className="flex-1 sm:flex-none"
className="flex-1 sm:flex-none shadow-lg"
>
<Save className="w-4 h-4 mr-1" />
<Save className="w-4 h-4 mr-2" />
{t('common.save')}
</Button>
</div>

View File

@@ -361,83 +361,70 @@ const ProfilePage: React.FC = () => {
];
return (
<div className="p-6 space-y-6">
<div className="p-4 sm:p-6 space-y-6 pb-32">
<PageHeader
title="Ajustes"
title="Ajustes de Perfil"
description="Gestiona tu información personal y preferencias de comunicación"
/>
{/* Tab Navigation */}
<Tabs
items={tabItems}
activeTab={activeTab}
onTabChange={setActiveTab}
fullWidth={true}
variant="pills"
size="md"
/>
{/* Profile Header */}
{activeTab === 'profile' && (
<Card className="p-6">
<div className="flex items-center gap-6">
<div className="relative">
<Avatar
src={profile?.avatar || undefined}
alt={profile?.full_name || `${profileData.first_name} ${profileData.last_name}` || 'Usuario'}
name={profile?.avatar ? (profile?.full_name || `${profileData.first_name} ${profileData.last_name}`) : undefined}
size="xl"
className="w-20 h-20"
/>
<button className="absolute -bottom-1 -right-1 bg-color-primary text-white rounded-full p-2 shadow-lg hover:bg-color-primary-dark transition-colors">
<Camera className="w-4 h-4" />
</button>
</div>
<div className="flex-1">
<h1 className="text-2xl font-bold text-text-primary mb-1">
{profileData.first_name} {profileData.last_name}
</h1>
<p className="text-text-secondary">{profileData.email}</p>
{user?.role && (
<p className="text-sm text-text-tertiary mt-1">
{t(`global_roles.${user.role}`)}
</p>
)}
<div className="flex items-center gap-2 mt-2">
<div className="w-2 h-2 bg-green-500 rounded-full"></div>
<span className="text-sm text-text-tertiary">{t('settings:profile.online', 'En línea')}</span>
</div>
</div>
<div className="flex gap-2">
{!isEditing && (
<Button
variant="outline"
onClick={() => setIsEditing(true)}
className="flex items-center gap-2"
<Card className="p-2 bg-[var(--bg-secondary)]">
<div className="w-full grid grid-cols-1 sm:grid-cols-3 gap-2">
<button
onClick={() => setActiveTab('profile')}
className={`flex items-center justify-center gap-2 px-4 py-3 rounded-lg text-sm font-semibold transition-all ${
activeTab === 'profile'
? 'bg-[var(--color-primary)] text-white shadow-md'
: 'bg-transparent text-[var(--text-secondary)] hover:bg-[var(--bg-tertiary)]'
}`}
>
<User className="w-4 h-4" />
{t('settings:profile.edit_profile', 'Editar Perfil')}
</Button>
)}
<Button
variant="outline"
onClick={() => setShowPasswordForm(!showPasswordForm)}
className="flex items-center gap-2"
<span className="hidden sm:inline">Información Personal</span>
<span className="sm:hidden">Perfil</span>
</button>
<button
onClick={() => setActiveTab('preferences')}
className={`flex items-center justify-center gap-2 px-4 py-3 rounded-lg text-sm font-semibold transition-all ${
activeTab === 'preferences'
? 'bg-[var(--color-primary)] text-white shadow-md'
: 'bg-transparent text-[var(--text-secondary)] hover:bg-[var(--bg-tertiary)]'
}`}
>
<Lock className="w-4 h-4" />
{t('settings:profile.change_password', 'Cambiar Contraseña')}
</Button>
</div>
<Bell className="w-4 h-4" />
<span className="hidden sm:inline">Preferencias de Comunicación</span>
<span className="sm:hidden">Preferencias</span>
</button>
<button
onClick={() => setActiveTab('subscription')}
className={`flex items-center justify-center gap-2 px-4 py-3 rounded-lg text-sm font-semibold transition-all ${
activeTab === 'subscription'
? 'bg-[var(--color-primary)] text-white shadow-md'
: 'bg-transparent text-[var(--text-secondary)] hover:bg-[var(--bg-tertiary)]'
}`}
>
<Crown className="w-4 h-4" />
<span className="hidden sm:inline">Suscripción y Facturación</span>
<span className="sm:hidden">Suscripción</span>
</button>
</div>
</Card>
)}
{/* Profile Form */}
{activeTab === 'profile' && (
<>
{/* Personal Information Card */}
<Card className="p-6">
<h2 className="text-lg font-semibold mb-4">{t('settings:profile.personal_info', 'Información Personal')}</h2>
<div className="flex items-center gap-3 mb-6">
<div className="p-2 bg-[var(--color-primary)]/10 rounded-lg">
<User className="w-5 h-5 text-[var(--color-primary)]" />
</div>
<div>
<h2 className="text-lg font-semibold text-[var(--text-primary)]">{t('settings:profile.personal_info', 'Información Personal')}</h2>
<p className="text-sm text-[var(--text-secondary)]">Actualiza tus datos personales</p>
</div>
</div>
<div className="grid grid-cols-1 md:grid-cols-2 xl:grid-cols-3 gap-6">
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
<Input
label={t('settings:profile.fields.first_name', 'Nombre')}
value={profileData.first_name}
@@ -496,12 +483,12 @@ const ProfilePage: React.FC = () => {
</div>
{isEditing && (
<div className="flex gap-3 mt-6 pt-4 border-t">
<div className="flex gap-3 mt-6 pt-6 border-t border-[var(--border-primary)]">
<Button
variant="outline"
onClick={() => setIsEditing(false)}
disabled={isLoading}
className="flex items-center gap-2"
className="flex items-center gap-2 border-2"
>
<X className="w-4 h-4" />
Cancelar
@@ -511,7 +498,7 @@ const ProfilePage: React.FC = () => {
onClick={handleSaveProfile}
isLoading={isLoading}
loadingText="Guardando..."
className="flex items-center gap-2"
className="flex items-center gap-2 shadow-lg"
>
<Save className="w-4 h-4" />
Guardar Cambios
@@ -519,14 +506,23 @@ const ProfilePage: React.FC = () => {
</div>
)}
</Card>
</>
)}
{/* Password Change Form */}
{activeTab === 'profile' && showPasswordForm && (
<Card className="p-6">
<h2 className="text-lg font-semibold mb-4">Cambiar Contraseña</h2>
<Card className="p-6 bg-gradient-to-br from-[var(--bg-primary)] to-[var(--bg-secondary)]">
<div className="flex items-center gap-3 mb-6">
<div className="p-2 bg-[var(--color-warning)]/10 rounded-lg">
<Lock className="w-5 h-5 text-[var(--color-warning)]" />
</div>
<div>
<h2 className="text-lg font-semibold text-[var(--text-primary)]">Cambiar Contraseña</h2>
<p className="text-sm text-[var(--text-secondary)]">Actualiza tu contraseña de acceso</p>
</div>
</div>
<div className="grid grid-cols-1 md:grid-cols-2 xl:grid-cols-3 gap-6 max-w-4xl">
<div className="grid grid-cols-1 md:grid-cols-3 gap-6">
<Input
type="password"
label="Contraseña Actual"
@@ -558,7 +554,7 @@ const ProfilePage: React.FC = () => {
/>
</div>
<div className="flex gap-3 pt-6 mt-6 border-t">
<div className="flex gap-3 pt-6 mt-6 border-t border-[var(--border-primary)]">
<Button
variant="outline"
onClick={() => {
@@ -567,7 +563,9 @@ const ProfilePage: React.FC = () => {
setErrors({});
}}
disabled={isLoading}
className="border-2"
>
<X className="w-4 h-4 mr-2" />
Cancelar
</Button>
<Button
@@ -575,7 +573,9 @@ const ProfilePage: React.FC = () => {
onClick={handleChangePasswordSubmit}
isLoading={isLoading}
loadingText="Cambiando..."
className="shadow-lg"
>
<Save className="w-4 h-4 mr-2" />
Cambiar Contraseña
</Button>
</div>