Files
ExecuBoard/app/admin/page.tsx
2025-08-01 01:19:19 +08:00

704 lines
37 KiB
TypeScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

"use client"
import { useState, useEffect } from "react"
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"
import { Badge } from "@/components/ui/badge"
import { Button } from "@/components/ui/button"
import { Input } from "@/components/ui/input"
import { Label } from "@/components/ui/label"
import { Textarea } from "@/components/ui/textarea"
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs"
import {
Dialog,
DialogContent,
DialogDescription,
DialogFooter,
DialogHeader,
DialogTitle,
} from "@/components/ui/dialog"
import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow } from "@/components/ui/table"
import { Avatar, AvatarFallback } from "@/components/ui/avatar"
import { Switch } from "@/components/ui/switch"
import { Users, Settings, Target, MessageSquare, Plus, Edit, Trash2, Shield, Database, Bell } from "lucide-react"
// 用戶介面
interface User {
id: string
name: string
email: string
role: "executive" | "manager" | "hr"
avatar_url?: string
created_at: string
updated_at: string
}
// 用戶顏色映射
const getUserColor = (id: string) => {
const colors = ["emerald", "blue", "purple", "orange", "cyan", "pink", "indigo", "yellow"]
const index = id.charCodeAt(id.length - 1) % colors.length
return colors[index]
}
const kpiTemplates = [
{ id: 1, name: "營收成長率", category: "財務", defaultWeight: 30, isActive: true, usageCount: 15, color: "emerald" },
{ id: 2, name: "團隊滿意度", category: "團隊", defaultWeight: 25, isActive: true, usageCount: 12, color: "blue" },
{ id: 3, name: "市場佔有率", category: "營運", defaultWeight: 20, isActive: true, usageCount: 8, color: "purple" },
{ id: 4, name: "創新指數", category: "創新", defaultWeight: 25, isActive: false, usageCount: 3, color: "orange" },
{ id: 5, name: "客戶滿意度", category: "營運", defaultWeight: 20, isActive: true, usageCount: 10, color: "cyan" },
{ id: 6, name: "成本控制率", category: "財務", defaultWeight: 15, isActive: true, usageCount: 6, color: "red" },
]
const questionTemplates = [
{ id: 1, question: "您如何評價本季度的整體表現?", type: "評分", category: "績效", isRequired: true, color: "blue" },
{ id: 2, question: "本期間您的主要成就是什麼?", type: "文字", category: "績效", isRequired: true, color: "emerald" },
{
id: 3,
question: "您面臨了哪些挑戰,如何克服?",
type: "文字",
category: "挑戰",
isRequired: false,
color: "yellow",
},
{ id: 4, question: "您如何有效領導團隊?", type: "評分", category: "領導力", isRequired: true, color: "purple" },
{ id: 5, question: "下一季度的目標是什麼?", type: "文字", category: "目標", isRequired: true, color: "orange" },
]
const systemSettings = [
{ id: 1, name: "自動提醒", description: "自動發送審查提醒通知", enabled: true, category: "通知" },
{ id: 2, name: "郵件通知", description: "透過郵件發送系統通知", enabled: true, category: "通知" },
{ id: 3, name: "資料備份", description: "每日自動備份系統資料", enabled: true, category: "安全" },
{ id: 4, name: "雙重驗證", description: "要求使用者啟用雙重驗證", enabled: false, category: "安全" },
{ id: 5, name: "審計日誌", description: "記錄所有系統操作日誌", enabled: true, category: "安全" },
]
export default function AdminPanel() {
const [activeTab, setActiveTab] = useState("users")
const [isDialogOpen, setIsDialogOpen] = useState(false)
const [dialogType, setDialogType] = useState("")
const [users, setUsers] = useState<User[]>([])
const [loading, setLoading] = useState(true)
const [error, setError] = useState<string | null>(null)
const [editingUser, setEditingUser] = useState<User | null>(null)
const [editForm, setEditForm] = useState({
name: "",
email: "",
role: "manager" as "executive" | "manager" | "hr"
})
// 載入用戶數據
useEffect(() => {
fetchUsers()
}, [])
const fetchUsers = async () => {
try {
setLoading(true)
const response = await fetch('/api/users')
if (!response.ok) {
throw new Error('載入用戶數據失敗')
}
const data = await response.json()
setUsers(data)
} catch (err) {
setError(err instanceof Error ? err.message : '載入用戶數據失敗')
} finally {
setLoading(false)
}
}
const handleOpenDialog = (type: string) => {
setDialogType(type)
setIsDialogOpen(true)
}
const handleEditUser = (user: User) => {
setEditingUser(user)
setEditForm({
name: user.name,
email: user.email,
role: user.role
})
setDialogType("edit-user")
setIsDialogOpen(true)
}
const handleUpdateUser = async () => {
if (!editingUser) return
try {
const response = await fetch(`/api/users/${editingUser.id}`, {
method: 'PUT',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify(editForm),
})
if (!response.ok) {
throw new Error('更新用戶失敗')
}
// 重新載入用戶數據
await fetchUsers()
setIsDialogOpen(false)
setEditingUser(null)
setEditForm({ name: "", email: "", role: "manager" })
} catch (err) {
setError(err instanceof Error ? err.message : '更新用戶失敗')
}
}
const handleDeleteUser = async (userId: string) => {
if (!confirm('確定要刪除此用戶嗎?此操作無法復原。')) {
return
}
try {
const response = await fetch(`/api/users/${userId}`, {
method: 'DELETE',
})
if (!response.ok) {
throw new Error('刪除用戶失敗')
}
// 重新載入用戶數據
await fetchUsers()
} catch (err) {
setError(err instanceof Error ? err.message : '刪除用戶失敗')
}
}
const adminStats = {
totalUsers: users.length,
activeUsers: users.length, // 所有用戶都視為活躍
totalKPIs: kpiTemplates.length,
activeKPIs: kpiTemplates.filter((k) => k.isActive).length,
totalQuestions: questionTemplates.length,
requiredQuestions: questionTemplates.filter((q) => q.isRequired).length,
}
return (
<div className="min-h-screen bg-gradient-to-br from-slate-50 via-purple-50 to-pink-50 p-3 sm:p-6">
<div className="max-w-7xl mx-auto space-y-4 sm:space-y-6">
{/* Header */}
<div className="flex flex-col space-y-4 sm:flex-row sm:items-center sm:justify-between sm:space-y-0">
<div className="flex flex-col sm:flex-row sm:items-center space-y-2 sm:space-y-0 sm:space-x-4">
<img
src=""
alt="公司 Logo"
className="h-10 w-auto sm:h-12 md:hidden"
/>
<div>
<h1 className="text-xl sm:text-2xl lg:text-3xl font-bold bg-gradient-to-r from-purple-600 to-pink-600 bg-clip-text text-transparent">
</h1>
<p className="text-sm sm:text-base text-gray-600 mt-1"></p>
</div>
</div>
<div className="flex space-x-2">
<Button
variant="outline"
className="bg-white/80 backdrop-blur-sm"
onClick={() => handleOpenDialog("backup")}
>
<Database className="h-4 w-4 mr-2" />
</Button>
<Button
className="bg-gradient-to-r from-purple-500 to-pink-500 hover:from-purple-600 hover:to-pink-600 text-white shadow-lg"
onClick={() => handleOpenDialog("user")}
>
<Plus className="h-4 w-4 mr-2" />
</Button>
</div>
</div>
{/* Stats Cards */}
<div className="grid grid-cols-2 sm:grid-cols-3 lg:grid-cols-6 gap-3 sm:gap-4">
<Card className="shadow-lg bg-gradient-to-br from-blue-50 to-cyan-100 border-0">
<CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
<CardTitle className="text-sm font-medium text-blue-700"></CardTitle>
<Users className="h-4 w-4 text-blue-600" />
</CardHeader>
<CardContent>
<div className="text-2xl font-bold text-blue-800">{adminStats.totalUsers}</div>
</CardContent>
</Card>
<Card className="shadow-lg bg-gradient-to-br from-green-50 to-emerald-100 border-0">
<CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
<CardTitle className="text-sm font-medium text-green-700"></CardTitle>
<Shield className="h-4 w-4 text-green-600" />
</CardHeader>
<CardContent>
<div className="text-2xl font-bold text-green-800">{adminStats.activeUsers}</div>
</CardContent>
</Card>
<Card className="shadow-lg bg-gradient-to-br from-purple-50 to-violet-100 border-0">
<CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
<CardTitle className="text-sm font-medium text-purple-700">KPI </CardTitle>
<Target className="h-4 w-4 text-purple-600" />
</CardHeader>
<CardContent>
<div className="text-2xl font-bold text-purple-800">{adminStats.totalKPIs}</div>
</CardContent>
</Card>
<Card className="shadow-lg bg-gradient-to-br from-orange-50 to-amber-100 border-0">
<CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
<CardTitle className="text-sm font-medium text-orange-700"> KPI</CardTitle>
<Settings className="h-4 w-4 text-orange-600" />
</CardHeader>
<CardContent>
<div className="text-2xl font-bold text-orange-800">{adminStats.activeKPIs}</div>
</CardContent>
</Card>
<Card className="shadow-lg bg-gradient-to-br from-pink-50 to-rose-100 border-0">
<CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
<CardTitle className="text-sm font-medium text-pink-700"></CardTitle>
<MessageSquare className="h-4 w-4 text-pink-600" />
</CardHeader>
<CardContent>
<div className="text-2xl font-bold text-pink-800">{adminStats.totalQuestions}</div>
</CardContent>
</Card>
<Card className="shadow-lg bg-gradient-to-br from-indigo-50 to-blue-100 border-0">
<CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
<CardTitle className="text-sm font-medium text-indigo-700"></CardTitle>
<Bell className="h-4 w-4 text-indigo-600" />
</CardHeader>
<CardContent>
<div className="text-2xl font-bold text-indigo-800">{adminStats.requiredQuestions}</div>
</CardContent>
</Card>
</div>
{/* Main Content */}
<Card className="shadow-lg bg-white/80 backdrop-blur-sm">
<CardHeader>
<CardTitle className="text-lg font-bold bg-gradient-to-r from-gray-800 to-gray-600 bg-clip-text text-transparent">
</CardTitle>
<CardDescription></CardDescription>
</CardHeader>
<CardContent>
<Tabs value={activeTab} onValueChange={setActiveTab}>
<TabsList className="grid w-full grid-cols-4">
<TabsTrigger value="users"></TabsTrigger>
<TabsTrigger value="kpi-templates">KPI </TabsTrigger>
<TabsTrigger value="questions"></TabsTrigger>
<TabsTrigger value="settings"></TabsTrigger>
</TabsList>
{/* Users Tab */}
<TabsContent value="users" className="space-y-4">
<div className="flex justify-between items-center">
<h3 className="text-lg font-semibold"></h3>
<Button onClick={() => handleOpenDialog("user")}>
<Plus className="h-4 w-4 mr-2" />
</Button>
</div>
{error && (
<div className="bg-red-50 border border-red-200 text-red-700 px-4 py-3 rounded">
{error}
</div>
)}
{loading ? (
<div className="text-center py-8">
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-purple-600 mx-auto"></div>
<p className="mt-2 text-gray-600">...</p>
</div>
) : (
<Table>
<TableHeader>
<TableRow>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
</TableRow>
</TableHeader>
<TableBody>
{users.map((user) => {
const userColor = getUserColor(user.id)
return (
<TableRow key={user.id}>
<TableCell>
<div className="flex items-center space-x-3">
<Avatar className="h-8 w-8">
<AvatarFallback
className={`bg-gradient-to-br ${
userColor === "emerald"
? "from-emerald-400 to-green-500"
: userColor === "blue"
? "from-blue-400 to-cyan-500"
: userColor === "purple"
? "from-purple-400 to-violet-500"
: userColor === "orange"
? "from-orange-400 to-amber-500"
: userColor === "cyan"
? "from-cyan-400 to-teal-500"
: userColor === "pink"
? "from-pink-400 to-rose-500"
: userColor === "indigo"
? "from-indigo-400 to-blue-500"
: "from-yellow-400 to-amber-500"
} text-white`}
>
{user.name.charAt(0)}
</AvatarFallback>
</Avatar>
<div>
<div className="font-medium">{user.name}</div>
<div className="text-sm text-gray-500">{user.email}</div>
</div>
</div>
</TableCell>
<TableCell>
<Badge variant={user.role === "executive" ? "default" : "secondary"}>
{user.role === "executive" ? "高階主管" : user.role === "manager" ? "經理" : "HR"}
</Badge>
</TableCell>
<TableCell className="text-sm text-gray-500">
{new Date(user.created_at).toLocaleDateString('zh-TW')}
</TableCell>
<TableCell>
<div className="flex items-center space-x-2">
<Button
variant="ghost"
size="sm"
onClick={() => handleEditUser(user)}
>
<Edit className="h-4 w-4" />
</Button>
<Button
variant="ghost"
size="sm"
className="text-red-500"
onClick={() => handleDeleteUser(user.id)}
>
<Trash2 className="h-4 w-4" />
</Button>
</div>
</TableCell>
</TableRow>
)
})}
</TableBody>
</Table>
)}
</TabsContent>
{/* KPI Templates Tab */}
<TabsContent value="kpi-templates" className="space-y-4">
<div className="flex justify-between items-center">
<h3 className="text-lg font-semibold">KPI </h3>
<Button onClick={() => handleOpenDialog("kpi")}>
<Plus className="h-4 w-4 mr-2" />
</Button>
</div>
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
{kpiTemplates.map((template) => (
<Card
key={template.id}
className={`shadow-md hover:shadow-lg transition-shadow ${
template.color === "emerald"
? "bg-gradient-to-br from-emerald-50 to-green-100"
: template.color === "blue"
? "bg-gradient-to-br from-blue-50 to-cyan-100"
: template.color === "purple"
? "bg-gradient-to-br from-purple-50 to-violet-100"
: template.color === "orange"
? "bg-gradient-to-br from-orange-50 to-amber-100"
: template.color === "cyan"
? "bg-gradient-to-br from-cyan-50 to-teal-100"
: "bg-gradient-to-br from-red-50 to-pink-100"
}`}
>
<CardHeader className="pb-3">
<div className="flex items-center justify-between">
<CardTitle className="text-base">{template.name}</CardTitle>
<Badge variant={template.isActive ? "default" : "secondary"}>
{template.isActive ? "啟用" : "停用"}
</Badge>
</div>
</CardHeader>
<CardContent className="space-y-3">
<div className="flex justify-between text-sm">
<span className="text-gray-600">:</span>
<span className="font-medium">{template.category}</span>
</div>
<div className="flex justify-between text-sm">
<span className="text-gray-600">:</span>
<span className="font-medium">{template.defaultWeight}%</span>
</div>
<div className="flex justify-between text-sm">
<span className="text-gray-600">使:</span>
<span className="font-medium">{template.usageCount}</span>
</div>
<div className="flex justify-between pt-2 border-t">
<Button variant="ghost" size="sm">
<Edit className="h-4 w-4" />
</Button>
<Button variant="ghost" size="sm" className="text-red-500">
<Trash2 className="h-4 w-4" />
</Button>
</div>
</CardContent>
</Card>
))}
</div>
</TabsContent>
{/* Questions Tab */}
<TabsContent value="questions" className="space-y-4">
<div className="flex justify-between items-center">
<h3 className="text-lg font-semibold"></h3>
<Button onClick={() => handleOpenDialog("question")}>
<Plus className="h-4 w-4 mr-2" />
</Button>
</div>
<div className="space-y-3">
{questionTemplates.map((question) => (
<Card key={question.id} className="shadow-md">
<CardContent className="p-4">
<div className="flex items-start justify-between">
<div className="flex-1">
<div className="flex items-center space-x-2 mb-2">
<Badge
variant="outline"
className={`${
question.color === "blue"
? "border-blue-300 text-blue-700 bg-blue-50"
: question.color === "emerald"
? "border-emerald-300 text-emerald-700 bg-emerald-50"
: question.color === "yellow"
? "border-yellow-300 text-yellow-700 bg-yellow-50"
: question.color === "purple"
? "border-purple-300 text-purple-700 bg-purple-50"
: "border-orange-300 text-orange-700 bg-orange-50"
}`}
>
{question.category}
</Badge>
<Badge variant={question.type === "評分" ? "default" : "secondary"}>
{question.type}
</Badge>
{question.isRequired && (
<Badge variant="destructive" className="text-xs">
</Badge>
)}
</div>
<p className="text-gray-900 font-medium">{question.question}</p>
</div>
<div className="flex items-center space-x-2 ml-4">
<Button variant="ghost" size="sm">
<Edit className="h-4 w-4" />
</Button>
<Button variant="ghost" size="sm" className="text-red-500">
<Trash2 className="h-4 w-4" />
</Button>
</div>
</div>
</CardContent>
</Card>
))}
</div>
</TabsContent>
{/* Settings Tab */}
<TabsContent value="settings" className="space-y-4">
<h3 className="text-lg font-semibold"></h3>
<div className="space-y-4">
{systemSettings.map((setting) => (
<Card key={setting.id} className="shadow-md">
<CardContent className="p-4">
<div className="flex items-center justify-between">
<div className="flex-1">
<div className="flex items-center space-x-2 mb-1">
<h4 className="font-medium text-gray-900">{setting.name}</h4>
<Badge
variant="outline"
className={
setting.category === "通知"
? "border-blue-300 text-blue-700 bg-blue-50"
: "border-red-300 text-red-700 bg-red-50"
}
>
{setting.category}
</Badge>
</div>
<p className="text-sm text-gray-600">{setting.description}</p>
</div>
<Switch checked={setting.enabled} />
</div>
</CardContent>
</Card>
))}
</div>
</TabsContent>
</Tabs>
</CardContent>
</Card>
{/* Dialog for various actions */}
<Dialog open={isDialogOpen} onOpenChange={setIsDialogOpen}>
<DialogContent className="max-w-md">
<DialogHeader>
<DialogTitle className="bg-gradient-to-r from-purple-600 to-pink-600 bg-clip-text text-transparent">
{dialogType === "user"
? "新增用戶"
: dialogType === "edit-user"
? "編輯用戶"
: dialogType === "kpi"
? "新增 KPI 模板"
: dialogType === "question"
? "新增審查問題"
: "備份資料"}
</DialogTitle>
<DialogDescription>
{dialogType === "user"
? "建立新的系統用戶帳號"
: dialogType === "edit-user"
? "修改用戶資訊"
: dialogType === "kpi"
? "建立新的 KPI 模板"
: dialogType === "question"
? "建立新的審查問題"
: "備份系統資料到安全位置"}
</DialogDescription>
</DialogHeader>
<div className="space-y-4">
{(dialogType === "user" || dialogType === "edit-user") && (
<>
<div>
<Label htmlFor="userName"></Label>
<Input
id="userName"
placeholder="輸入用戶姓名"
value={editForm.name}
onChange={(e) => setEditForm({ ...editForm, name: e.target.value })}
/>
</div>
<div>
<Label htmlFor="userEmail"></Label>
<Input
id="userEmail"
type="email"
placeholder="輸入電子郵件"
value={editForm.email}
onChange={(e) => setEditForm({ ...editForm, email: e.target.value })}
/>
</div>
<div>
<Label htmlFor="userRole"></Label>
<Select
value={editForm.role}
onValueChange={(value: "executive" | "manager" | "hr") =>
setEditForm({ ...editForm, role: value })
}
>
<SelectTrigger>
<SelectValue placeholder="選擇角色" />
</SelectTrigger>
<SelectContent>
<SelectItem value="executive"></SelectItem>
<SelectItem value="manager"></SelectItem>
<SelectItem value="hr">HR</SelectItem>
</SelectContent>
</Select>
</div>
</>
)}
{dialogType === "kpi" && (
<>
<div>
<Label htmlFor="kpiName">KPI </Label>
<Input id="kpiName" placeholder="輸入 KPI 名稱" />
</div>
<div>
<Label htmlFor="kpiCategory"></Label>
<Select>
<SelectTrigger>
<SelectValue placeholder="選擇類別" />
</SelectTrigger>
<SelectContent>
<SelectItem value="財務"></SelectItem>
<SelectItem value="團隊"></SelectItem>
<SelectItem value="營運"></SelectItem>
<SelectItem value="創新"></SelectItem>
</SelectContent>
</Select>
</div>
<div>
<Label htmlFor="defaultWeight"> (%)</Label>
<Input id="defaultWeight" type="number" placeholder="0-100" />
</div>
</>
)}
{dialogType === "question" && (
<>
<div>
<Label htmlFor="questionText"></Label>
<Textarea id="questionText" placeholder="輸入審查問題..." />
</div>
<div>
<Label htmlFor="questionType"></Label>
<Select>
<SelectTrigger>
<SelectValue placeholder="選擇類型" />
</SelectTrigger>
<SelectContent>
<SelectItem value="評分"></SelectItem>
<SelectItem value="文字"></SelectItem>
<SelectItem value="選擇"></SelectItem>
</SelectContent>
</Select>
</div>
<div>
<Label htmlFor="questionCategory"></Label>
<Select>
<SelectTrigger>
<SelectValue placeholder="選擇類別" />
</SelectTrigger>
<SelectContent>
<SelectItem value="績效"></SelectItem>
<SelectItem value="領導力"></SelectItem>
<SelectItem value="目標"></SelectItem>
<SelectItem value="挑戰"></SelectItem>
</SelectContent>
</Select>
</div>
</>
)}
{dialogType === "backup" && (
<div className="text-center py-4">
<Database className="h-12 w-12 mx-auto text-gray-400 mb-4" />
<p className="text-gray-600"></p>
<p className="text-sm text-gray-500 mt-2"></p>
</div>
)}
</div>
<DialogFooter>
<Button
type="submit"
className="bg-gradient-to-r from-green-500 to-emerald-500 hover:from-green-600 hover:to-emerald-600 text-white"
onClick={dialogType === "edit-user" ? handleUpdateUser : undefined}
>
{dialogType === "backup" ? "開始備份" : dialogType === "edit-user" ? "更新" : "建立"}
</Button>
</DialogFooter>
</DialogContent>
</Dialog>
</div>
</div>
)
}