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

311 lines
10 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 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"
import { Skeleton } from "@/components/ui/skeleton"
import { fetchScenes, transformSceneItem } from "@/api/scenarios"
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
}
// AI场景列表服务端暂未提供
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",
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",
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",
plans: [
{
id: "ai-plan-3",
name: "AI多场景营销",
isNew: true,
status: "active",
acquisitionCount: 43,
},
],
},
]
export default function ScenariosPage() {
const router = useRouter()
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()
}, [])
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 "未知状态"
}
}
// 展示场景骨架屏
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>
))
}
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]">
<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">
{/* 错误提示 */}
{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>
)}
{/* Traditional channels */}
<div className="grid grid-cols-2 sm:grid-cols-3 md:grid-cols-4 lg:grid-cols-5 gap-4">
{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">
<img
src={channel.icon || "/placeholder.svg"}
alt={channel.name}
className="w-8 h-8 object-contain"
onError={(e) => {
// 图片加载失败时,使用默认图标
const target = e.target as HTMLImageElement
target.src = "/assets/icons/poster-icon.svg"
}}
/>
</div>
<h3 className="text-sm font-medium text-blue-600">{channel.name}</h3>
<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>
</div>
<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>
</div>
</Card>
</div>
))
)}
</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">
{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>
)
}