Files
cunkebao_v3/Cunkebao/app/scenarios/page.tsx

311 lines
10 KiB
TypeScript
Raw Normal View History

2025-03-29 16:50:39 +08:00
"use client"
2025-04-07 10:17:19 +08:00
import { useState, useEffect } from "react"
2025-03-29 16:50:39 +08:00
import type React from "react"
import { TrendingUp, Users, ChevronLeft, Bot, Sparkles, Plus, Phone } from "lucide-react"
import { Card } from "@/components/ui/card"
import { Button } from "@/components/ui/button"
import { useRouter } from "next/navigation"
2025-04-07 10:17:19 +08:00
import { Skeleton } from "@/components/ui/skeleton"
import { fetchScenes, transformSceneItem } from "@/api/scenarios"
2025-03-29 16:50:39 +08:00
interface Channel {
id: string
name: string
icon: string
stats: {
daily: number
growth: number
}
link?: string
plans?: Plan[]
}
interface Plan {
id: string
name: string
isNew?: boolean
status: "active" | "paused" | "completed"
acquisitionCount: number
}
2025-04-07 10:17:19 +08:00
// AI场景列表服务端暂未提供
2025-03-29 16:50:39 +08:00
const aiScenarios = [
{
id: "ai-friend",
name: "AI智能加友",
icon: "https://hebbkx1anhila5yf.public.blob.vercel-storage.com/image-azCH8EgGfidWXOqiM2D1jLH0VFRUtW.png",
description: "智能分析目标用户画像,自动筛选优质客户",
stats: {
daily: 245,
growth: 18.5,
},
link: "/scenarios/ai-friend",
2025-03-29 16:50:39 +08:00
plans: [
{
id: "ai-plan-1",
name: "AI智能筛选计划",
isNew: true,
status: "active",
acquisitionCount: 78,
},
],
},
{
id: "ai-group",
name: "AI群引流",
icon: "https://hebbkx1anhila5yf.public.blob.vercel-storage.com/image-azCH8EgGfidWXOqiM2D1jLH0VFRUtW.png",
description: "智能群聊互动,提高群活跃度和转化率",
stats: {
daily: 178,
growth: 15.2,
},
link: "/scenarios/ai-group",
2025-03-29 16:50:39 +08:00
plans: [
{
id: "ai-plan-2",
name: "AI群聊互动计划",
status: "active",
acquisitionCount: 56,
},
],
},
{
id: "ai-conversion",
name: "AI场景转化",
icon: "https://hebbkx1anhila5yf.public.blob.vercel-storage.com/image-m4ENUaZon82EPFHod2dP1dajlrRdVG.png",
description: "多场景智能营销,提升获客转化效果",
stats: {
daily: 134,
growth: 12.8,
},
link: "/scenarios/ai-conversion",
2025-03-29 16:50:39 +08:00
plans: [
{
id: "ai-plan-3",
name: "AI多场景营销",
isNew: true,
status: "active",
acquisitionCount: 43,
},
],
},
]
export default function ScenariosPage() {
const router = useRouter()
2025-04-07 10:17:19 +08:00
const [channels, setChannels] = useState<Channel[]>([])
const [loading, setLoading] = useState(true)
const [error, setError] = useState<string | null>(null)
useEffect(() => {
const loadScenes = async () => {
try {
setLoading(true)
const response = await fetchScenes({ limit: 50 })
if (response.code === 200 && response.data?.list) {
// 转换场景数据为前端展示格式
const transformedScenes = response.data.list.map((scene) => {
const transformedScene = transformSceneItem(scene)
// 添加link属性用于导航
return {
...transformedScene,
link: `/scenarios/${scene.id}`
}
})
setChannels(transformedScenes)
} else {
setError(response.msg || "获取场景列表失败")
}
} catch (err) {
console.error("Failed to fetch scenes:", err)
setError("获取场景列表失败")
} finally {
setLoading(false)
}
}
loadScenes()
}, [])
2025-03-29 16:50:39 +08:00
const handleChannelClick = (channelId: string, event: React.MouseEvent) => {
router.push(`/scenarios/${channelId}`)
}
const getStatusColor = (status: string) => {
switch (status) {
case "active":
return "bg-green-100 text-green-700"
case "paused":
return "bg-amber-100 text-amber-700"
case "completed":
return "bg-blue-100 text-blue-700"
default:
return "bg-gray-100 text-gray-700"
}
}
const getStatusText = (status: string) => {
switch (status) {
case "active":
return "执行中"
case "paused":
return "已暂停"
case "completed":
return "已完成"
default:
return "未知状态"
}
}
2025-04-07 10:17:19 +08:00
// 展示场景骨架屏
const renderSkeletons = () => {
return Array(8)
.fill(0)
.map((_, index) => (
<div key={`skeleton-${index}`} className="flex flex-col">
<Card className="p-4">
<div className="flex flex-col items-center text-center space-y-3">
<Skeleton className="w-12 h-12 rounded-xl" />
<Skeleton className="h-4 w-24" />
<div className="flex items-center space-x-1">
<Skeleton className="h-3 w-3 rounded-full" />
<Skeleton className="h-4 w-16" />
</div>
<Skeleton className="h-3 w-12" />
</div>
</Card>
</div>
))
}
2025-03-29 16:50:39 +08:00
return (
<div className="flex-1 bg-gray-50">
<div className="w-full mx-auto bg-white min-h-screen lg:max-w-7xl xl:max-w-[1200px]">
2025-03-29 16:50:39 +08:00
<header className="sticky top-0 z-10 bg-white border-b">
<div className="flex justify-between items-center p-4">
<div className="flex items-center space-x-3">
<Button variant="ghost" size="icon" onClick={() => window.history.back()}>
<ChevronLeft className="h-5 w-5" />
</Button>
<h1 className="text-xl font-semibold text-blue-600"></h1>
</div>
<Button className="bg-blue-600 hover:bg-blue-700" onClick={() => router.push("/plans/new")}>
<Plus className="h-4 w-4 mr-2" />
</Button>
</div>
</header>
<div className="p-4 space-y-6">
2025-04-07 10:17:19 +08:00
{/* 错误提示 */}
{error && (
<div className="bg-red-50 text-red-600 p-4 rounded-md">
<p>{error}</p>
<Button
variant="outline"
className="mt-2"
onClick={() => window.location.reload()}
>
</Button>
</div>
)}
2025-03-29 16:50:39 +08:00
{/* Traditional channels */}
<div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 lg:grid-cols-5 gap-4">
2025-04-07 10:17:19 +08:00
{loading ? (
renderSkeletons()
) : (
channels.map((channel) => (
<div key={channel.id} className="flex flex-col">
<Card
className={`p-4 hover:shadow-lg transition-all cursor-pointer`}
onClick={() => router.push(channel.link || "")}
>
<div className="flex flex-col items-center text-center space-y-3">
<div className="w-12 h-12 rounded-xl bg-white flex items-center justify-center shadow-sm">
2025-03-29 16:50:39 +08:00
<img
src={channel.icon || "/placeholder.svg"}
alt={channel.name}
className="w-8 h-8 object-contain"
2025-04-07 10:17:19 +08:00
onError={(e) => {
// 图片加载失败时,使用默认图标
const target = e.target as HTMLImageElement
target.src = "/assets/icons/poster-icon.svg"
}}
2025-03-29 16:50:39 +08:00
/>
2025-04-07 10:17:19 +08:00
</div>
2025-03-29 16:50:39 +08:00
2025-04-07 10:17:19 +08:00
<h3 className="text-sm font-medium text-blue-600">{channel.name}</h3>
2025-03-29 16:50:39 +08:00
2025-04-07 10:17:19 +08:00
<div className="flex items-center space-x-1">
<Users className="w-3 h-3 text-gray-400" />
<div className="flex items-baseline">
<span className="text-xs text-gray-500"></span>
<span className="text-base font-medium ml-1">{channel.stats.daily}</span>
</div>
2025-03-29 16:50:39 +08:00
</div>
2025-04-07 10:17:19 +08:00
<div className="flex items-center text-green-500 text-xs">
<TrendingUp className="w-3 h-3 mr-1" />
<span>{channel.stats.growth > 0 ? "+" : ""}{channel.stats.growth}%</span>
</div>
2025-03-29 16:50:39 +08:00
</div>
2025-04-07 10:17:19 +08:00
</Card>
</div>
))
)}
2025-03-29 16:50:39 +08:00
</div>
{/* AI scenarios */}
<div>
<div className="flex items-center space-x-2 mb-3">
<Bot className="w-5 h-5 text-blue-600" />
<h2 className="text-lg font-medium">AI智能获客</h2>
<span className="px-2 py-0.5 bg-blue-50 text-blue-600 text-xs rounded-full">Beta</span>
</div>
<div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 lg:grid-cols-5 gap-4">
2025-03-29 16:50:39 +08:00
{aiScenarios.map((scenario) => (
<div key={scenario.id} className="flex flex-col">
<Card
className={`p-4 hover:shadow-lg transition-all bg-gradient-to-br from-blue-50/50 to-white border-2 border-blue-100`}
onClick={() => router.push(scenario.link || "")}
>
<div className="flex flex-col items-center text-center space-y-3">
<div className="w-12 h-12 rounded-xl bg-blue-50 flex items-center justify-center shadow-sm">
<Sparkles className="w-6 h-6 text-blue-500" />
</div>
<h3 className="text-sm font-medium text-blue-600">{scenario.name}</h3>
<p className="text-xs text-gray-500 text-center line-clamp-2">{scenario.description}</p>
<div className="flex items-center space-x-1">
<Users className="w-3 h-3 text-gray-400" />
<div className="flex items-baseline">
<span className="text-xs text-gray-500"></span>
<span className="text-base font-medium ml-1">{scenario.stats.daily}</span>
</div>
</div>
<div className="flex items-center text-green-500 text-xs">
<TrendingUp className="w-3 h-3 mr-1" />
<span>+{scenario.stats.growth}%</span>
</div>
</div>
</Card>
</div>
))}
</div>
</div>
</div>
</div>
</div>
)
}