- Nueva estructura de tabs: Resumen, Dimensiones, Agentic Readiness, Roadmap - Componentes de visualización McKinsey: - BulletChart: actual vs benchmark con rangos de color - WaterfallChart: impacto económico con costes y ahorros - OpportunityTreemap: priorización por volumen y readiness - 5 dimensiones actualizadas (sin satisfaction ni economy) - Header sticky con navegación animada - Integración completa con datos existentes Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
91 lines
2.8 KiB
TypeScript
91 lines
2.8 KiB
TypeScript
import { motion } from 'framer-motion';
|
|
import { LayoutDashboard, Layers, Bot, Map } from 'lucide-react';
|
|
|
|
export type TabId = 'executive' | 'dimensions' | 'readiness' | 'roadmap';
|
|
|
|
export interface TabConfig {
|
|
id: TabId;
|
|
label: string;
|
|
icon: React.ElementType;
|
|
}
|
|
|
|
interface DashboardHeaderProps {
|
|
title?: string;
|
|
activeTab: TabId;
|
|
onTabChange: (id: TabId) => void;
|
|
}
|
|
|
|
const TABS: TabConfig[] = [
|
|
{ id: 'executive', label: 'Resumen', icon: LayoutDashboard },
|
|
{ id: 'dimensions', label: 'Dimensiones', icon: Layers },
|
|
{ id: 'readiness', label: 'Agentic Readiness', icon: Bot },
|
|
{ id: 'roadmap', label: 'Roadmap', icon: Map },
|
|
];
|
|
|
|
const formatDate = (): string => {
|
|
const now = new Date();
|
|
const months = [
|
|
'Enero', 'Febrero', 'Marzo', 'Abril', 'Mayo', 'Junio',
|
|
'Julio', 'Agosto', 'Septiembre', 'Octubre', 'Noviembre', 'Diciembre'
|
|
];
|
|
return `${months[now.getMonth()]} ${now.getFullYear()}`;
|
|
};
|
|
|
|
export function DashboardHeader({
|
|
title = 'AIR EUROPA - Beyond CX Analytics',
|
|
activeTab,
|
|
onTabChange
|
|
}: DashboardHeaderProps) {
|
|
return (
|
|
<header className="sticky top-0 z-50 bg-white border-b border-slate-200 shadow-sm">
|
|
{/* Top row: Title and Date */}
|
|
<div className="max-w-7xl mx-auto px-6 py-4">
|
|
<div className="flex items-center justify-between">
|
|
<h1 className="text-xl font-bold text-slate-800">{title}</h1>
|
|
<span className="text-sm text-slate-500">{formatDate()}</span>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Tab Navigation */}
|
|
<nav className="max-w-7xl mx-auto px-6">
|
|
<div className="flex space-x-1">
|
|
{TABS.map((tab) => {
|
|
const Icon = tab.icon;
|
|
const isActive = activeTab === tab.id;
|
|
|
|
return (
|
|
<button
|
|
key={tab.id}
|
|
onClick={() => onTabChange(tab.id)}
|
|
className={`
|
|
relative flex items-center gap-2 px-4 py-3 text-sm font-medium
|
|
transition-colors duration-200
|
|
${isActive
|
|
? 'text-[#6D84E3]'
|
|
: 'text-slate-500 hover:text-slate-700'
|
|
}
|
|
`}
|
|
>
|
|
<Icon className="w-4 h-4" />
|
|
<span className="hidden sm:inline">{tab.label}</span>
|
|
|
|
{/* Active indicator */}
|
|
{isActive && (
|
|
<motion.div
|
|
layoutId="activeTab"
|
|
className="absolute bottom-0 left-0 right-0 h-0.5 bg-[#6D84E3]"
|
|
initial={false}
|
|
transition={{ type: 'spring', stiffness: 500, damping: 30 }}
|
|
/>
|
|
)}
|
|
</button>
|
|
);
|
|
})}
|
|
</div>
|
|
</nav>
|
|
</header>
|
|
);
|
|
}
|
|
|
|
export default DashboardHeader;
|