代码提交

This commit is contained in:
柳清爽
2025-04-15 16:10:09 +08:00
parent d05cfc58ea
commit 005c68e6a8
5 changed files with 94 additions and 52 deletions

View File

@@ -1,5 +1,6 @@
"use client"
import { useEffect, useState } from "react"
import Link from "next/link"
import { Button } from "@/components/ui/button"
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card"
@@ -7,19 +8,10 @@ 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"
// Sample customer data
const customerData = {
id: "1",
name: "张三",
avatar: "/placeholder.svg?height=100&width=100",
wechatId: "zhangsan123",
gender: "男",
region: "北京",
source: "微信搜索",
tags: ["潜在客户", "高消费"],
projectName: "电商平台项目",
addedDate: "2023-06-10",
// 示例数据(用于详细信息部分)
const detailData = {
devices: [
{ id: "d1", name: "iPhone 13 Pro", addedDate: "2023-06-10" },
{ id: "d2", name: "MacBook Pro", addedDate: "2023-06-12" },
@@ -63,6 +55,43 @@ const customerData = {
}
export default function CustomerDetailPage({ params }: { params: { id: string } }) {
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(params.id)
if (response.code === 200) {
setCustomer(response.data)
setError(null)
} else {
setError(response.msg || "获取客户详情失败")
}
} catch (err: any) {
setError(err.message || "获取客户详情失败")
} finally {
setIsLoading(false)
}
}
fetchCustomerDetail()
}, [params.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">
@@ -86,15 +115,14 @@ export default function CustomerDetailPage({ params }: { params: { id: string }
</CardHeader>
<CardContent className="flex flex-col items-center text-center">
<Avatar className="h-24 w-24 mb-4">
<AvatarImage src={customerData.avatar} alt={customerData.name} />
<AvatarFallback>{customerData.name.slice(0, 2)}</AvatarFallback>
<AvatarImage src={customer.avatar} alt={customer.nickname} />
<AvatarFallback>{customer.nickname.slice(0, 2)}</AvatarFallback>
</Avatar>
<h3 className="text-xl font-bold">{customerData.name}</h3>
<p className="text-sm text-muted-foreground mb-4">{customerData.wechatId}</p>
<h3 className="text-xl font-bold">{customer.nickname}</h3>
<div className="flex flex-wrap justify-center gap-1 mb-6">
{customerData.tags.map((tag) => (
<Badge key={tag} variant="outline">
{customer.tags.map((tag, index) => (
<Badge key={index} variant="outline">
{tag}
</Badge>
))}
@@ -103,23 +131,23 @@ export default function CustomerDetailPage({ params }: { params: { id: string }
<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">{customerData.gender}</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">{customerData.region}</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">{customerData.source}</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">{customerData.projectName}</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">{customerData.addedDate}</span>
<span className="text-sm">{customer.addTime || '暂无'}</span>
</div>
</div>
</CardContent>
@@ -138,7 +166,7 @@ export default function CustomerDetailPage({ params }: { params: { id: string }
</TabsList>
<TabsContent value="interactions" className="space-y-4">
{customerData.interactions.map((interaction) => (
{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 === "消息" ? (
@@ -162,7 +190,7 @@ export default function CustomerDetailPage({ params }: { params: { id: string }
</TabsContent>
<TabsContent value="devices">
{customerData.devices.map((device) => (
{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>
@@ -173,8 +201,8 @@ export default function CustomerDetailPage({ params }: { params: { id: string }
</TabsContent>
<TabsContent value="transactions">
{customerData.transactions.length > 0 ? (
customerData.transactions.map((transaction) => (
{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>

View File

@@ -294,9 +294,9 @@ export default function CustomersPage() {
<Table>
<TableHeader>
<TableRow>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead>ID</TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
@@ -325,31 +325,23 @@ export default function CustomersPage() {
<Avatar>
<AvatarImage
src={customer.avatar || "/placeholder.svg?height=40&width=40"}
alt={customer.nickname}
alt={customer.nickname || "未知"}
onError={(e) => {
// 图片加载失败时使用默认图片
const target = e.target as HTMLImageElement;
target.src = "/placeholder.svg?height=40&width=40";
}}
/>
<AvatarFallback>{customer.nickname.slice(0, 2)}</AvatarFallback>
<AvatarFallback>{(customer.nickname || "未知").slice(0, 2)}</AvatarFallback>
</Avatar>
<div>
<div className="font-medium">{customer.nickname}</div>
<div className="font-medium">{customer.nickname || "未知"}</div>
<div className="text-xs text-muted-foreground">{customer.gender}</div>
</div>
</div>
</TableCell>
<TableCell>{customer.wechatId}</TableCell>
<TableCell>
<div className="flex flex-wrap gap-1">
{customer.tags.map((tag, index) => (
<Badge key={index} variant="outline">
{tag}
</Badge>
))}
</div>
</TableCell>
<TableCell>{customer.gender}</TableCell>
<TableCell>{customer.region}</TableCell>
<TableCell>{customer.source}</TableCell>
<TableCell>{customer.projectName}</TableCell>