2025-03-29 16:50:39 +08:00
|
|
|
|
"use client"
|
|
|
|
|
|
|
|
|
|
|
|
import { useState, useEffect } from "react"
|
|
|
|
|
|
import { Dialog, DialogContent, DialogHeader, DialogTitle } from "@/components/ui/dialog"
|
|
|
|
|
|
import { Input } from "@/components/ui/input"
|
|
|
|
|
|
import { Button } from "@/components/ui/button"
|
2025-07-11 11:40:24 +08:00
|
|
|
|
import { Search } from "lucide-react"
|
2025-03-29 16:50:39 +08:00
|
|
|
|
import { Checkbox } from "@/components/ui/checkbox"
|
|
|
|
|
|
import { Avatar, AvatarFallback, AvatarImage } from "@/components/ui/avatar"
|
|
|
|
|
|
|
2025-07-11 11:40:24 +08:00
|
|
|
|
interface WechatFriend {
|
2025-03-29 16:50:39 +08:00
|
|
|
|
id: string
|
|
|
|
|
|
nickname: string
|
|
|
|
|
|
wechatId: string
|
|
|
|
|
|
avatar: string
|
2025-07-11 11:40:24 +08:00
|
|
|
|
gender: "male" | "female"
|
|
|
|
|
|
customer: string
|
2025-03-29 16:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
interface WechatFriendSelectorProps {
|
|
|
|
|
|
open: boolean
|
|
|
|
|
|
onOpenChange: (open: boolean) => void
|
|
|
|
|
|
selectedFriends: WechatFriend[]
|
|
|
|
|
|
onSelect: (friends: WechatFriend[]) => void
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-11 11:40:24 +08:00
|
|
|
|
export function WechatFriendSelector({ open, onOpenChange, selectedFriends, onSelect }: WechatFriendSelectorProps) {
|
2025-03-29 16:50:39 +08:00
|
|
|
|
const [searchQuery, setSearchQuery] = useState("")
|
|
|
|
|
|
const [friends, setFriends] = useState<WechatFriend[]>([])
|
|
|
|
|
|
const [loading, setLoading] = useState(false)
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
if (open) {
|
2025-07-11 11:40:24 +08:00
|
|
|
|
fetchFriends()
|
2025-03-29 16:50:39 +08:00
|
|
|
|
}
|
2025-07-11 11:40:24 +08:00
|
|
|
|
}, [open])
|
2025-03-29 16:50:39 +08:00
|
|
|
|
|
2025-07-11 11:40:24 +08:00
|
|
|
|
const fetchFriends = async () => {
|
2025-03-29 16:50:39 +08:00
|
|
|
|
setLoading(true)
|
2025-07-11 11:40:24 +08:00
|
|
|
|
// 模拟从API获取好友列表
|
|
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 1000))
|
|
|
|
|
|
const mockFriends = Array.from({ length: 20 }, (_, i) => ({
|
|
|
|
|
|
id: `friend-${i}`,
|
|
|
|
|
|
nickname: `好友${i + 1}`,
|
|
|
|
|
|
wechatId: `wxid_${Math.random().toString(36).substr(2, 8)}`,
|
|
|
|
|
|
avatar: `/placeholder.svg?height=40&width=40&text=${i + 1}`,
|
|
|
|
|
|
gender: Math.random() > 0.5 ? "male" : "female",
|
|
|
|
|
|
customer: `客户${i + 1}`,
|
|
|
|
|
|
}))
|
|
|
|
|
|
setFriends(mockFriends)
|
|
|
|
|
|
setLoading(false)
|
2025-04-21 09:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-11 11:40:24 +08:00
|
|
|
|
const filteredFriends = friends.filter(
|
|
|
|
|
|
(friend) =>
|
|
|
|
|
|
friend.nickname.toLowerCase().includes(searchQuery.toLowerCase()) ||
|
|
|
|
|
|
friend.wechatId.toLowerCase().includes(searchQuery.toLowerCase()),
|
|
|
|
|
|
)
|
2025-03-29 16:50:39 +08:00
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
|
<Dialog open={open} onOpenChange={onOpenChange}>
|
|
|
|
|
|
<DialogContent className="max-w-md">
|
|
|
|
|
|
<DialogHeader>
|
|
|
|
|
|
<DialogTitle>选择微信好友</DialogTitle>
|
|
|
|
|
|
</DialogHeader>
|
2025-07-11 11:40:24 +08:00
|
|
|
|
<div className="relative">
|
|
|
|
|
|
<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"
|
|
|
|
|
|
/>
|
2025-03-29 16:50:39 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div className="mt-4 space-y-2 max-h-[400px] overflow-y-auto">
|
|
|
|
|
|
{loading ? (
|
|
|
|
|
|
<div className="text-center py-4">加载中...</div>
|
2025-07-11 11:40:24 +08:00
|
|
|
|
) : filteredFriends.length === 0 ? (
|
2025-03-29 16:50:39 +08:00
|
|
|
|
<div className="text-center py-4">未找到匹配的好友</div>
|
|
|
|
|
|
) : (
|
2025-07-11 11:40:24 +08:00
|
|
|
|
filteredFriends.map((friend) => (
|
2025-03-29 16:50:39 +08:00
|
|
|
|
<div key={friend.id} className="flex items-center space-x-3 p-2 hover:bg-gray-100 rounded-lg">
|
|
|
|
|
|
<Checkbox
|
2025-07-11 11:40:24 +08:00
|
|
|
|
checked={selectedFriends.some((f) => f.id === friend.id)}
|
2025-03-29 16:50:39 +08:00
|
|
|
|
onCheckedChange={(checked) => {
|
|
|
|
|
|
if (checked) {
|
2025-07-11 11:40:24 +08:00
|
|
|
|
onSelect([...selectedFriends, friend])
|
2025-03-29 16:50:39 +08:00
|
|
|
|
} else {
|
2025-07-11 11:40:24 +08:00
|
|
|
|
onSelect(selectedFriends.filter((f) => f.id !== friend.id))
|
2025-03-29 16:50:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
}}
|
|
|
|
|
|
/>
|
|
|
|
|
|
<Avatar>
|
|
|
|
|
|
<AvatarImage src={friend.avatar} />
|
2025-07-11 11:40:24 +08:00
|
|
|
|
<AvatarFallback>{friend.nickname[0]}</AvatarFallback>
|
2025-03-29 16:50:39 +08:00
|
|
|
|
</Avatar>
|
2025-07-11 11:40:24 +08:00
|
|
|
|
<div className="flex-1">
|
|
|
|
|
|
<div className="font-medium">{friend.nickname}</div>
|
2025-03-29 16:50:39 +08:00
|
|
|
|
<div className="text-sm text-gray-500">
|
2025-07-11 11:40:24 +08:00
|
|
|
|
<div>{friend.wechatId}</div>
|
|
|
|
|
|
<div>归属客户:{friend.customer}</div>
|
2025-03-29 16:50:39 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
))
|
|
|
|
|
|
)}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="flex justify-end space-x-2 mt-4">
|
|
|
|
|
|
<Button variant="outline" onClick={() => onOpenChange(false)}>
|
|
|
|
|
|
取消
|
|
|
|
|
|
</Button>
|
2025-07-11 11:40:24 +08:00
|
|
|
|
<Button onClick={() => onOpenChange(false)}>确定</Button>
|
2025-03-29 16:50:39 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</DialogContent>
|
|
|
|
|
|
</Dialog>
|
|
|
|
|
|
)
|
|
|
|
|
|
}
|