Files
cunkebao_v3/SuperAdmin/app/dashboard/customers/[id]/page.tsx

233 lines
8.7 KiB
TypeScript

"use client"
import { useEffect, useState, use } from "react"
import Link from "next/link"
import { Button } from "@/components/ui/button"
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card"
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs"
import { Badge } from "@/components/ui/badge"
import { Avatar, AvatarFallback, AvatarImage } from "@/components/ui/avatar"
import { ArrowLeft, MessageSquare, Phone, UserPlus } from "lucide-react"
import { getCustomerDetail, CustomerDetail } from "@/lib/traffic-pool-api"
// 示例数据(用于详细信息部分)
const detailData = {
devices: [
{ id: "d1", name: "iPhone 13 Pro", addedDate: "2023-06-10" },
{ id: "d2", name: "MacBook Pro", addedDate: "2023-06-12" },
],
interactions: [
{
id: "i1",
type: "消息",
content: "您好,我对您的产品很感兴趣,能详细介绍一下吗?",
date: "2023-06-15 14:30",
},
{
id: "i2",
type: "电话",
content: "讨论产品细节和价格",
duration: "15分钟",
date: "2023-06-16 10:45",
},
{
id: "i3",
type: "消息",
content: "谢谢您的详细介绍,我考虑一下再联系您。",
date: "2023-06-16 16:20",
},
{
id: "i4",
type: "消息",
content: "我决定购买您的产品,请问如何下单?",
date: "2023-06-18 09:15",
},
],
transactions: [
{
id: "t1",
product: "高级会员套餐",
amount: "¥1,299",
status: "已完成",
date: "2023-06-20",
},
],
}
export default function CustomerDetailPage({ params }: { params: { id: string } }) {
const { id } = use(params)
const [customer, setCustomer] = useState<CustomerDetail | null>(null)
const [isLoading, setIsLoading] = useState(true)
const [error, setError] = useState<string | null>(null)
useEffect(() => {
const fetchCustomerDetail = async () => {
try {
setIsLoading(true)
const response = await getCustomerDetail(id)
if (response.code === 200) {
setCustomer(response.data)
setError(null)
} else {
setError(response.msg || "获取客户详情失败")
}
} catch (err: any) {
setError(err.message || "获取客户详情失败")
} finally {
setIsLoading(false)
}
}
fetchCustomerDetail()
}, [id])
if (isLoading) {
return <div className="flex items-center justify-center min-h-screen">...</div>
}
if (error) {
return <div className="flex items-center justify-center min-h-screen text-red-500">{error}</div>
}
if (!customer) {
return <div className="flex items-center justify-center min-h-screen"></div>
}
return (
<div className="space-y-6">
<div className="flex items-center justify-between">
<div className="flex items-center gap-2">
<Button variant="outline" size="icon" asChild>
<Link href="/dashboard/customers">
<ArrowLeft className="h-4 w-4" />
</Link>
</Button>
<h1 className="text-2xl font-bold"></h1>
</div>
<Button>
<UserPlus className="mr-2 h-4 w-4" />
</Button>
</div>
<div className="grid gap-6 md:grid-cols-3">
<Card className="md:col-span-1">
<CardHeader>
<CardTitle></CardTitle>
</CardHeader>
<CardContent className="flex flex-col items-center text-center">
<Avatar className="h-24 w-24 mb-4">
<AvatarImage src={customer.avatar} alt={customer.nickname} />
<AvatarFallback>{customer.nickname.slice(0, 2)}</AvatarFallback>
</Avatar>
<h3 className="text-xl font-bold">{customer.nickname}</h3>
<div className="flex flex-wrap justify-center gap-1 mb-6">
{customer.tags.map((tag, index) => (
<Badge key={index} variant="outline">
{tag}
</Badge>
))}
</div>
<div className="w-full space-y-2 text-left">
<div className="flex justify-between">
<span className="text-sm text-muted-foreground"></span>
<span className="text-sm">{customer.gender}</span>
</div>
<div className="flex justify-between">
<span className="text-sm text-muted-foreground"></span>
<span className="text-sm">{customer.region}</span>
</div>
<div className="flex justify-between">
<span className="text-sm text-muted-foreground"></span>
<span className="text-sm">{customer.source}</span>
</div>
<div className="flex justify-between">
<span className="text-sm text-muted-foreground"></span>
<span className="text-sm">{customer.projectName}</span>
</div>
<div className="flex justify-between">
<span className="text-sm text-muted-foreground"></span>
<span className="text-sm">{customer.addTime || '暂无'}</span>
</div>
</div>
</CardContent>
</Card>
<Card className="md:col-span-2">
<CardHeader>
<CardTitle></CardTitle>
</CardHeader>
<CardContent>
<Tabs defaultValue="interactions" className="space-y-4">
<TabsList>
<TabsTrigger value="interactions"></TabsTrigger>
<TabsTrigger value="devices"></TabsTrigger>
<TabsTrigger value="transactions"></TabsTrigger>
</TabsList>
<TabsContent value="interactions" className="space-y-4">
{detailData.interactions.map((interaction) => (
<div key={interaction.id} className="flex gap-4 pb-4 border-b last:border-0">
<div className="flex h-10 w-10 items-center justify-center rounded-full bg-muted">
{interaction.type === "消息" ? (
<MessageSquare className="h-5 w-5" />
) : (
<Phone className="h-5 w-5" />
)}
</div>
<div className="flex-1 space-y-1">
<div className="flex items-center justify-between">
<p className="text-sm font-medium">{interaction.type}</p>
<p className="text-xs text-muted-foreground">{interaction.date}</p>
</div>
<p className="text-sm">{interaction.content}</p>
{interaction.duration && (
<p className="text-xs text-muted-foreground">: {interaction.duration}</p>
)}
</div>
</div>
))}
</TabsContent>
<TabsContent value="devices">
{detailData.devices.map((device) => (
<div key={device.id} className="flex items-center justify-between py-2 border-b last:border-0">
<div>
<p className="font-medium">{device.name}</p>
<p className="text-xs text-muted-foreground">: {device.addedDate}</p>
</div>
</div>
))}
</TabsContent>
<TabsContent value="transactions">
{detailData.transactions.length > 0 ? (
detailData.transactions.map((transaction) => (
<div key={transaction.id} className="flex items-center justify-between py-2 border-b last:border-0">
<div>
<p className="font-medium">{transaction.product}</p>
<p className="text-xs text-muted-foreground">: {transaction.date}</p>
</div>
<div className="text-right">
<p className="font-medium">{transaction.amount}</p>
<Badge variant={transaction.status === "已完成" ? "success" : "outline"}>
{transaction.status}
</Badge>
</div>
</div>
))
) : (
<p className="text-center text-muted-foreground py-4"></p>
)}
</TabsContent>
</Tabs>
</CardContent>
</Card>
</div>
</div>
)
}