Files
cunkebao_v3/Cunkebao/components/WechatFriendSelector.tsx

214 lines
6.9 KiB
TypeScript
Raw Normal View History

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-04-21 09:19:50 +08:00
import { Search, ChevronLeft, ChevronRight } 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-04-21 09:19:50 +08:00
import { api } from "@/lib/api"
import { showToast } from "@/lib/toast"
2025-03-29 16:50:39 +08:00
interface WechatFriend {
id: string
nickname: string
wechatId: string
avatar: string
2025-04-21 09:19:50 +08:00
gender?: "male" | "female"
customer?: string
alias?: string
ownerNickname?: string
ownerAlias?: string
createTime?: string
}
interface ApiResponse<T = any> {
code: number
msg: string
data: T
}
interface FriendListResponse {
list: any[]
total: number
2025-03-29 16:50:39 +08:00
}
interface WechatFriendSelectorProps {
open: boolean
onOpenChange: (open: boolean) => void
selectedFriends: WechatFriend[]
onSelect: (friends: WechatFriend[]) => void
}
export function WechatFriendSelector({ open, onOpenChange, selectedFriends, onSelect }: WechatFriendSelectorProps) {
const [searchQuery, setSearchQuery] = useState("")
const [friends, setFriends] = useState<WechatFriend[]>([])
const [loading, setLoading] = useState(false)
2025-04-21 09:19:50 +08:00
const [page, setPage] = useState(1)
const [totalPages, setTotalPages] = useState(1)
const [totalItems, setTotalItems] = useState(0)
const pageSize = 20
2025-03-29 16:50:39 +08:00
useEffect(() => {
if (open) {
2025-04-21 09:19:50 +08:00
fetchFriends(1)
2025-03-29 16:50:39 +08:00
}
}, [open])
2025-04-21 09:19:50 +08:00
const fetchFriends = async (pageNum: number) => {
2025-03-29 16:50:39 +08:00
setLoading(true)
2025-04-21 09:19:50 +08:00
try {
const queryParams = new URLSearchParams({
page: pageNum.toString(),
limit: pageSize.toString(),
...(searchQuery ? { keyword: searchQuery } : {})
})
const response = await api.get<ApiResponse<FriendListResponse>>(`/v1/friend?${queryParams.toString()}`)
if (response.code === 200 && response.data) {
const friendsList = response.data.list.map(item => ({
id: item.id || item.wechatId || `${item.nickname}-${Math.random()}`,
nickname: item.nickname || '未知好友',
wechatId: item.wechatId || '',
avatar: item.avatar || '/placeholder.svg',
alias: item.alias || '',
ownerNickname: item.ownerNickname || '',
ownerAlias: item.ownerAlias || item.ownerWechatId || '',
createTime: item.createTime || '--'
}))
setFriends(friendsList)
setTotalItems(response.data.total)
setTotalPages(Math.ceil(response.data.total / pageSize))
setPage(pageNum)
} else {
showToast(response.msg || "获取好友列表失败", "error")
}
} catch (error: any) {
console.error("获取好友列表失败:", error)
showToast(error?.message || "请检查网络连接", "error")
} finally {
setLoading(false)
}
2025-03-29 16:50:39 +08:00
}
2025-04-21 09:19:50 +08:00
const handleSearch = () => {
fetchFriends(1)
}
const handlePrevPage = () => {
if (page > 1) {
fetchFriends(page - 1)
}
}
const handleNextPage = () => {
if (page < totalPages) {
fetchFriends(page + 1)
}
}
2025-03-29 16:50:39 +08:00
return (
<Dialog open={open} onOpenChange={onOpenChange}>
<DialogContent className="max-w-md">
<DialogHeader>
<DialogTitle></DialogTitle>
</DialogHeader>
2025-04-21 09:19:50 +08:00
<div className="flex gap-2">
<div className="relative flex-1">
<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)}
onKeyDown={(e) => e.key === 'Enter' && handleSearch()}
className="pl-9"
/>
</div>
<Button
variant="outline"
size="icon"
onClick={handleSearch}
disabled={loading}
>
<Search className="h-4 w-4" />
</Button>
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-04-21 09:19:50 +08:00
) : friends.length === 0 ? (
2025-03-29 16:50:39 +08:00
<div className="text-center py-4"></div>
) : (
2025-04-21 09:19:50 +08:00
friends.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
checked={selectedFriends.some((f) => f.id === friend.id)}
onCheckedChange={(checked) => {
if (checked) {
onSelect([...selectedFriends, friend])
} else {
onSelect(selectedFriends.filter((f) => f.id !== friend.id))
}
}}
/>
<Avatar>
<AvatarImage src={friend.avatar} />
2025-04-21 09:19:50 +08:00
<AvatarFallback>{friend.nickname?.[0] || '?'}</AvatarFallback>
2025-03-29 16:50:39 +08:00
</Avatar>
2025-04-21 09:19:50 +08:00
<div className="flex-1 min-w-0">
<div className="font-medium truncate">{friend.nickname}</div>
2025-03-29 16:50:39 +08:00
<div className="text-sm text-gray-500">
2025-04-21 09:19:50 +08:00
{friend.wechatId && <div className="truncate">ID{friend.alias || friend.wechatId}</div>}
{friend.ownerNickname && <div className="truncate">{friend.ownerNickname} ({friend.ownerAlias || '--'})</div>}
2025-03-29 16:50:39 +08:00
</div>
</div>
</div>
))
)}
</div>
2025-04-21 09:19:50 +08:00
{/* 分页控制 */}
{totalPages > 1 && (
<div className="flex items-center justify-between border-t pt-4 mt-4">
<div className="text-sm text-gray-500">
{totalItems}
</div>
<div className="flex items-center space-x-2">
<Button
variant="outline"
size="icon"
onClick={handlePrevPage}
disabled={page === 1 || loading}
>
<ChevronLeft className="h-4 w-4" />
</Button>
<span className="text-sm">
{page} / {totalPages}
</span>
<Button
variant="outline"
size="icon"
onClick={handleNextPage}
disabled={page === totalPages || loading}
>
<ChevronRight className="h-4 w-4" />
</Button>
</div>
</div>
)}
2025-03-29 16:50:39 +08:00
<div className="flex justify-end space-x-2 mt-4">
<Button variant="outline" onClick={() => onOpenChange(false)}>
</Button>
2025-04-21 09:19:50 +08:00
<Button onClick={() => onOpenChange(false)}> ({selectedFriends.length})</Button>
2025-03-29 16:50:39 +08:00
</div>
</DialogContent>
</Dialog>
)
}