Consistency fixes: - Extract getThisWeekRange/getThisMonthRange to shared lib/utils/dateRanges.ts (removed 4 identical copy-pasted definitions) - Add error boundaries for behavior, cdn, search, pagespeed pages (4 new error.tsx files — previously fell through to generic parent error) - Add "View setup guide" CTA to empty states on journeys and behavior pages (previously showed text with no actionable button) - Fix non-lazy useState initializer in funnel detail page - Fix Bot & Spam settings header from text-xl to text-2xl (matches all other sections) - Add useMinimumLoading to PageSpeed skeleton (consistent with all other pages) Cleanup: - Remove 438 redundant dark: class prefixes (app is dark-mode only) text-neutral-500 dark:text-neutral-400 → text-neutral-400 (206 occurrences) text-neutral-900 dark:text-white → text-white (232 occurrences) - Remove dead @stripe/react-stripe-js and @stripe/stripe-js packages (billing migrated to Polar, no code imports Stripe) - Remove duplicate motion package (framer-motion is the one actually used)
64 lines
1.9 KiB
TypeScript
64 lines
1.9 KiB
TypeScript
'use client'
|
|
|
|
import { Button } from '@ciphera-net/ui'
|
|
|
|
interface ErrorDisplayProps {
|
|
title?: string
|
|
message?: string
|
|
onRetry?: () => void
|
|
onGoHome?: boolean
|
|
}
|
|
|
|
/**
|
|
* Shared error UI for route-level error.tsx boundaries.
|
|
* Matches the visual style of the 404 page.
|
|
*/
|
|
export default function ErrorDisplay({
|
|
title = 'Something went wrong',
|
|
message = 'An unexpected error occurred. Please try again or go back to the dashboard.',
|
|
onRetry,
|
|
onGoHome = true,
|
|
}: ErrorDisplayProps) {
|
|
return (
|
|
<div className="relative min-h-[80vh] flex flex-col items-center justify-center overflow-hidden">
|
|
<div className="absolute inset-0 -z-10 pointer-events-none">
|
|
<div className="absolute top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 w-[500px] h-[500px] bg-red-500/10 rounded-full blur-[128px] opacity-60" />
|
|
<div
|
|
className="absolute inset-0 bg-grid-pattern opacity-[0.02] dark:opacity-[0.05]"
|
|
style={{ maskImage: 'radial-gradient(ellipse at center, black 0%, transparent 70%)' }}
|
|
/>
|
|
</div>
|
|
|
|
<div className="text-center px-4 z-10">
|
|
<img
|
|
src="/illustrations/server-down.svg"
|
|
alt="Something went wrong"
|
|
className="w-56 h-auto mx-auto mb-8"
|
|
/>
|
|
|
|
<h2 className="text-2xl font-bold text-white mb-4">
|
|
{title}
|
|
</h2>
|
|
<p className="text-lg text-neutral-600 dark:text-neutral-400 max-w-md mx-auto mb-10 leading-relaxed">
|
|
{message}
|
|
</p>
|
|
|
|
<div className="flex flex-col sm:flex-row gap-4 justify-center items-center">
|
|
{onRetry && (
|
|
<Button variant="primary" onClick={onRetry} className="px-8 py-3">
|
|
Try again
|
|
</Button>
|
|
)}
|
|
{onGoHome && (
|
|
<a href="/">
|
|
<Button variant="secondary" className="px-8 py-3">
|
|
Go to dashboard
|
|
</Button>
|
|
</a>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|