1509 lines
67 KiB
TypeScript
1509 lines
67 KiB
TypeScript
"use client"
|
||
|
||
import { useState, useEffect } from "react"
|
||
import Link from "next/link"
|
||
import { useRouter } from "next/navigation"
|
||
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"
|
||
import { Button } from "@/components/ui/button"
|
||
import { Badge } from "@/components/ui/badge"
|
||
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs"
|
||
import { Dialog, DialogContent, DialogDescription, DialogHeader, DialogTitle } from "@/components/ui/dialog"
|
||
import {
|
||
Download,
|
||
Eye,
|
||
Users,
|
||
Heart,
|
||
FileText,
|
||
BarChart3,
|
||
RefreshCw,
|
||
Search,
|
||
Filter,
|
||
ArrowLeft,
|
||
Sparkles,
|
||
Settings,
|
||
Plus,
|
||
ChevronLeft,
|
||
ChevronRight,
|
||
TrendingUp,
|
||
TrendingDown,
|
||
Minus,
|
||
Target,
|
||
BookOpen,
|
||
ChevronDown,
|
||
ChevronUp,
|
||
Shield,
|
||
EyeOff,
|
||
HelpCircle,
|
||
X,
|
||
Calendar,
|
||
User,
|
||
Mail,
|
||
Tag,
|
||
Star,
|
||
Image as ImageIcon,
|
||
Clock,
|
||
MapPin,
|
||
Monitor
|
||
} from "lucide-react"
|
||
import { Input } from "@/components/ui/input"
|
||
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"
|
||
import HeaderMusicControl from "@/components/header-music-control"
|
||
import IpDisplay from "@/components/ip-display"
|
||
import RadarChart from "@/components/radar-chart"
|
||
import { categories, categorizeWishMultiple, type Wish } from "@/lib/categorization"
|
||
|
||
// 擴展 Wish 接口以包含額外屬性
|
||
interface ExtendedWish extends Wish {
|
||
isPublic?: boolean
|
||
email?: string
|
||
images?: any[]
|
||
like_count?: number
|
||
}
|
||
|
||
interface WishData {
|
||
id: number
|
||
title: string
|
||
current_pain: string
|
||
expected_solution: string
|
||
expected_effect: string
|
||
is_public: boolean
|
||
email: string
|
||
images: any[]
|
||
user_session: string
|
||
status: string
|
||
category: string
|
||
priority: number
|
||
like_count: number
|
||
created_at: string
|
||
updated_at: string
|
||
}
|
||
|
||
interface CategoryData {
|
||
name: string
|
||
count: number
|
||
percentage: number
|
||
color: string
|
||
keywords: string[]
|
||
description?: string
|
||
}
|
||
|
||
interface AdminStats {
|
||
totalWishes: number
|
||
publicWishes: number
|
||
privateWishes: number
|
||
totalLikes: number
|
||
categories: { [key: string]: number }
|
||
recentWishes: number
|
||
categoryDetails: CategoryData[]
|
||
recentTrends: {
|
||
thisWeek: number
|
||
lastWeek: number
|
||
growth: number
|
||
growthLabel: string
|
||
growthIcon: "up" | "down" | "flat"
|
||
growthColor: string
|
||
}
|
||
topKeywords: { word: string; count: number }[]
|
||
}
|
||
|
||
export default function AdminPage() {
|
||
const router = useRouter()
|
||
const [isAuthenticated, setIsAuthenticated] = useState(false)
|
||
const [wishes, setWishes] = useState<WishData[]>([])
|
||
const [filteredWishes, setFilteredWishes] = useState<WishData[]>([])
|
||
const [stats, setStats] = useState<AdminStats | null>(null)
|
||
const [loading, setLoading] = useState(true)
|
||
const [searchTerm, setSearchTerm] = useState("")
|
||
const [statusFilter, setStatusFilter] = useState("all")
|
||
const [visibilityFilter, setVisibilityFilter] = useState("all")
|
||
const [isExporting, setIsExporting] = useState(false)
|
||
const [isExportingExcel, setIsExportingExcel] = useState(false)
|
||
const [showCategoryGuide, setShowCategoryGuide] = useState(false)
|
||
const [showPrivacyDetails, setShowPrivacyDetails] = useState(false)
|
||
const [selectedWish, setSelectedWish] = useState<WishData | null>(null)
|
||
const [showWishDetails, setShowWishDetails] = useState(false)
|
||
const [wishDetails, setWishDetails] = useState<any>(null)
|
||
const [loadingDetails, setLoadingDetails] = useState(false)
|
||
const [showImageModal, setShowImageModal] = useState(false)
|
||
const [selectedImage, setSelectedImage] = useState<any>(null)
|
||
|
||
// 分頁狀態
|
||
const [currentPage, setCurrentPage] = useState(1)
|
||
const itemsPerPage = 10
|
||
|
||
// 分析許願內容(包含所有數據,包括私密的)
|
||
const analyzeWishes = (wishList: WishData[]): AdminStats => {
|
||
const totalWishes = wishList.length
|
||
const publicWishes = wishList.filter((wish) => wish.is_public !== false).length
|
||
const privateWishes = wishList.filter((wish) => wish.is_public === false).length
|
||
const totalLikes = wishList.reduce((sum, wish) => sum + wish.like_count, 0)
|
||
|
||
const categoryStats: { [key: string]: number } = {}
|
||
const keywordCount: { [key: string]: number } = {}
|
||
|
||
// 初始化分類統計
|
||
categories.forEach((cat) => {
|
||
categoryStats[cat.name] = 0
|
||
})
|
||
|
||
// 分析每個許願(多標籤統計)- 包含所有數據
|
||
wishList.forEach((wish) => {
|
||
// 轉換數據格式以匹配 categorization.ts 的 Wish 接口
|
||
const convertedWish: ExtendedWish = {
|
||
id: wish.id,
|
||
title: wish.title,
|
||
currentPain: wish.current_pain,
|
||
expectedSolution: wish.expected_solution,
|
||
expectedEffect: wish.expected_effect || "",
|
||
createdAt: wish.created_at,
|
||
isPublic: wish.is_public,
|
||
email: wish.email,
|
||
images: wish.images,
|
||
like_count: wish.like_count || 0,
|
||
}
|
||
|
||
const wishCategories = categorizeWishMultiple(convertedWish)
|
||
|
||
wishCategories.forEach((category) => {
|
||
categoryStats[category.name]++
|
||
|
||
// 統計關鍵字
|
||
if (category.keywords) {
|
||
const fullText =
|
||
`${wish.title} ${wish.current_pain} ${wish.expected_solution} ${wish.expected_effect}`.toLowerCase()
|
||
category.keywords.forEach((keyword: string) => {
|
||
if (fullText.includes(keyword.toLowerCase())) {
|
||
keywordCount[keyword] = (keywordCount[keyword] || 0) + 1
|
||
}
|
||
})
|
||
}
|
||
})
|
||
})
|
||
|
||
// 計算百分比和準備數據,保留"其他問題"分類
|
||
const categoryDetails: CategoryData[] = categories.map((cat) => ({
|
||
name: cat.name,
|
||
count: categoryStats[cat.name] || 0,
|
||
percentage: totalWishes > 0 ? Math.round(((categoryStats[cat.name] || 0) / totalWishes) * 100) : 0,
|
||
color: cat.color,
|
||
keywords: cat.keywords,
|
||
description: cat.description,
|
||
}))
|
||
|
||
// 改進的趨勢計算
|
||
const now = new Date()
|
||
const oneWeekAgo = new Date(now.getTime() - 7 * 24 * 60 * 60 * 1000)
|
||
const twoWeeksAgo = new Date(now.getTime() - 14 * 24 * 60 * 60 * 1000)
|
||
|
||
const thisWeek = wishList.filter((wish) => new Date(wish.created_at) >= oneWeekAgo).length
|
||
const lastWeek = wishList.filter((wish) => {
|
||
const date = new Date(wish.created_at)
|
||
return date >= twoWeeksAgo && date < oneWeekAgo
|
||
}).length
|
||
|
||
// 改進的成長趨勢計算
|
||
let growth = 0
|
||
let growthLabel = "持平"
|
||
let growthIcon: "up" | "down" | "flat" = "flat"
|
||
let growthColor = "#6B7280"
|
||
|
||
if (lastWeek === 0 && thisWeek > 0) {
|
||
// 上週沒有,本週有 → 全新開始
|
||
growth = 100
|
||
growthLabel = "開始增長"
|
||
growthIcon = "up"
|
||
growthColor = "#10B981"
|
||
} else if (lastWeek === 0 && thisWeek === 0) {
|
||
// 兩週都沒有
|
||
growth = 0
|
||
growthLabel = "尚無數據"
|
||
growthIcon = "flat"
|
||
growthColor = "#6B7280"
|
||
} else if (lastWeek > 0) {
|
||
// 正常計算成長率
|
||
growth = Math.round(((thisWeek - lastWeek) / lastWeek) * 100)
|
||
|
||
if (growth > 0) {
|
||
growthLabel = "持續增長"
|
||
growthIcon = "up"
|
||
growthColor = "#10B981"
|
||
} else if (growth < 0) {
|
||
growthLabel = "有所下降"
|
||
growthIcon = "down"
|
||
growthColor = "#EF4444"
|
||
} else {
|
||
growthLabel = "保持穩定"
|
||
growthIcon = "flat"
|
||
growthColor = "#6B7280"
|
||
}
|
||
}
|
||
|
||
// 取得熱門關鍵字
|
||
const topKeywords = Object.entries(keywordCount)
|
||
.sort(([, a], [, b]) => b - a)
|
||
.slice(0, 15)
|
||
.map(([word, count]) => ({ word, count }))
|
||
|
||
return {
|
||
totalWishes,
|
||
publicWishes,
|
||
privateWishes,
|
||
totalLikes,
|
||
categories: categoryStats,
|
||
recentWishes: thisWeek,
|
||
categoryDetails,
|
||
recentTrends: {
|
||
thisWeek,
|
||
lastWeek,
|
||
growth,
|
||
growthLabel,
|
||
growthIcon,
|
||
growthColor,
|
||
},
|
||
topKeywords,
|
||
}
|
||
}
|
||
|
||
// 獲取所有數據
|
||
const fetchData = async () => {
|
||
try {
|
||
setLoading(true)
|
||
|
||
// 獲取困擾案例數據
|
||
const wishesResponse = await fetch('/api/admin/wishes')
|
||
const wishesResult = await wishesResponse.json()
|
||
|
||
if (wishesResult.success) {
|
||
const wishesData = wishesResult.data
|
||
setWishes(wishesData)
|
||
setFilteredWishes(wishesData)
|
||
|
||
// 使用本地分析函數生成詳細統計
|
||
const detailedStats = analyzeWishes(wishesData)
|
||
setStats(detailedStats)
|
||
}
|
||
|
||
} catch (error) {
|
||
console.error('獲取數據失敗:', error)
|
||
} finally {
|
||
setLoading(false)
|
||
}
|
||
}
|
||
|
||
// 過濾數據
|
||
const filterData = () => {
|
||
let filtered = wishes
|
||
|
||
// 搜索過濾
|
||
if (searchTerm) {
|
||
filtered = filtered.filter(wish =>
|
||
wish.title.toLowerCase().includes(searchTerm.toLowerCase()) ||
|
||
wish.current_pain.toLowerCase().includes(searchTerm.toLowerCase()) ||
|
||
wish.expected_solution.toLowerCase().includes(searchTerm.toLowerCase())
|
||
)
|
||
}
|
||
|
||
// 狀態過濾
|
||
if (statusFilter !== "all") {
|
||
filtered = filtered.filter(wish => wish.status === statusFilter)
|
||
}
|
||
|
||
// 可見性過濾
|
||
if (visibilityFilter !== "all") {
|
||
filtered = filtered.filter(wish =>
|
||
visibilityFilter === "public" ? wish.is_public : !wish.is_public
|
||
)
|
||
}
|
||
|
||
setFilteredWishes(filtered)
|
||
setCurrentPage(1) // 重置到第一頁
|
||
}
|
||
|
||
// 分頁計算
|
||
const totalPages = Math.ceil(filteredWishes.length / itemsPerPage)
|
||
const startIndex = (currentPage - 1) * itemsPerPage
|
||
const endIndex = startIndex + itemsPerPage
|
||
const currentWishes = filteredWishes.slice(startIndex, endIndex)
|
||
|
||
// 分頁組件
|
||
const PaginationComponent = () => {
|
||
if (totalPages <= 1) return null
|
||
|
||
const getPageNumbers = () => {
|
||
const pages = []
|
||
const maxVisiblePages = 5
|
||
|
||
if (totalPages <= maxVisiblePages) {
|
||
for (let i = 1; i <= totalPages; i++) {
|
||
pages.push(i)
|
||
}
|
||
} else {
|
||
if (currentPage <= 3) {
|
||
for (let i = 1; i <= 4; i++) {
|
||
pages.push(i)
|
||
}
|
||
pages.push('...')
|
||
pages.push(totalPages)
|
||
} else if (currentPage >= totalPages - 2) {
|
||
pages.push(1)
|
||
pages.push('...')
|
||
for (let i = totalPages - 3; i <= totalPages; i++) {
|
||
pages.push(i)
|
||
}
|
||
} else {
|
||
pages.push(1)
|
||
pages.push('...')
|
||
for (let i = currentPage - 1; i <= currentPage + 1; i++) {
|
||
pages.push(i)
|
||
}
|
||
pages.push('...')
|
||
pages.push(totalPages)
|
||
}
|
||
}
|
||
|
||
return pages
|
||
}
|
||
|
||
return (
|
||
<div className="flex items-center justify-between mt-6">
|
||
<div className="text-sm text-blue-300">
|
||
顯示第 {startIndex + 1} - {Math.min(endIndex, filteredWishes.length)} 筆,共 {filteredWishes.length} 筆
|
||
</div>
|
||
<div className="flex items-center gap-2">
|
||
<Button
|
||
variant="outline"
|
||
size="sm"
|
||
onClick={() => setCurrentPage(Math.max(1, currentPage - 1))}
|
||
disabled={currentPage === 1}
|
||
className="text-blue-200 border-slate-600/50 hover:bg-slate-700/50 hover:text-white hover:border-cyan-400/50"
|
||
>
|
||
<ChevronLeft className="w-4 h-4" />
|
||
</Button>
|
||
|
||
{getPageNumbers().map((page, index) => (
|
||
<Button
|
||
key={index}
|
||
variant={page === currentPage ? "default" : "outline"}
|
||
size="sm"
|
||
onClick={() => typeof page === 'number' && setCurrentPage(page)}
|
||
disabled={page === '...'}
|
||
className={
|
||
page === currentPage
|
||
? "bg-cyan-600 hover:bg-cyan-700 text-white"
|
||
: "text-blue-200 border-slate-600/50 hover:bg-slate-700/50 hover:text-white hover:border-cyan-400/50"
|
||
}
|
||
>
|
||
{page}
|
||
</Button>
|
||
))}
|
||
|
||
<Button
|
||
variant="outline"
|
||
size="sm"
|
||
onClick={() => setCurrentPage(Math.min(totalPages, currentPage + 1))}
|
||
disabled={currentPage === totalPages}
|
||
className="text-blue-200 border-slate-600/50 hover:bg-slate-700/50 hover:text-white hover:border-cyan-400/50"
|
||
>
|
||
<ChevronRight className="w-4 h-4" />
|
||
</Button>
|
||
</div>
|
||
</div>
|
||
)
|
||
}
|
||
|
||
// 匯出 CSV
|
||
const exportToCSV = async () => {
|
||
try {
|
||
setIsExporting(true)
|
||
|
||
const response = await fetch('/api/admin/export-csv', {
|
||
method: 'POST',
|
||
headers: {
|
||
'Content-Type': 'application/json',
|
||
},
|
||
body: JSON.stringify({
|
||
data: filteredWishes,
|
||
filename: `困擾案例數據_${new Date().toISOString().split('T')[0]}.csv`
|
||
})
|
||
})
|
||
|
||
if (response.ok) {
|
||
const blob = await response.blob()
|
||
const url = window.URL.createObjectURL(blob)
|
||
const a = document.createElement('a')
|
||
a.href = url
|
||
a.download = `困擾案例數據_${new Date().toISOString().split('T')[0]}.csv`
|
||
document.body.appendChild(a)
|
||
a.click()
|
||
window.URL.revokeObjectURL(url)
|
||
document.body.removeChild(a)
|
||
} else {
|
||
throw new Error('匯出失敗')
|
||
}
|
||
} catch (error) {
|
||
console.error('匯出 CSV 失敗:', error)
|
||
alert('匯出失敗,請稍後再試')
|
||
} finally {
|
||
setIsExporting(false)
|
||
}
|
||
}
|
||
|
||
// 匯出 Excel
|
||
const exportToExcel = async () => {
|
||
try {
|
||
setIsExportingExcel(true)
|
||
|
||
const response = await fetch('/api/admin/export-excel', {
|
||
method: 'POST',
|
||
headers: {
|
||
'Content-Type': 'application/json',
|
||
},
|
||
body: JSON.stringify({
|
||
data: filteredWishes,
|
||
filename: `困擾案例數據_${new Date().toISOString().split('T')[0]}.xlsx`
|
||
})
|
||
})
|
||
|
||
if (response.ok) {
|
||
const blob = await response.blob()
|
||
const url = window.URL.createObjectURL(blob)
|
||
const a = document.createElement('a')
|
||
a.href = url
|
||
a.download = `困擾案例數據_${new Date().toISOString().split('T')[0]}.xlsx`
|
||
document.body.appendChild(a)
|
||
a.click()
|
||
window.URL.revokeObjectURL(url)
|
||
document.body.removeChild(a)
|
||
} else {
|
||
throw new Error('匯出失敗')
|
||
}
|
||
} catch (error) {
|
||
console.error('匯出 Excel 失敗:', error)
|
||
alert('匯出失敗,請稍後再試')
|
||
} finally {
|
||
setIsExportingExcel(false)
|
||
}
|
||
}
|
||
|
||
// 查看詳細資訊
|
||
const viewWishDetails = async (wish: WishData) => {
|
||
try {
|
||
setLoadingDetails(true)
|
||
setSelectedWish(wish)
|
||
|
||
const response = await fetch(`/api/admin/wishes/${wish.id}`)
|
||
const result = await response.json()
|
||
|
||
if (result.success) {
|
||
setWishDetails(result.data)
|
||
setShowWishDetails(true)
|
||
} else {
|
||
throw new Error(result.error || 'Failed to fetch details')
|
||
}
|
||
} catch (error) {
|
||
console.error('獲取詳細資訊失敗:', error)
|
||
alert('獲取詳細資訊失敗,請稍後再試')
|
||
} finally {
|
||
setLoadingDetails(false)
|
||
}
|
||
}
|
||
|
||
// 查看大圖
|
||
const viewImage = (image: any) => {
|
||
setSelectedImage(image)
|
||
setShowImageModal(true)
|
||
}
|
||
|
||
// 登出
|
||
const handleLogout = () => {
|
||
sessionStorage.removeItem("adminLoggedIn")
|
||
router.push("/admin/login")
|
||
}
|
||
|
||
// 檢查登入狀態
|
||
useEffect(() => {
|
||
const checkAuth = () => {
|
||
const loggedIn = sessionStorage.getItem("adminLoggedIn")
|
||
if (loggedIn === "true") {
|
||
setIsAuthenticated(true)
|
||
fetchData()
|
||
} else {
|
||
router.push("/admin/login")
|
||
}
|
||
}
|
||
|
||
checkAuth()
|
||
}, [router])
|
||
|
||
useEffect(() => {
|
||
if (isAuthenticated) {
|
||
fetchData()
|
||
}
|
||
}, [isAuthenticated])
|
||
|
||
useEffect(() => {
|
||
filterData()
|
||
}, [searchTerm, statusFilter, visibilityFilter, wishes])
|
||
|
||
if (loading) {
|
||
return (
|
||
<div className="min-h-screen bg-gradient-to-b from-slate-900 via-blue-900 to-indigo-900 flex items-center justify-center">
|
||
<div className="text-center">
|
||
<RefreshCw className="w-8 h-8 animate-spin text-white mx-auto mb-4" />
|
||
<p className="text-white">載入中...</p>
|
||
</div>
|
||
</div>
|
||
)
|
||
}
|
||
|
||
// 如果未認證,顯示載入中
|
||
if (!isAuthenticated) {
|
||
return (
|
||
<div className="min-h-screen bg-gradient-to-br from-slate-900 via-blue-900 to-slate-900 flex items-center justify-center">
|
||
<div className="text-center">
|
||
<div className="w-8 h-8 border-4 border-cyan-400 border-t-transparent rounded-full animate-spin mx-auto mb-4"></div>
|
||
<p className="text-white">驗證中...</p>
|
||
</div>
|
||
</div>
|
||
)
|
||
}
|
||
|
||
return (
|
||
<div className="min-h-screen bg-gradient-to-b from-slate-900 via-blue-900 to-indigo-900 relative">
|
||
{/* 星空背景 */}
|
||
<div className="absolute inset-0 overflow-hidden pointer-events-none">
|
||
<div className="absolute top-1/4 left-1/2 transform -translate-x-1/2 w-64 h-64 md:w-96 md:h-96 bg-gradient-radial from-cyan-400/20 via-blue-500/10 to-transparent rounded-full blur-3xl"></div>
|
||
</div>
|
||
|
||
{/* Header */}
|
||
<header className="border-b border-blue-800/50 bg-slate-900/80 backdrop-blur-md sticky top-0 z-[9999] shadow-lg shadow-slate-900/50">
|
||
<div className="container mx-auto px-3 sm:px-4 py-3 md:py-4">
|
||
<div className="flex items-center justify-between gap-2">
|
||
{/* Logo 區域 */}
|
||
<Link href="/" className="flex items-center gap-2 md:gap-3 min-w-0 flex-shrink-0 hover:opacity-80 transition-opacity">
|
||
<div className="w-7 h-7 sm:w-8 sm:h-8 md:w-10 md:h-10 bg-gradient-to-br from-cyan-400 to-blue-500 rounded-full flex items-center justify-center shadow-lg shadow-cyan-500/25">
|
||
<Sparkles className="w-3.5 h-3.5 sm:w-4 sm:h-4 md:w-6 md:h-6 text-white" />
|
||
</div>
|
||
<h1 className="text-base sm:text-lg md:text-2xl font-bold text-white whitespace-nowrap">資訊部.心願星河</h1>
|
||
</Link>
|
||
|
||
{/* 導航區域 */}
|
||
<nav className="flex items-center gap-1 sm:gap-2 md:gap-4 flex-shrink-0">
|
||
{/* IP顯示 */}
|
||
<div className="hidden sm:block">
|
||
<IpDisplay />
|
||
</div>
|
||
|
||
{/* 音樂控制 */}
|
||
<div className="hidden sm:block">
|
||
<HeaderMusicControl />
|
||
</div>
|
||
<div className="sm:hidden">
|
||
<HeaderMusicControl mobileSimplified />
|
||
</div>
|
||
|
||
{/* 登出按鈕 */}
|
||
<Button
|
||
onClick={handleLogout}
|
||
variant="ghost"
|
||
className="text-blue-200 hover:text-white hover:bg-blue-800/50 px-4"
|
||
>
|
||
<Settings className="w-4 h-4 mr-2" />
|
||
登出
|
||
</Button>
|
||
</nav>
|
||
</div>
|
||
</div>
|
||
</header>
|
||
|
||
{/* 主要內容 */}
|
||
<main className="relative z-10 py-8">
|
||
<div className="container mx-auto px-3 sm:px-4">
|
||
{/* 標題區域 */}
|
||
<div className="text-center mb-8">
|
||
<div className="flex items-center justify-center gap-3 mb-4">
|
||
<Settings className="w-8 h-8 text-cyan-400" />
|
||
<h2 className="text-3xl md:text-4xl font-bold text-white">後台管理系統</h2>
|
||
</div>
|
||
<p className="text-blue-200 text-lg">困擾案例數據管理與分析</p>
|
||
</div>
|
||
|
||
{/* 統計卡片 */}
|
||
{stats && (
|
||
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4 md:gap-6 mb-8">
|
||
<Card className="bg-slate-800/50 backdrop-blur-sm border-slate-700/50 hover:bg-slate-800/70 transition-all duration-300">
|
||
<CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
|
||
<CardTitle className="text-sm font-medium text-blue-200">總案例數</CardTitle>
|
||
<FileText className="h-4 w-4 text-cyan-400" />
|
||
</CardHeader>
|
||
<CardContent>
|
||
<div className="text-2xl font-bold text-white">{stats.totalWishes}</div>
|
||
<p className="text-xs text-blue-300">
|
||
公開 {stats.publicWishes} + 私密 {stats.privateWishes}
|
||
</p>
|
||
</CardContent>
|
||
</Card>
|
||
|
||
<Card className="bg-slate-800/50 backdrop-blur-sm border-slate-700/50 hover:bg-slate-800/70 transition-all duration-300">
|
||
<CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
|
||
<CardTitle className="text-sm font-medium text-blue-200">總點讚數</CardTitle>
|
||
<Heart className="h-4 w-4 text-pink-400" />
|
||
</CardHeader>
|
||
<CardContent>
|
||
<div className="text-2xl font-bold text-white">{stats.totalLikes}</div>
|
||
<p className="text-xs text-blue-300">用戶支持總數</p>
|
||
</CardContent>
|
||
</Card>
|
||
|
||
<Card className="bg-slate-800/50 backdrop-blur-sm border-slate-700/50 hover:bg-slate-800/70 transition-all duration-300">
|
||
<CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
|
||
<CardTitle className="text-sm font-medium text-blue-200">問題領域</CardTitle>
|
||
<BarChart3 className="h-4 w-4 text-green-400" />
|
||
</CardHeader>
|
||
<CardContent>
|
||
<div className="text-2xl font-bold text-white">{stats.categoryDetails?.filter((c) => c.count > 0).length || 0}</div>
|
||
<p className="text-xs text-blue-300">不同類別</p>
|
||
</CardContent>
|
||
</Card>
|
||
|
||
<Card className="bg-slate-800/50 backdrop-blur-sm border-slate-700/50 hover:bg-slate-800/70 transition-all duration-300">
|
||
<CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
|
||
<CardTitle className="text-sm font-medium text-blue-200">本週新增</CardTitle>
|
||
<Users className="h-4 w-4 text-yellow-400" />
|
||
</CardHeader>
|
||
<CardContent>
|
||
<div className="text-2xl font-bold text-white">{stats.recentWishes}</div>
|
||
<p className="text-xs text-blue-300">最近7天</p>
|
||
</CardContent>
|
||
</Card>
|
||
</div>
|
||
)}
|
||
|
||
{/* 主要內容區域 */}
|
||
<Tabs defaultValue="data" className="space-y-6">
|
||
<TabsList className="grid w-full grid-cols-2 bg-slate-800/50 border-slate-700/50">
|
||
<TabsTrigger value="data" className="text-blue-200 data-[state=active]:text-white data-[state=active]:bg-slate-700/50">數據管理</TabsTrigger>
|
||
<TabsTrigger value="analytics" className="text-blue-200 data-[state=active]:text-white data-[state=active]:bg-slate-700/50">數據分析</TabsTrigger>
|
||
</TabsList>
|
||
|
||
<TabsContent value="data" className="space-y-6">
|
||
{/* 搜索和過濾區域 */}
|
||
<Card className="bg-slate-800/50 backdrop-blur-sm border-slate-700/50">
|
||
<CardHeader>
|
||
<CardTitle className="text-white flex items-center gap-2">
|
||
<Filter className="w-5 h-5 text-cyan-400" />
|
||
數據篩選
|
||
</CardTitle>
|
||
<CardDescription className="text-blue-200">
|
||
搜索和過濾困擾案例數據
|
||
</CardDescription>
|
||
</CardHeader>
|
||
<CardContent>
|
||
<div className="grid grid-cols-1 md:grid-cols-5 gap-4">
|
||
<div className="relative">
|
||
<Search className="absolute left-3 top-3 h-4 w-4 text-blue-400" />
|
||
<Input
|
||
placeholder="搜索標題或內容..."
|
||
value={searchTerm}
|
||
onChange={(e) => setSearchTerm(e.target.value)}
|
||
className="pl-10 bg-slate-700/50 border-slate-600/50 text-white placeholder:text-blue-300 focus:border-cyan-400/50"
|
||
/>
|
||
</div>
|
||
|
||
<Select value={statusFilter} onValueChange={setStatusFilter}>
|
||
<SelectTrigger className="bg-slate-700/50 border-slate-600/50 text-white focus:border-cyan-400/50">
|
||
<SelectValue placeholder="狀態" />
|
||
</SelectTrigger>
|
||
<SelectContent className="bg-slate-800 border-slate-700">
|
||
<SelectItem value="all" className="text-white">所有狀態</SelectItem>
|
||
<SelectItem value="active" className="text-white">活躍</SelectItem>
|
||
<SelectItem value="inactive" className="text-white">非活躍</SelectItem>
|
||
</SelectContent>
|
||
</Select>
|
||
|
||
<Select value={visibilityFilter} onValueChange={setVisibilityFilter}>
|
||
<SelectTrigger className="bg-slate-700/50 border-slate-600/50 text-white focus:border-cyan-400/50">
|
||
<SelectValue placeholder="可見性" />
|
||
</SelectTrigger>
|
||
<SelectContent className="bg-slate-800 border-slate-700">
|
||
<SelectItem value="all" className="text-white">全部</SelectItem>
|
||
<SelectItem value="public" className="text-white">公開</SelectItem>
|
||
<SelectItem value="private" className="text-white">私密</SelectItem>
|
||
</SelectContent>
|
||
</Select>
|
||
|
||
<Button
|
||
onClick={exportToCSV}
|
||
disabled={isExporting}
|
||
className="bg-gradient-to-r from-green-600 to-emerald-600 hover:from-green-700 hover:to-emerald-700 text-white shadow-lg shadow-green-500/25"
|
||
>
|
||
{isExporting ? (
|
||
<RefreshCw className="w-4 h-4 mr-2 animate-spin" />
|
||
) : (
|
||
<Download className="w-4 h-4 mr-2" />
|
||
)}
|
||
匯出 CSV
|
||
</Button>
|
||
|
||
<Button
|
||
onClick={exportToExcel}
|
||
disabled={isExportingExcel}
|
||
className="bg-gradient-to-r from-blue-600 to-indigo-600 hover:from-blue-700 hover:to-indigo-700 text-white shadow-lg shadow-blue-500/25"
|
||
>
|
||
{isExportingExcel ? (
|
||
<RefreshCw className="w-4 h-4 mr-2 animate-spin" />
|
||
) : (
|
||
<Download className="w-4 h-4 mr-2" />
|
||
)}
|
||
匯出 Excel
|
||
</Button>
|
||
</div>
|
||
</CardContent>
|
||
</Card>
|
||
|
||
{/* 數據表格 */}
|
||
<Card className="bg-slate-800/50 backdrop-blur-sm border-slate-700/50">
|
||
<CardHeader>
|
||
<CardTitle className="text-white flex items-center gap-2">
|
||
<FileText className="w-5 h-5 text-cyan-400" />
|
||
困擾案例列表
|
||
</CardTitle>
|
||
<CardDescription className="text-blue-200">
|
||
共 {filteredWishes.length} 筆數據
|
||
</CardDescription>
|
||
</CardHeader>
|
||
<CardContent>
|
||
<div className="overflow-x-auto">
|
||
<table className="w-full text-sm">
|
||
<thead>
|
||
<tr className="border-b border-slate-600/50">
|
||
<th className="text-left py-3 px-4 text-blue-200 font-medium">ID</th>
|
||
<th className="text-left py-3 px-4 text-blue-200 font-medium">標題</th>
|
||
<th className="text-left py-3 px-4 text-blue-200 font-medium">狀態</th>
|
||
<th className="text-left py-3 px-4 text-blue-200 font-medium">可見性</th>
|
||
<th className="text-left py-3 px-4 text-blue-200 font-medium">點讚數</th>
|
||
<th className="text-left py-3 px-4 text-blue-200 font-medium">創建時間</th>
|
||
<th className="text-left py-3 px-4 text-blue-200 font-medium">操作</th>
|
||
</tr>
|
||
</thead>
|
||
<tbody>
|
||
{currentWishes.map((wish) => (
|
||
<tr key={wish.id} className="border-b border-slate-700/30 hover:bg-slate-700/30 transition-colors">
|
||
<td className="py-3 px-4 text-white font-mono text-xs">{wish.id}</td>
|
||
<td className="py-3 px-4 text-white max-w-xs truncate" title={wish.title}>
|
||
{wish.title}
|
||
</td>
|
||
<td className="py-3 px-4">
|
||
<Badge
|
||
variant={wish.status === 'active' ? 'default' : 'secondary'}
|
||
className={wish.status === 'active'
|
||
? 'bg-green-600/20 text-green-300 border-green-500/30'
|
||
: 'bg-slate-600/20 text-slate-300 border-slate-500/30'
|
||
}
|
||
>
|
||
{wish.status === 'active' ? '活躍' : '非活躍'}
|
||
</Badge>
|
||
</td>
|
||
<td className="py-3 px-4">
|
||
<Badge
|
||
variant={wish.is_public ? 'default' : 'outline'}
|
||
className={wish.is_public
|
||
? 'bg-blue-600/20 text-blue-300 border-blue-500/30'
|
||
: 'bg-orange-600/20 text-orange-300 border-orange-500/30'
|
||
}
|
||
>
|
||
{wish.is_public ? '公開' : '私密'}
|
||
</Badge>
|
||
</td>
|
||
<td className="py-3 px-4 text-white flex items-center gap-1">
|
||
<Heart className="w-3 h-3 text-pink-400" />
|
||
{wish.like_count}
|
||
</td>
|
||
<td className="py-3 px-4 text-blue-300 text-xs">
|
||
{new Date(wish.created_at).toLocaleDateString('zh-TW')}
|
||
</td>
|
||
<td className="py-3 px-4">
|
||
<Button
|
||
size="sm"
|
||
variant="outline"
|
||
onClick={() => viewWishDetails(wish)}
|
||
disabled={loadingDetails}
|
||
className="text-blue-200 border-slate-600/50 hover:bg-slate-700/50 hover:text-white hover:border-cyan-400/50"
|
||
>
|
||
{loadingDetails ? (
|
||
<RefreshCw className="w-4 h-4 mr-1 animate-spin" />
|
||
) : (
|
||
<Eye className="w-4 h-4 mr-1" />
|
||
)}
|
||
查看
|
||
</Button>
|
||
</td>
|
||
</tr>
|
||
))}
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
|
||
{/* 分頁組件 */}
|
||
<PaginationComponent />
|
||
</CardContent>
|
||
</Card>
|
||
</TabsContent>
|
||
|
||
<TabsContent value="analytics" className="space-y-6">
|
||
{/* 隱私說明卡片 */}
|
||
<Card className="bg-gradient-to-r from-blue-900/80 to-indigo-800/80 backdrop-blur-sm border border-blue-500/50">
|
||
<CardHeader className="pb-3 md:pb-4">
|
||
<div className="flex items-center justify-between">
|
||
<div className="flex items-center gap-2 md:gap-3 min-w-0 flex-1">
|
||
<div className="w-6 h-6 md:w-8 md:h-8 bg-gradient-to-br from-indigo-600 to-purple-700 rounded-full flex items-center justify-center flex-shrink-0">
|
||
<Shield className="w-3 h-3 md:w-4 md:h-4 text-white" />
|
||
</div>
|
||
<div className="min-w-0 flex-1">
|
||
<CardTitle className="text-base md:text-xl lg:text-2xl text-white truncate">數據隱私說明</CardTitle>
|
||
<CardDescription className="text-white/90 text-xs md:text-sm lg:text-base">
|
||
本分析包含所有提交的案例,包括選擇保持私密的困擾
|
||
</CardDescription>
|
||
</div>
|
||
</div>
|
||
<Button
|
||
variant="ghost"
|
||
size="sm"
|
||
onClick={() => setShowPrivacyDetails(!showPrivacyDetails)}
|
||
className="text-indigo-200 hover:text-white hover:bg-indigo-700/50 px-2 md:px-3 flex-shrink-0 md:hidden"
|
||
>
|
||
{showPrivacyDetails ? <ChevronUp className="w-4 h-4" /> : <ChevronDown className="w-4 h-4" />}
|
||
</Button>
|
||
</div>
|
||
</CardHeader>
|
||
<div className={`${showPrivacyDetails ? "block" : "hidden"} md:block`}>
|
||
<CardContent className="pt-0">
|
||
<div className="grid sm:grid-cols-2 gap-3 md:gap-4 text-sm">
|
||
<div className="space-y-2">
|
||
<h4 className="font-semibold text-indigo-200 flex items-center gap-2">
|
||
<Eye className="w-3 h-3 md:w-4 md:h-4" />
|
||
公開案例 ({stats?.publicWishes || 0} 個)
|
||
</h4>
|
||
<p className="text-indigo-100 text-xs md:text-sm leading-relaxed">
|
||
這些案例會顯示在「聆聽心聲」頁面,供其他人查看和產生共鳴
|
||
</p>
|
||
</div>
|
||
<div className="space-y-2">
|
||
<h4 className="font-semibold text-indigo-200 flex items-center gap-2">
|
||
<EyeOff className="w-3 h-3 md:w-4 md:h-4" />
|
||
私密案例 ({stats?.privateWishes || 0} 個)
|
||
</h4>
|
||
<p className="text-indigo-100 text-xs md:text-sm leading-relaxed">
|
||
這些案例保持匿名且私密,僅用於統計分析,幫助了解整體趨勢
|
||
</p>
|
||
</div>
|
||
</div>
|
||
</CardContent>
|
||
</div>
|
||
</Card>
|
||
|
||
|
||
{/* 分類指南 */}
|
||
<Card className="bg-gradient-to-r from-blue-900/80 to-indigo-800/80 backdrop-blur-sm border border-blue-500/50">
|
||
<CardHeader>
|
||
<div className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-3">
|
||
<div className="flex items-center gap-2 md:gap-3 min-w-0 flex-1">
|
||
<div className="w-6 h-6 md:w-8 md:h-8 bg-gradient-to-br from-indigo-600 to-purple-700 rounded-full flex items-center justify-center flex-shrink-0">
|
||
<BookOpen className="w-3 h-3 md:w-4 md:h-4 text-white" />
|
||
</div>
|
||
<div className="min-w-0 flex-1">
|
||
<CardTitle className="text-base md:text-xl lg:text-2xl text-white">問題分類說明</CardTitle>
|
||
<CardDescription className="text-white/90 text-xs md:text-sm lg:text-base">
|
||
了解我們如何分類和分析各種職場困擾
|
||
</CardDescription>
|
||
</div>
|
||
</div>
|
||
<Button
|
||
variant="ghost"
|
||
size="sm"
|
||
onClick={() => setShowCategoryGuide(!showCategoryGuide)}
|
||
className="text-indigo-200 hover:text-white hover:bg-indigo-800/50 self-start sm:self-auto flex-shrink-0"
|
||
>
|
||
{showCategoryGuide ? (
|
||
<>
|
||
<ChevronUp className="w-4 h-4 mr-1" />
|
||
收起
|
||
</>
|
||
) : (
|
||
<>
|
||
<ChevronDown className="w-4 h-4 mr-1" />
|
||
展開
|
||
</>
|
||
)}
|
||
</Button>
|
||
</div>
|
||
</CardHeader>
|
||
|
||
{showCategoryGuide && (
|
||
<CardContent>
|
||
<div className="grid md:grid-cols-2 gap-3 md:gap-4">
|
||
{categories.map((category, index) => (
|
||
<div
|
||
key={category.name}
|
||
className="p-3 md:p-4 rounded-lg bg-slate-800/50 border border-slate-600/30 hover:bg-slate-700/60 transition-all duration-200"
|
||
>
|
||
<div className="flex items-start gap-2 md:gap-3 mb-2">
|
||
<div className="text-lg md:text-2xl">{category.icon}</div>
|
||
<div className="flex-1 min-w-0">
|
||
<div className="flex items-center gap-2 mb-1">
|
||
<h4 className="font-semibold text-white text-sm md:text-base">{category.name}</h4>
|
||
<div
|
||
className="w-2 h-2 md:w-3 md:h-3 rounded-full flex-shrink-0"
|
||
style={{ backgroundColor: category.color }}
|
||
></div>
|
||
</div>
|
||
<p className="text-xs md:text-sm text-slate-300 leading-relaxed">{category.description}</p>
|
||
</div>
|
||
</div>
|
||
<div className="mt-2 md:mt-3 pt-2 md:pt-3 border-t border-slate-600/30">
|
||
<div className="text-xs text-slate-400 mb-2">常見關鍵字:</div>
|
||
<div className="flex flex-wrap gap-1">
|
||
{category.keywords.slice(0, 6).map((keyword, idx) => (
|
||
<Badge
|
||
key={idx}
|
||
variant="secondary"
|
||
className="text-xs px-1.5 md:px-2 py-0.5 bg-slate-600/50 text-slate-300 border-slate-500/50"
|
||
>
|
||
{keyword}
|
||
</Badge>
|
||
))}
|
||
{category.keywords.length > 6 && (
|
||
<Badge
|
||
variant="secondary"
|
||
className="text-xs px-1.5 md:px-2 py-0.5 bg-slate-600/30 text-slate-400 border-slate-500/30"
|
||
>
|
||
+{category.keywords.length - 6}
|
||
</Badge>
|
||
)}
|
||
</div>
|
||
</div>
|
||
</div>
|
||
))}
|
||
</div>
|
||
</CardContent>
|
||
)}
|
||
</Card>
|
||
|
||
{/* 手機版:垂直佈局,桌面版:並排佈局 */}
|
||
<div className="space-y-6 md:space-y-0 lg:grid lg:grid-cols-2 lg:gap-8 md:gap-12">
|
||
{/* 雷達圖 */}
|
||
<Card className="bg-slate-800/50 backdrop-blur-sm border border-slate-600/50">
|
||
<CardHeader>
|
||
<CardTitle className="text-lg md:text-xl lg:text-2xl text-white flex items-center gap-2 md:gap-3">
|
||
<div className="w-6 h-6 md:w-8 md:h-8 bg-gradient-to-br from-purple-600 to-indigo-700 rounded-full flex items-center justify-center">
|
||
<BarChart3 className="w-3 h-3 md:w-4 md:h-4 text-white" />
|
||
</div>
|
||
問題分布圖譜
|
||
</CardTitle>
|
||
<CardDescription className="text-white/90 text-xs md:text-sm">
|
||
各類職場困擾的完整案例分布(包含私密數據)
|
||
</CardDescription>
|
||
</CardHeader>
|
||
<CardContent>
|
||
<div className="h-64 sm:h-80 md:h-64 lg:h-80 xl:h-96">
|
||
{stats?.categoryDetails && <RadarChart data={stats.categoryDetails} />}
|
||
</div>
|
||
</CardContent>
|
||
</Card>
|
||
|
||
{/* 分類詳細統計 */}
|
||
<Card className="bg-slate-800/50 backdrop-blur-sm border border-slate-600/50">
|
||
<CardHeader>
|
||
<CardTitle className="text-lg md:text-xl lg:text-2xl text-white flex items-center gap-2 md:gap-3">
|
||
<div className="w-6 h-6 md:w-8 md:h-8 bg-gradient-to-br from-cyan-600 to-blue-700 rounded-full flex items-center justify-center">
|
||
<Target className="w-3 h-3 md:w-4 md:h-4 text-white" />
|
||
</div>
|
||
完整案例統計
|
||
<Badge className="bg-gradient-to-r from-pink-700/60 to-purple-700/60 text-white border border-pink-400/50 text-xs px-2 py-1">
|
||
含私密數據
|
||
</Badge>
|
||
</CardTitle>
|
||
<CardDescription className="text-white/90 text-xs md:text-sm">
|
||
每個領域的所有案例數量(包含公開和私密案例)
|
||
{stats?.categoryDetails?.filter((cat) => cat.count > 0).length && (
|
||
<span className="block text-xs text-slate-400 mt-1">
|
||
共 {stats.categoryDetails.filter((cat) => cat.count > 0).length} 個活躍分類
|
||
{stats.categoryDetails.filter((cat) => cat.count > 0).length > 4 && ",可滾動查看全部"}
|
||
</span>
|
||
)}
|
||
</CardDescription>
|
||
</CardHeader>
|
||
<CardContent className="space-y-3 md:space-y-4">
|
||
<div className="max-h-64 md:max-h-80 overflow-y-auto pr-2 space-y-3 md:space-y-4 scrollbar-thin scrollbar-thumb-slate-600 scrollbar-track-slate-800">
|
||
{stats?.categoryDetails
|
||
?.filter((cat) => cat.count > 0)
|
||
.sort((a, b) => b.count - a.count)
|
||
.map((category, index) => (
|
||
<div
|
||
key={category.name}
|
||
className="flex items-center justify-between p-3 md:p-4 rounded-lg bg-slate-700/30 border border-slate-600/30 hover:bg-slate-600/40 transition-all duration-200"
|
||
>
|
||
<div className="flex items-center gap-2 md:gap-3 min-w-0 flex-1">
|
||
<div className="text-base md:text-xl">
|
||
{categories.find((cat) => cat.name === category.name)?.icon || "❓"}
|
||
</div>
|
||
<div className="min-w-0 flex-1">
|
||
<div className="font-semibold text-white flex items-center gap-2 mb-1">
|
||
<span className="truncate">{category.name}</span>
|
||
<div
|
||
className="w-2 h-2 md:w-3 md:h-3 rounded-full flex-shrink-0"
|
||
style={{ backgroundColor: category.color }}
|
||
></div>
|
||
{index < 3 && (
|
||
<span className="text-xs bg-gradient-to-r from-cyan-500/20 to-blue-500/20 text-cyan-200 px-1.5 md:px-2 py-0.5 rounded-full border border-cyan-500/30 flex-shrink-0">
|
||
TOP {index + 1}
|
||
</span>
|
||
)}
|
||
</div>
|
||
<div className="text-xs md:text-sm text-slate-300">{category.count} 個案例</div>
|
||
{category.description && (
|
||
<div className="text-xs text-slate-400 mt-1 line-clamp-2">{category.description}</div>
|
||
)}
|
||
</div>
|
||
</div>
|
||
<Badge variant="secondary" className="bg-slate-600/50 text-slate-200 flex-shrink-0 ml-2">
|
||
{category.percentage}%
|
||
</Badge>
|
||
</div>
|
||
))}
|
||
</div>
|
||
|
||
{/* 滾動提示 */}
|
||
{stats?.categoryDetails?.filter((cat) => cat.count > 0).length && stats.categoryDetails.filter((cat) => cat.count > 0).length > 4 && (
|
||
<div className="text-center pt-2 border-t border-slate-600/30">
|
||
<div className="text-xs text-slate-400 flex items-center justify-center gap-2">
|
||
<div className="w-1 h-1 bg-slate-400 rounded-full animate-bounce"></div>
|
||
<span>向下滾動查看更多分類</span>
|
||
<div
|
||
className="w-1 h-1 bg-slate-400 rounded-full animate-bounce"
|
||
style={{ animationDelay: "0.5s" }}
|
||
></div>
|
||
</div>
|
||
</div>
|
||
)}
|
||
</CardContent>
|
||
</Card>
|
||
</div>
|
||
|
||
{/* 熱門關鍵字 */}
|
||
{stats?.topKeywords && stats.topKeywords.length > 0 && (
|
||
<Card className="bg-slate-800/50 backdrop-blur-sm border border-slate-600/50">
|
||
<CardHeader>
|
||
<CardTitle className="text-lg md:text-xl lg:text-2xl text-white flex items-center gap-2 md:gap-3">
|
||
<div className="w-6 h-6 md:w-8 md:h-8 bg-gradient-to-br from-green-400 to-emerald-500 rounded-full flex items-center justify-center">
|
||
<TrendingUp className="w-3 h-3 md:w-4 md:h-4 text-white" />
|
||
</div>
|
||
最常見的問題關鍵字
|
||
</CardTitle>
|
||
<CardDescription className="text-white/90 text-xs md:text-sm">
|
||
在所有案例中最常出現的詞彙,反映團隊面臨的核心挑戰
|
||
</CardDescription>
|
||
</CardHeader>
|
||
<CardContent>
|
||
<div className="flex flex-wrap gap-2 sm:gap-3">
|
||
{stats.topKeywords.map((keyword, index) => (
|
||
<Badge
|
||
key={keyword.word}
|
||
variant="secondary"
|
||
className="bg-gradient-to-r from-cyan-500/20 to-blue-500/20 text-cyan-200 border border-cyan-500/30 px-2 md:px-3 py-1 md:py-1.5 text-xs md:text-sm"
|
||
>
|
||
{keyword.word} ({keyword.count})
|
||
</Badge>
|
||
))}
|
||
</div>
|
||
</CardContent>
|
||
</Card>
|
||
)}
|
||
</TabsContent>
|
||
</Tabs>
|
||
</div>
|
||
</main>
|
||
|
||
{/* 詳細資訊對話框 */}
|
||
<Dialog open={showWishDetails} onOpenChange={setShowWishDetails}>
|
||
<DialogContent className="w-[95vw] max-w-6xl h-[85vh] bg-slate-800 border-slate-700 flex flex-col p-0 z-[99999] mt-16 mb-8">
|
||
<DialogHeader className="px-6 py-4 border-b border-slate-600/50 flex-shrink-0">
|
||
<DialogTitle className="text-white flex items-center gap-2">
|
||
<Eye className="w-5 h-5 text-cyan-400" />
|
||
困擾案例詳細資訊
|
||
</DialogTitle>
|
||
<DialogDescription className="text-blue-200">
|
||
查看困擾案例的完整資訊,包括點讚記錄
|
||
</DialogDescription>
|
||
</DialogHeader>
|
||
|
||
<div className="flex-1 overflow-y-auto px-6 py-4">
|
||
{wishDetails && (
|
||
<div className="space-y-6">
|
||
{/* 基本信息 */}
|
||
<Card className="bg-slate-700/50 border-slate-600/50">
|
||
<CardHeader>
|
||
<CardTitle className="text-white flex items-center gap-2">
|
||
<FileText className="w-5 h-5 text-green-400" />
|
||
基本信息
|
||
</CardTitle>
|
||
</CardHeader>
|
||
<CardContent className="space-y-4">
|
||
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
|
||
<div className="flex items-center gap-2">
|
||
<Tag className="w-4 h-4 text-blue-400" />
|
||
<span className="text-blue-200">ID:</span>
|
||
<span className="text-white font-mono">{wishDetails.id}</span>
|
||
</div>
|
||
<div className="flex items-center gap-2">
|
||
<Star className="w-4 h-4 text-yellow-400" />
|
||
<span className="text-blue-200">優先級:</span>
|
||
<Badge variant="outline" className="text-white border-slate-500">
|
||
{wishDetails.priority}
|
||
</Badge>
|
||
</div>
|
||
<div className="flex items-center gap-2">
|
||
<Eye className="w-4 h-4 text-cyan-400" />
|
||
<span className="text-blue-200">可見性:</span>
|
||
<Badge
|
||
variant={wishDetails.isPublic ? "default" : "outline"}
|
||
className={wishDetails.isPublic
|
||
? "bg-blue-600/20 text-blue-300 border-blue-500/30"
|
||
: "bg-orange-600/20 text-orange-300 border-orange-500/30"
|
||
}
|
||
>
|
||
{wishDetails.isPublic ? '公開' : '私密'}
|
||
</Badge>
|
||
</div>
|
||
<div className="flex items-center gap-2">
|
||
<Target className="w-4 h-4 text-green-400" />
|
||
<span className="text-blue-200">狀態:</span>
|
||
<Badge
|
||
variant={wishDetails.status === 'active' ? "default" : "secondary"}
|
||
className={wishDetails.status === 'active'
|
||
? "bg-green-600/20 text-green-300 border-green-500/30"
|
||
: "bg-slate-600/20 text-slate-300 border-slate-500/30"
|
||
}
|
||
>
|
||
{wishDetails.status === 'active' ? '活躍' : '非活躍'}
|
||
</Badge>
|
||
</div>
|
||
</div>
|
||
|
||
<div>
|
||
<h4 className="text-blue-200 font-semibold mb-2">標題</h4>
|
||
<p className="text-white bg-slate-800/50 p-3 rounded-lg border border-slate-600/30">
|
||
{wishDetails.title}
|
||
</p>
|
||
</div>
|
||
|
||
{wishDetails.category && (
|
||
<div>
|
||
<h4 className="text-blue-200 font-semibold mb-2">分類</h4>
|
||
<Badge variant="outline" className="text-white border-slate-500">
|
||
{wishDetails.category}
|
||
</Badge>
|
||
</div>
|
||
)}
|
||
</CardContent>
|
||
</Card>
|
||
|
||
{/* 困擾內容 */}
|
||
<Card className="bg-slate-700/50 border-slate-600/50">
|
||
<CardHeader>
|
||
<CardTitle className="text-white flex items-center gap-2">
|
||
<Users className="w-5 h-5 text-red-400" />
|
||
遇到的困擾
|
||
</CardTitle>
|
||
</CardHeader>
|
||
<CardContent>
|
||
<p className="text-white bg-slate-800/50 p-4 rounded-lg border border-slate-600/30 whitespace-pre-wrap">
|
||
{wishDetails.currentPain}
|
||
</p>
|
||
</CardContent>
|
||
</Card>
|
||
|
||
{/* 期望解決方式 */}
|
||
<Card className="bg-slate-700/50 border-slate-600/50">
|
||
<CardHeader>
|
||
<CardTitle className="text-white flex items-center gap-2">
|
||
<Target className="w-5 h-5 text-green-400" />
|
||
期望的解決方式
|
||
</CardTitle>
|
||
</CardHeader>
|
||
<CardContent>
|
||
<p className="text-white bg-slate-800/50 p-4 rounded-lg border border-slate-600/30 whitespace-pre-wrap">
|
||
{wishDetails.expectedSolution}
|
||
</p>
|
||
</CardContent>
|
||
</Card>
|
||
|
||
{/* 預期效果 */}
|
||
{wishDetails.expectedEffect && (
|
||
<Card className="bg-slate-700/50 border-slate-600/50">
|
||
<CardHeader>
|
||
<CardTitle className="text-white flex items-center gap-2">
|
||
<Star className="w-5 h-5 text-yellow-400" />
|
||
預期效果
|
||
</CardTitle>
|
||
</CardHeader>
|
||
<CardContent>
|
||
<p className="text-white bg-slate-800/50 p-4 rounded-lg border border-slate-600/30 whitespace-pre-wrap">
|
||
{wishDetails.expectedEffect}
|
||
</p>
|
||
</CardContent>
|
||
</Card>
|
||
)}
|
||
|
||
{/* 圖片 */}
|
||
{wishDetails.images && Array.isArray(wishDetails.images) && wishDetails.images.length > 0 && (
|
||
<Card className="bg-slate-700/50 border-slate-600/50">
|
||
<CardHeader>
|
||
<CardTitle className="text-white flex items-center gap-2">
|
||
<ImageIcon className="w-5 h-5 text-purple-400" />
|
||
相關圖片
|
||
</CardTitle>
|
||
</CardHeader>
|
||
<CardContent>
|
||
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
|
||
{wishDetails.images.map((image: any, index: number) => (
|
||
<div key={index} className="bg-slate-800/50 rounded-lg border border-slate-600/30 overflow-hidden">
|
||
{/* 圖片顯示 */}
|
||
<div className="aspect-video bg-slate-900/50 flex items-center justify-center relative group cursor-pointer" onClick={() => viewImage(image)}>
|
||
{(image.url || image.base64 || image.storage_path || image.public_url) ? (
|
||
<>
|
||
<img
|
||
src={image.public_url || image.url || image.base64 || image.storage_path}
|
||
alt={image.name || `圖片 ${index + 1}`}
|
||
className="max-w-full max-h-full object-contain rounded-lg transition-transform group-hover:scale-105"
|
||
onLoad={(e) => {
|
||
const target = e.target as HTMLImageElement;
|
||
target.nextElementSibling?.classList.add('hidden');
|
||
}}
|
||
onError={(e) => {
|
||
const target = e.target as HTMLImageElement;
|
||
target.style.display = 'none';
|
||
target.nextElementSibling?.classList.remove('hidden');
|
||
}}
|
||
/>
|
||
{/* 載入中指示器 */}
|
||
<div className="absolute inset-0 flex flex-col items-center justify-center text-slate-400 p-8">
|
||
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-cyan-400 mb-2"></div>
|
||
<span className="text-sm">載入中...</span>
|
||
</div>
|
||
{/* 放大圖標 */}
|
||
<div className="absolute top-2 right-2 bg-black/50 backdrop-blur-sm rounded-full p-2 opacity-0 group-hover:opacity-100 transition-opacity">
|
||
<svg className="w-4 h-4 text-white" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
||
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0zM10 7v3m0 0v3m0-3h3m-3 0H7" />
|
||
</svg>
|
||
</div>
|
||
</>
|
||
) : (
|
||
<div className="flex flex-col items-center justify-center text-slate-400 p-8">
|
||
<ImageIcon className="w-12 h-12 mb-2 text-slate-500" />
|
||
<span className="text-sm">無圖片數據</span>
|
||
</div>
|
||
)}
|
||
</div>
|
||
{/* 圖片資訊 */}
|
||
<div className="p-3 border-t border-slate-600/30">
|
||
<div className="flex items-center justify-between">
|
||
<div className="text-sm text-white font-medium truncate">
|
||
{image.name || `圖片 ${index + 1}`}
|
||
</div>
|
||
<div className="text-xs text-cyan-400 opacity-0 group-hover:opacity-100 transition-opacity">
|
||
點擊放大
|
||
</div>
|
||
</div>
|
||
{image.size && (
|
||
<div className="text-xs text-slate-400 mt-1">
|
||
大小: {typeof image.size === 'number' ? `${(image.size / 1024).toFixed(1)} KB` : image.size}
|
||
</div>
|
||
)}
|
||
{image.type && (
|
||
<div className="text-xs text-slate-400">
|
||
類型: {image.type}
|
||
</div>
|
||
)}
|
||
</div>
|
||
</div>
|
||
))}
|
||
</div>
|
||
</CardContent>
|
||
</Card>
|
||
)}
|
||
|
||
{/* 用戶資訊 */}
|
||
<Card className="bg-slate-700/50 border-slate-600/50">
|
||
<CardHeader>
|
||
<CardTitle className="text-white flex items-center gap-2">
|
||
<User className="w-5 h-5 text-cyan-400" />
|
||
用戶資訊
|
||
</CardTitle>
|
||
</CardHeader>
|
||
<CardContent className="space-y-3">
|
||
<div className="flex items-center gap-2">
|
||
<User className="w-4 h-4 text-blue-400" />
|
||
<span className="text-blue-200">會話 ID:</span>
|
||
<span className="text-white font-mono text-sm">{wishDetails.userSession}</span>
|
||
</div>
|
||
{wishDetails.email && (
|
||
<div className="flex items-center gap-2">
|
||
<Mail className="w-4 h-4 text-green-400" />
|
||
<span className="text-blue-200">電子郵件:</span>
|
||
<span className="text-white">{wishDetails.email}</span>
|
||
</div>
|
||
)}
|
||
</CardContent>
|
||
</Card>
|
||
|
||
{/* 時間資訊 */}
|
||
<Card className="bg-slate-700/50 border-slate-600/50">
|
||
<CardHeader>
|
||
<CardTitle className="text-white flex items-center gap-2">
|
||
<Clock className="w-5 h-5 text-orange-400" />
|
||
時間資訊
|
||
</CardTitle>
|
||
</CardHeader>
|
||
<CardContent className="space-y-3">
|
||
<div className="flex items-center gap-2">
|
||
<Calendar className="w-4 h-4 text-blue-400" />
|
||
<span className="text-blue-200">創建時間:</span>
|
||
<span className="text-white">
|
||
{new Date(wishDetails.createdAt).toLocaleString('zh-TW')}
|
||
</span>
|
||
</div>
|
||
<div className="flex items-center gap-2">
|
||
<Clock className="w-4 h-4 text-green-400" />
|
||
<span className="text-blue-200">更新時間:</span>
|
||
<span className="text-white">
|
||
{new Date(wishDetails.updatedAt).toLocaleString('zh-TW')}
|
||
</span>
|
||
</div>
|
||
</CardContent>
|
||
</Card>
|
||
|
||
{/* 點讚記錄 */}
|
||
<Card className="bg-slate-700/50 border-slate-600/50">
|
||
<CardHeader>
|
||
<CardTitle className="text-white flex items-center gap-2">
|
||
<Heart className="w-5 h-5 text-pink-400" />
|
||
點讚記錄 ({wishDetails.likes?.length || 0} 個)
|
||
</CardTitle>
|
||
</CardHeader>
|
||
<CardContent>
|
||
{wishDetails.likes && wishDetails.likes.length > 0 ? (
|
||
<div className="space-y-3 max-h-48 overflow-y-auto scrollbar-thin scrollbar-thumb-slate-600 scrollbar-track-slate-800">
|
||
{wishDetails.likes.map((like: any, index: number) => (
|
||
<div key={like.id} className="bg-slate-800/50 p-3 rounded-lg border border-slate-600/30">
|
||
<div className="flex items-center justify-between mb-2">
|
||
<div className="flex items-center gap-2">
|
||
<Heart className="w-4 h-4 text-pink-400" />
|
||
<span className="text-blue-200">點讚 #{index + 1}</span>
|
||
</div>
|
||
<span className="text-slate-400 text-sm">
|
||
{new Date(like.createdAt).toLocaleString('zh-TW')}
|
||
</span>
|
||
</div>
|
||
<div className="grid grid-cols-1 md:grid-cols-2 gap-2 text-sm">
|
||
<div className="flex items-center gap-2">
|
||
<User className="w-3 h-3 text-blue-400" />
|
||
<span className="text-blue-200">會話:</span>
|
||
<span className="text-white font-mono text-xs">{like.userSession}</span>
|
||
</div>
|
||
{like.ipAddress && (
|
||
<div className="flex items-center gap-2">
|
||
<MapPin className="w-3 h-3 text-green-400" />
|
||
<span className="text-blue-200">IP:</span>
|
||
<span className="text-white font-mono text-xs">{like.ipAddress}</span>
|
||
</div>
|
||
)}
|
||
</div>
|
||
{like.userAgent && (
|
||
<div className="mt-2 flex items-start gap-2">
|
||
<Monitor className="w-3 h-3 text-purple-400 mt-0.5" />
|
||
<span className="text-blue-200 text-xs">瀏覽器:</span>
|
||
<span className="text-slate-300 text-xs break-all">{like.userAgent}</span>
|
||
</div>
|
||
)}
|
||
</div>
|
||
))}
|
||
</div>
|
||
) : (
|
||
<div className="text-center py-8">
|
||
<Heart className="w-8 h-8 text-slate-500 mx-auto mb-2" />
|
||
<p className="text-slate-400">暫無點讚記錄</p>
|
||
</div>
|
||
)}
|
||
</CardContent>
|
||
</Card>
|
||
</div>
|
||
)}
|
||
</div>
|
||
|
||
<div className="flex justify-end px-6 py-4 border-t border-slate-600/50 flex-shrink-0">
|
||
<Button
|
||
onClick={() => setShowWishDetails(false)}
|
||
className="bg-slate-700 hover:bg-slate-600 text-white"
|
||
>
|
||
<X className="w-4 h-4 mr-2" />
|
||
關閉
|
||
</Button>
|
||
</div>
|
||
</DialogContent>
|
||
</Dialog>
|
||
|
||
{/* 圖片放大對話框 */}
|
||
<Dialog open={showImageModal} onOpenChange={setShowImageModal}>
|
||
<DialogContent className="max-w-[95vw] max-h-[85vh] bg-black border-slate-700 p-0 overflow-hidden z-[100000] mt-16 mb-8">
|
||
<DialogHeader className="px-6 py-4 border-b border-slate-600/50 flex-shrink-0">
|
||
<DialogTitle className="text-white flex items-center gap-2">
|
||
<ImageIcon className="w-5 h-5 text-purple-400" />
|
||
{selectedImage?.name || '圖片預覽'}
|
||
</DialogTitle>
|
||
<DialogDescription className="text-blue-200">
|
||
點擊圖片外部或按 ESC 鍵關閉
|
||
</DialogDescription>
|
||
</DialogHeader>
|
||
|
||
<div className="flex-1 flex items-center justify-center p-4 bg-black relative">
|
||
{selectedImage && (
|
||
<>
|
||
<img
|
||
src={selectedImage.public_url || selectedImage.url || selectedImage.base64 || selectedImage.storage_path}
|
||
alt={selectedImage.name || '圖片預覽'}
|
||
className="max-w-full max-h-full object-contain"
|
||
onClick={(e) => e.stopPropagation()}
|
||
/>
|
||
{/* 圖片載入失敗處理 */}
|
||
<div className="hidden flex-col items-center justify-center text-slate-400">
|
||
<ImageIcon className="w-16 h-16 mb-4 text-slate-500" />
|
||
<span className="text-lg">圖片載入失敗</span>
|
||
</div>
|
||
</>
|
||
)}
|
||
</div>
|
||
|
||
<div className="flex justify-between items-center px-6 py-4 border-t border-slate-600/50 flex-shrink-0 bg-slate-800/50">
|
||
<div className="text-sm text-slate-300">
|
||
{selectedImage?.size && (
|
||
<span>大小: {typeof selectedImage.size === 'number' ? `${(selectedImage.size / 1024).toFixed(1)} KB` : selectedImage.size}</span>
|
||
)}
|
||
{selectedImage?.type && (
|
||
<span className="ml-4">類型: {selectedImage.type}</span>
|
||
)}
|
||
</div>
|
||
<Button
|
||
onClick={() => setShowImageModal(false)}
|
||
className="bg-slate-700 hover:bg-slate-600 text-white"
|
||
>
|
||
<X className="w-4 h-4 mr-2" />
|
||
關閉
|
||
</Button>
|
||
</div>
|
||
</DialogContent>
|
||
</Dialog>
|
||
</div>
|
||
)
|
||
}
|