Files
finanzas/components/alerts/AlertPanel.tsx
renato97 712b06f118 feat: initial commit - finanzas app
Complete personal finance management application with:
- Dashboard with financial metrics and alerts
- Credit card management and payments
- Fixed and variable debt tracking
- Monthly budget planning
- Intelligent alert system
- Responsive design with Tailwind CSS

Tech stack: Next.js 14, TypeScript, Zustand, Recharts

🤖 Generated with [Claude Code](https://claude.com/claude-code)
2026-01-29 00:00:32 +00:00

149 lines
5.2 KiB
TypeScript

'use client'
import { useState } from 'react'
import { Bell, CheckCheck, Trash2, Inbox } from 'lucide-react'
import { cn } from '@/lib/utils'
import { useFinanzasStore } from '@/lib/store'
import { AlertItem } from './AlertItem'
import { AlertBadge } from './AlertBadge'
type TabType = 'all' | 'unread'
export function AlertPanel() {
const [activeTab, setActiveTab] = useState<TabType>('all')
const alerts = useFinanzasStore((state) => state.alerts)
const markAlertAsRead = useFinanzasStore((state) => state.markAlertAsRead)
const deleteAlert = useFinanzasStore((state) => state.deleteAlert)
const clearAllAlerts = useFinanzasStore((state) => state.clearAllAlerts)
const unreadAlerts = alerts.filter((alert) => !alert.isRead)
const unreadCount = unreadAlerts.length
const displayedAlerts = activeTab === 'unread' ? unreadAlerts : alerts
const handleMarkAllRead = () => {
unreadAlerts.forEach((alert) => {
markAlertAsRead(alert.id)
})
}
const handleClearAll = () => {
clearAllAlerts()
}
return (
<div className="w-full max-w-md bg-gray-900 rounded-xl border border-gray-800 shadow-xl">
{/* Header */}
<div className="flex items-center justify-between p-4 border-b border-gray-800">
<div className="flex items-center gap-2">
<div className="relative">
<Bell className="h-5 w-5 text-gray-400" />
{unreadCount > 0 && <AlertBadge count={unreadCount} variant="dot" />}
</div>
<h3 className="font-semibold text-white">Alertas</h3>
{unreadCount > 0 && (
<span className="text-xs text-gray-500">({unreadCount})</span>
)}
</div>
<div className="flex items-center gap-1">
{unreadCount > 0 && (
<button
onClick={handleMarkAllRead}
className={cn(
'p-2 rounded-md transition-colors',
'text-gray-500 hover:text-green-400 hover:bg-green-400/10',
'focus:outline-none focus:ring-2 focus:ring-green-400/20'
)}
title="Marcar todas como leídas"
aria-label="Marcar todas como leídas"
>
<CheckCheck className="h-4 w-4" />
</button>
)}
{alerts.length > 0 && (
<button
onClick={handleClearAll}
className={cn(
'p-2 rounded-md transition-colors',
'text-gray-500 hover:text-red-400 hover:bg-red-400/10',
'focus:outline-none focus:ring-2 focus:ring-red-400/20'
)}
title="Limpiar todas"
aria-label="Limpiar todas las alertas"
>
<Trash2 className="h-4 w-4" />
</button>
)}
</div>
</div>
{/* Tabs */}
{alerts.length > 0 && (
<div className="flex border-b border-gray-800">
<button
onClick={() => setActiveTab('all')}
className={cn(
'flex-1 px-4 py-2 text-sm font-medium transition-colors',
'focus:outline-none focus:ring-2 focus:ring-inset focus:ring-blue-500/20',
activeTab === 'all'
? 'text-white border-b-2 border-blue-500'
: 'text-gray-500 hover:text-gray-300'
)}
>
Todas
<span className="ml-1.5 text-xs text-gray-600">({alerts.length})</span>
</button>
<button
onClick={() => setActiveTab('unread')}
className={cn(
'flex-1 px-4 py-2 text-sm font-medium transition-colors',
'focus:outline-none focus:ring-2 focus:ring-inset focus:ring-blue-500/20',
activeTab === 'unread'
? 'text-white border-b-2 border-blue-500'
: 'text-gray-500 hover:text-gray-300'
)}
>
No leídas
{unreadCount > 0 && (
<span className="ml-1.5 text-xs text-blue-400">({unreadCount})</span>
)}
</button>
</div>
)}
{/* Alert List */}
<div className="max-h-[400px] overflow-y-auto">
{displayedAlerts.length === 0 ? (
<div className="flex flex-col items-center justify-center py-12 px-4 text-center">
<div className="h-12 w-12 rounded-full bg-gray-800 flex items-center justify-center mb-3">
<Inbox className="h-6 w-6 text-gray-600" />
</div>
<p className="text-gray-400 text-sm">
{activeTab === 'unread'
? 'No tienes alertas sin leer'
: 'No tienes alertas'}
</p>
<p className="text-gray-600 text-xs mt-1">
Las alertas aparecerán cuando haya pagos próximos o eventos importantes
</p>
</div>
) : (
<div className="divide-y divide-gray-800/50" role="list">
{displayedAlerts.map((alert) => (
<AlertItem
key={alert.id}
alert={alert}
onMarkRead={() => markAlertAsRead(alert.id)}
onDelete={() => deleteAlert(alert.id)}
/>
))}
</div>
)}
</div>
</div>
)
}