36 lines
971 B
TypeScript
36 lines
971 B
TypeScript
|
|
import { useEffect, useRef } from 'react'
|
||
|
|
import { Navigate, useLocation } from 'react-router-dom'
|
||
|
|
import { toast } from 'sonner'
|
||
|
|
import { useAuth } from '../contexts/AuthContext'
|
||
|
|
|
||
|
|
export function AdminRoute({ children }: { children: React.ReactNode }) {
|
||
|
|
const { user, loading, isAdmin } = useAuth()
|
||
|
|
const location = useLocation()
|
||
|
|
const toastShownRef = useRef(false)
|
||
|
|
|
||
|
|
useEffect(() => {
|
||
|
|
if (!loading && user && !isAdmin && !toastShownRef.current) {
|
||
|
|
toastShownRef.current = true
|
||
|
|
toast.error('Admin access required')
|
||
|
|
}
|
||
|
|
}, [loading, user, isAdmin])
|
||
|
|
|
||
|
|
if (loading) {
|
||
|
|
return (
|
||
|
|
<div className="min-h-screen flex items-center justify-center">
|
||
|
|
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-accent-500" />
|
||
|
|
</div>
|
||
|
|
)
|
||
|
|
}
|
||
|
|
|
||
|
|
if (!user) {
|
||
|
|
return <Navigate to="/login" state={{ from: location }} replace />
|
||
|
|
}
|
||
|
|
|
||
|
|
if (!isAdmin) {
|
||
|
|
return <Navigate to="/" replace />
|
||
|
|
}
|
||
|
|
|
||
|
|
return <>{children}</>
|
||
|
|
}
|