管理后台

This commit is contained in:
xujiang
2025-07-09 17:50:29 +08:00
parent 0651b6626a
commit 5f2152c7a6
40 changed files with 3839 additions and 795 deletions

View File

@ -0,0 +1,110 @@
import { Component, ErrorInfo, ReactNode } from 'react'
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card'
import { Button } from '@/components/ui/button'
import { Alert, AlertDescription } from '@/components/ui/alert'
import { AlertTriangle, RefreshCw, Home } from 'lucide-react'
interface Props {
children: ReactNode
fallback?: ReactNode
}
interface State {
hasError: boolean
error?: Error
errorInfo?: ErrorInfo
}
export class ErrorBoundary extends Component<Props, State> {
constructor(props: Props) {
super(props)
this.state = { hasError: false }
}
static getDerivedStateFromError(error: Error): State {
return { hasError: true, error }
}
componentDidCatch(error: Error, errorInfo: ErrorInfo) {
this.setState({
error,
errorInfo
})
// 这里可以将错误发送到日志服务
console.error('ErrorBoundary caught an error:', error, errorInfo)
}
handleRetry = () => {
this.setState({ hasError: false, error: undefined, errorInfo: undefined })
}
handleGoHome = () => {
window.location.href = '/dashboard'
}
render() {
if (this.state.hasError) {
if (this.props.fallback) {
return this.props.fallback
}
return (
<div className="min-h-screen flex items-center justify-center bg-gray-50 p-4">
<Card className="w-full max-w-md">
<CardHeader className="text-center">
<div className="flex justify-center mb-4">
<AlertTriangle className="h-12 w-12 text-red-500" />
</div>
<CardTitle className="text-xl"></CardTitle>
</CardHeader>
<CardContent className="space-y-4">
<Alert variant="destructive">
<AlertTriangle className="h-4 w-4" />
<AlertDescription>
</AlertDescription>
</Alert>
{import.meta.env.DEV && this.state.error && (
<div className="p-4 bg-gray-100 rounded-md">
<h4 className="font-medium text-sm mb-2"></h4>
<pre className="text-xs text-red-600 whitespace-pre-wrap">
{this.state.error.toString()}
</pre>
{this.state.errorInfo && (
<pre className="text-xs text-gray-600 mt-2 whitespace-pre-wrap">
{this.state.errorInfo.componentStack}
</pre>
)}
</div>
)}
<div className="flex gap-2">
<Button
onClick={this.handleRetry}
className="flex-1"
variant="outline"
>
<RefreshCw className="h-4 w-4 mr-2" />
</Button>
<Button
onClick={this.handleGoHome}
className="flex-1"
>
<Home className="h-4 w-4 mr-2" />
</Button>
</div>
</CardContent>
</Card>
</div>
)
}
return this.props.children
}
}
export default ErrorBoundary