feat: enhance analytics and performance tracking with real data metrics

- Integrate real page view data from the database for accurate analytics.
- Implement cache-busting for fresh data retrieval in analytics dashboard.
- Calculate and display bounce rate, average session duration, and unique users.
- Refactor performance metrics to ensure only real data is considered.
- Improve user experience with toast notifications for success and error messages.
- Update project editor with undo/redo functionality and enhanced content management.
This commit is contained in:
2026-01-10 03:08:25 +01:00
parent b051d9d2ef
commit 40d9489395
12 changed files with 1156 additions and 445 deletions

View File

@@ -34,8 +34,8 @@ const ToastItem = ({ toast, onRemove }: ToastProps) => {
useEffect(() => {
if (toast.duration !== 0) {
const timer = setTimeout(() => {
setTimeout(() => onRemove(toast.id), 300);
}, toast.duration || 5000);
setTimeout(() => onRemove(toast.id), 200);
}, toast.duration || 3000);
return () => clearTimeout(timer);
}
@@ -59,39 +59,39 @@ const ToastItem = ({ toast, onRemove }: ToastProps) => {
const getColors = () => {
switch (toast.type) {
case 'success':
return 'bg-white border-green-300 text-green-900 shadow-lg';
return 'bg-stone-50 border-green-200 text-green-800 shadow-md';
case 'error':
return 'bg-white border-red-300 text-red-900 shadow-lg';
return 'bg-stone-50 border-red-200 text-red-800 shadow-md';
case 'warning':
return 'bg-white border-yellow-300 text-yellow-900 shadow-lg';
return 'bg-stone-50 border-yellow-200 text-yellow-800 shadow-md';
case 'info':
return 'bg-white border-blue-300 text-blue-900 shadow-lg';
return 'bg-stone-50 border-stone-200 text-stone-800 shadow-md';
default:
return 'bg-white border-gray-300 text-gray-900 shadow-lg';
return 'bg-stone-50 border-stone-200 text-stone-800 shadow-md';
}
};
return (
<motion.div
initial={{ opacity: 0, y: -50, scale: 0.9 }}
initial={{ opacity: 0, y: -20, scale: 0.95 }}
animate={{ opacity: 1, y: 0, scale: 1 }}
exit={{ opacity: 0, y: -50, scale: 0.9 }}
transition={{ duration: 0.3, ease: "easeOut" }}
className={`relative p-4 rounded-xl border ${getColors()} shadow-xl hover:shadow-2xl transition-all duration-300 max-w-sm`}
exit={{ opacity: 0, y: -10, scale: 0.95 }}
transition={{ duration: 0.2, ease: "easeOut" }}
className={`relative p-3 rounded-lg border ${getColors()} shadow-lg hover:shadow-xl transition-all duration-200 max-w-xs text-sm`}
>
<div className="flex items-start space-x-3">
<div className="flex items-start space-x-2">
<div className="flex-shrink-0 mt-0.5">
{getIcon()}
</div>
<div className="flex-1 min-w-0">
<h4 className="text-sm font-semibold mb-1">{toast.title}</h4>
<p className="text-sm opacity-90">{toast.message}</p>
<h4 className="text-xs font-semibold mb-0.5 leading-tight">{toast.title}</h4>
<p className="text-xs opacity-90 leading-tight">{toast.message}</p>
{toast.action && (
<button
onClick={toast.action.onClick}
className="mt-2 text-xs font-medium underline hover:no-underline transition-all"
className="mt-1.5 text-xs font-medium underline hover:no-underline transition-all"
>
{toast.action.label}
</button>
@@ -100,9 +100,9 @@ const ToastItem = ({ toast, onRemove }: ToastProps) => {
<button
onClick={() => onRemove(toast.id)}
className="flex-shrink-0 p-1 rounded-lg hover:bg-gray-100 transition-colors"
className="flex-shrink-0 p-0.5 rounded hover:bg-gray-100/50 transition-colors"
>
<X className="w-4 h-4 text-gray-500" />
<X className="w-3 h-3 text-gray-500" />
</button>
</div>
@@ -111,8 +111,8 @@ const ToastItem = ({ toast, onRemove }: ToastProps) => {
<motion.div
initial={{ width: '100%' }}
animate={{ width: '0%' }}
transition={{ duration: (toast.duration || 5000) / 1000, ease: "linear" }}
className="absolute bottom-0 left-0 h-1 bg-gradient-to-r from-stone-400 to-stone-600 rounded-b-xl"
transition={{ duration: (toast.duration || 3000) / 1000, ease: "linear" }}
className="absolute bottom-0 left-0 h-0.5 bg-gradient-to-r from-stone-400 to-stone-600 rounded-b-lg"
/>
)}
</motion.div>
@@ -195,7 +195,7 @@ export const ToastProvider = ({ children }: { children: React.ReactNode }) => {
type: 'error',
title,
message: message || '',
duration: 6000
duration: 4000 // Shorter duration
});
}, [addToast]);
@@ -291,7 +291,7 @@ export const ToastProvider = ({ children }: { children: React.ReactNode }) => {
{children}
{/* Toast Container */}
<div className="fixed top-4 right-4 z-50 space-y-3 max-w-sm">
<div className="fixed top-4 right-4 z-50 space-y-2 max-w-xs">
<AnimatePresence>
{toasts.map((toast) => (
<ToastItem