Files
cunkebao_v3/Cunkebao/app/content/page.tsx
2025-04-09 09:31:09 +08:00

215 lines
8.1 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 } from "react"
import { ChevronLeft, Filter, Search, RefreshCw, Plus, Edit, Trash2, Eye, MoreVertical } from "lucide-react"
import { Card } from "@/components/ui/card"
import { Button } from "@/components/ui/button"
import { Input } from "@/components/ui/input"
import { Tabs, TabsList, TabsTrigger } from "@/components/ui/tabs"
import { Badge } from "@/components/ui/badge"
import { useRouter } from "next/navigation"
import { DropdownMenu, DropdownMenuContent, DropdownMenuItem, DropdownMenuTrigger } from "@/components/ui/dropdown-menu"
import Image from "next/image"
interface ContentLibrary {
id: string
name: string
source: "friends" | "groups"
targetAudience: {
id: string
nickname: string
avatar: string
}[]
creator: string
itemCount: number
lastUpdated: string
enabled: boolean
}
export default function ContentLibraryPage() {
const router = useRouter()
const [libraries, setLibraries] = useState<ContentLibrary[]>([
{
id: "129",
name: "微信好友广告",
source: "friends",
targetAudience: [
{ id: "1", nickname: "张三", avatar: "/placeholder.svg?height=40&width=40" },
{ id: "2", nickname: "李四", avatar: "/placeholder.svg?height=40&width=40" },
{ id: "3", nickname: "王五", avatar: "/placeholder.svg?height=40&width=40" },
],
creator: "海尼",
itemCount: 0,
lastUpdated: "2024-02-09 12:30",
enabled: false,
},
{
id: "127",
name: "开发群",
source: "groups",
targetAudience: [{ id: "4", nickname: "开发群1", avatar: "/placeholder.svg?height=40&width=40" }],
creator: "karuo",
itemCount: 0,
lastUpdated: "2024-02-09 12:30",
enabled: true,
},
])
const [searchQuery, setSearchQuery] = useState("")
const [activeTab, setActiveTab] = useState("all")
const handleCreateNew = () => {
// 模拟创建新内容库
const newId = Date.now().toString()
const newLibrary = {
id: newId,
name: "新内容库",
source: "friends" as const,
targetAudience: [],
creator: "当前用户",
itemCount: 0,
lastUpdated: new Date().toISOString(),
enabled: true,
}
setLibraries([newLibrary, ...libraries])
router.push(`/content/${newId}`)
}
const handleEdit = (id: string) => {
router.push(`/content/${id}`)
}
const handleDelete = (id: string) => {
// 实现删除功能
setLibraries(libraries.filter((lib) => lib.id !== id))
}
const handleViewMaterials = (id: string) => {
router.push(`/content/${id}/materials`)
}
const filteredLibraries = libraries.filter(
(library) =>
(activeTab === "all" || library.source === activeTab) &&
(library.name.toLowerCase().includes(searchQuery.toLowerCase()) ||
library.targetAudience.some((target) => target.nickname.toLowerCase().includes(searchQuery.toLowerCase()))),
)
return (
<div className="flex-1 bg-gray-50 min-h-screen">
<header className="sticky top-0 z-10 bg-white border-b">
<div className="flex items-center justify-between p-4">
<div className="flex items-center space-x-3">
<Button variant="ghost" size="icon" onClick={() => router.back()}>
<ChevronLeft className="h-5 w-5" />
</Button>
<h1 className="text-lg font-medium"></h1>
</div>
<Button onClick={handleCreateNew}>
<Plus className="h-4 w-4 mr-2" />
</Button>
</div>
</header>
<div className="p-4 space-y-4">
<Card className="p-4">
<div className="space-y-4">
<div className="flex items-center space-x-2">
<div className="relative flex-1">
<Search className="absolute left-3 top-2.5 h-4 w-4 text-gray-400" />
<Input
placeholder="搜索内容库..."
value={searchQuery}
onChange={(e) => setSearchQuery(e.target.value)}
className="pl-9"
/>
</div>
<Button variant="outline" size="icon">
<Filter className="h-4 w-4" />
</Button>
<Button variant="outline" size="icon">
<RefreshCw className="h-4 w-4" />
</Button>
</div>
<Tabs defaultValue="all" value={activeTab} onValueChange={setActiveTab}>
<TabsList className="grid w-full grid-cols-3">
<TabsTrigger value="all"></TabsTrigger>
<TabsTrigger value="friends"></TabsTrigger>
<TabsTrigger value="groups"></TabsTrigger>
</TabsList>
</Tabs>
<div className="space-y-3">
{filteredLibraries.map((library) => (
<Card key={library.id} className="p-4">
<div className="flex items-start justify-between">
<div className="space-y-1">
<div className="flex items-center space-x-2">
<h3 className="font-medium">{library.name}</h3>
<Badge variant={library.enabled ? "success" : "secondary"}>
{library.enabled ? "已启用" : "已停用"}
</Badge>
</div>
<div className="text-sm text-gray-500">
<div className="flex items-center space-x-1">
<span></span>
<div className="flex -space-x-2 overflow-hidden">
{library.targetAudience.slice(0, 3).map((target) => (
<Image
key={target.id}
src={target.avatar || "/placeholder.svg"}
alt={target.nickname}
width={24}
height={24}
className="inline-block h-6 w-6 rounded-full ring-2 ring-white"
/>
))}
{library.targetAudience.length > 3 && (
<span className="flex items-center justify-center w-6 h-6 text-xs font-medium text-white bg-gray-400 rounded-full ring-2 ring-white">
+{library.targetAudience.length - 3}
</span>
)}
</div>
</div>
<div>{library.creator}</div>
<div>{library.itemCount}</div>
<div>{library.lastUpdated}</div>
</div>
</div>
<div className="flex space-x-2">
<DropdownMenu>
<DropdownMenuTrigger asChild>
<Button variant="ghost" size="sm">
<MoreVertical className="h-4 w-4" />
</Button>
</DropdownMenuTrigger>
<DropdownMenuContent align="end">
<DropdownMenuItem onClick={() => handleEdit(library.id)}>
<Edit className="h-4 w-4 mr-2" />
</DropdownMenuItem>
<DropdownMenuItem onClick={() => handleDelete(library.id)}>
<Trash2 className="h-4 w-4 mr-2" />
</DropdownMenuItem>
<DropdownMenuItem onClick={() => handleViewMaterials(library.id)}>
<Eye className="h-4 w-4 mr-2" />
</DropdownMenuItem>
</DropdownMenuContent>
</DropdownMenu>
</div>
</div>
</Card>
))}
</div>
</div>
</Card>
</div>
</div>
)
}