348 lines
10 KiB
TypeScript
348 lines
10 KiB
TypeScript
"use client"
|
||
|
||
import { useState, useEffect } from "react"
|
||
import { Button } from "@/components/ui/button"
|
||
import { Input } from "@/components/ui/input"
|
||
import { Label } from "@/components/ui/label"
|
||
import { Checkbox } from "@/components/ui/checkbox"
|
||
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"
|
||
import { Search, Check, Smartphone } from "lucide-react"
|
||
|
||
// 模拟数据
|
||
const mockDevices = [
|
||
{ id: "1", name: "iPhone 13 Pro", online: true },
|
||
{ id: "2", name: "Xiaomi 12", online: true },
|
||
{ id: "3", name: "Samsung Galaxy S22", online: false },
|
||
{ id: "4", name: "OPPO Find X5", online: true },
|
||
]
|
||
|
||
const mockFriends = [
|
||
{
|
||
id: "1-friend-1",
|
||
deviceId: "1",
|
||
name: "张三",
|
||
avatar: "/placeholder.svg?height=40&width=40",
|
||
tags: ["同事", "重要"],
|
||
region: "北京",
|
||
},
|
||
{
|
||
id: "1-friend-2",
|
||
deviceId: "1",
|
||
name: "李四",
|
||
avatar: "/placeholder.svg?height=40&width=40",
|
||
tags: ["朋友"],
|
||
region: "上海",
|
||
},
|
||
{
|
||
id: "1-friend-3",
|
||
deviceId: "1",
|
||
name: "王五",
|
||
avatar: "/placeholder.svg?height=40&width=40",
|
||
tags: ["家人"],
|
||
region: "广州",
|
||
},
|
||
{
|
||
id: "1-friend-4",
|
||
deviceId: "1",
|
||
name: "赵六",
|
||
avatar: "/placeholder.svg?height=40&width=40",
|
||
tags: ["客户", "重要"],
|
||
region: "深圳",
|
||
},
|
||
{
|
||
id: "2-friend-1",
|
||
deviceId: "2",
|
||
name: "陈一",
|
||
avatar: "/placeholder.svg?height=40&width=40",
|
||
tags: ["同学"],
|
||
region: "北京",
|
||
},
|
||
{
|
||
id: "2-friend-2",
|
||
deviceId: "2",
|
||
name: "杨二",
|
||
avatar: "/placeholder.svg?height=40&width=40",
|
||
tags: ["朋友", "重要"],
|
||
region: "上海",
|
||
},
|
||
{
|
||
id: "2-friend-3",
|
||
deviceId: "2",
|
||
name: "刘三",
|
||
avatar: "/placeholder.svg?height=40&width=40",
|
||
tags: ["同事"],
|
||
region: "广州",
|
||
},
|
||
{
|
||
id: "3-friend-1",
|
||
deviceId: "3",
|
||
name: "周七",
|
||
avatar: "/placeholder.svg?height=40&width=40",
|
||
tags: ["客户"],
|
||
region: "北京",
|
||
},
|
||
{
|
||
id: "3-friend-2",
|
||
deviceId: "3",
|
||
name: "吴八",
|
||
avatar: "/placeholder.svg?height=40&width=40",
|
||
tags: ["朋友"],
|
||
region: "上海",
|
||
},
|
||
{
|
||
id: "3-friend-3",
|
||
deviceId: "3",
|
||
name: "郑九",
|
||
avatar: "/placeholder.svg?height=40&width=40",
|
||
tags: ["家人", "重要"],
|
||
region: "广州",
|
||
},
|
||
{
|
||
id: "4-friend-1",
|
||
deviceId: "4",
|
||
name: "冯十",
|
||
avatar: "/placeholder.svg?height=40&width=40",
|
||
tags: ["同事"],
|
||
region: "深圳",
|
||
},
|
||
{
|
||
id: "4-friend-2",
|
||
deviceId: "4",
|
||
name: "蒋十一",
|
||
avatar: "/placeholder.svg?height=40&width=40",
|
||
tags: ["朋友"],
|
||
region: "北京",
|
||
},
|
||
]
|
||
|
||
interface Friend {
|
||
id: string
|
||
deviceId: string
|
||
name: string
|
||
avatar: string
|
||
tags: string[]
|
||
region: string
|
||
}
|
||
|
||
interface FriendSelectorProps {
|
||
onSelectionChange: (friends: Friend[]) => void
|
||
defaultSelectedFriendIds?: string[]
|
||
}
|
||
|
||
export function FriendSelector({ onSelectionChange, defaultSelectedFriendIds = [] }: FriendSelectorProps) {
|
||
const [selectedDevices, setSelectedDevices] = useState<string[]>([])
|
||
const [selectedFriendIds, setSelectedFriendIds] = useState<string[]>(defaultSelectedFriendIds)
|
||
const [searchQuery, setSearchQuery] = useState("")
|
||
const [selectedTag, setSelectedTag] = useState<string>("all")
|
||
const [selectedRegion, setSelectedRegion] = useState<string>("all")
|
||
|
||
// 获取所有可用的标签和地区
|
||
const allTags = Array.from(new Set(mockFriends.flatMap((friend) => friend.tags)))
|
||
const allRegions = Array.from(new Set(mockFriends.map((friend) => friend.region)))
|
||
|
||
// 根据选择的设备过滤好友
|
||
const filteredFriends = mockFriends
|
||
.filter((friend) => {
|
||
// 如果没有选择设备,显示所有好友
|
||
if (selectedDevices.length === 0) return true
|
||
|
||
// 只显示选中设备的好友
|
||
return selectedDevices.includes(friend.deviceId)
|
||
})
|
||
.filter((friend) => {
|
||
// 搜索过滤
|
||
if (searchQuery) {
|
||
return friend.name.toLowerCase().includes(searchQuery.toLowerCase())
|
||
}
|
||
return true
|
||
})
|
||
.filter((friend) => {
|
||
// 标签过滤
|
||
if (selectedTag !== "all") {
|
||
return friend.tags.includes(selectedTag)
|
||
}
|
||
return true
|
||
})
|
||
.filter((friend) => {
|
||
// 地区过滤
|
||
if (selectedRegion !== "all") {
|
||
return friend.region === selectedRegion
|
||
}
|
||
return true
|
||
})
|
||
|
||
// 当选择的好友ID变化时,通知父组件
|
||
useEffect(() => {
|
||
const selectedFriends = mockFriends.filter((friend) => selectedFriendIds.includes(friend.id))
|
||
onSelectionChange(selectedFriends)
|
||
}, [selectedFriendIds, onSelectionChange])
|
||
|
||
const toggleDeviceSelection = (deviceId: string) => {
|
||
setSelectedDevices((prev) => {
|
||
if (prev.includes(deviceId)) {
|
||
return prev.filter((id) => id !== deviceId)
|
||
} else {
|
||
return [...prev, deviceId]
|
||
}
|
||
})
|
||
}
|
||
|
||
const toggleFriendSelection = (friendId: string) => {
|
||
setSelectedFriendIds((prev) => {
|
||
if (prev.includes(friendId)) {
|
||
return prev.filter((id) => id !== friendId)
|
||
} else {
|
||
return [...prev, friendId]
|
||
}
|
||
})
|
||
}
|
||
|
||
const selectAllFriends = () => {
|
||
setSelectedFriendIds(filteredFriends.map((friend) => friend.id))
|
||
}
|
||
|
||
const deselectAllFriends = () => {
|
||
setSelectedFriendIds([])
|
||
}
|
||
|
||
return (
|
||
<div className="space-y-6">
|
||
{/* 设备选择 */}
|
||
<div>
|
||
<Label className="mb-2 block">选择设备</Label>
|
||
<div className="flex flex-wrap gap-2">
|
||
{mockDevices.map((device) => (
|
||
<Button
|
||
key={device.id}
|
||
variant={selectedDevices.includes(device.id) ? "default" : "outline"}
|
||
size="sm"
|
||
onClick={() => toggleDeviceSelection(device.id)}
|
||
className="flex items-center gap-1"
|
||
disabled={!device.online}
|
||
>
|
||
<Smartphone className="h-4 w-4" />
|
||
{device.name}
|
||
{selectedDevices.includes(device.id) && <Check className="h-3 w-3 ml-1" />}
|
||
{!device.online && <span className="text-xs ml-1 text-gray-500">(离线)</span>}
|
||
</Button>
|
||
))}
|
||
</div>
|
||
</div>
|
||
|
||
{/* 好友筛选 */}
|
||
<div className="flex flex-wrap gap-3">
|
||
<div className="flex-1 min-w-[200px]">
|
||
<Label htmlFor="search" className="mb-2 block">
|
||
搜索好友
|
||
</Label>
|
||
<div className="relative">
|
||
<Search className="absolute left-2 top-2.5 h-4 w-4 text-gray-400" />
|
||
<Input
|
||
id="search"
|
||
placeholder="输入好友名称"
|
||
value={searchQuery}
|
||
onChange={(e) => setSearchQuery(e.target.value)}
|
||
className="pl-8"
|
||
/>
|
||
</div>
|
||
</div>
|
||
|
||
<div className="w-40">
|
||
<Label htmlFor="tag-filter" className="mb-2 block">
|
||
标签筛选
|
||
</Label>
|
||
<Select value={selectedTag} onValueChange={setSelectedTag}>
|
||
<SelectTrigger id="tag-filter">
|
||
<SelectValue placeholder="选择标签" />
|
||
</SelectTrigger>
|
||
<SelectContent>
|
||
<SelectItem value="all">全部标签</SelectItem>
|
||
{allTags.map((tag) => (
|
||
<SelectItem key={tag} value={tag}>
|
||
{tag}
|
||
</SelectItem>
|
||
))}
|
||
</SelectContent>
|
||
</Select>
|
||
</div>
|
||
|
||
<div className="w-40">
|
||
<Label htmlFor="region-filter" className="mb-2 block">
|
||
地区筛选
|
||
</Label>
|
||
<Select value={selectedRegion} onValueChange={setSelectedRegion}>
|
||
<SelectTrigger id="region-filter">
|
||
<SelectValue placeholder="选择地区" />
|
||
</SelectTrigger>
|
||
<SelectContent>
|
||
<SelectItem value="all">全部地区</SelectItem>
|
||
{allRegions.map((region) => (
|
||
<SelectItem key={region} value={region}>
|
||
{region}
|
||
</SelectItem>
|
||
))}
|
||
</SelectContent>
|
||
</Select>
|
||
</div>
|
||
</div>
|
||
|
||
{/* 好友列表 */}
|
||
<div>
|
||
<div className="flex justify-between items-center mb-2">
|
||
<Label className="font-medium">好友列表 ({filteredFriends.length})</Label>
|
||
<div className="flex gap-2">
|
||
<Button variant="outline" size="sm" onClick={selectAllFriends}>
|
||
全选
|
||
</Button>
|
||
<Button variant="outline" size="sm" onClick={deselectAllFriends}>
|
||
取消全选
|
||
</Button>
|
||
</div>
|
||
</div>
|
||
|
||
<div className="border rounded-md overflow-hidden">
|
||
<div className="max-h-[300px] overflow-y-auto p-1">
|
||
{filteredFriends.length > 0 ? (
|
||
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-2">
|
||
{filteredFriends.map((friend) => (
|
||
<div
|
||
key={friend.id}
|
||
className={`flex items-center gap-2 p-2 rounded-md border ${
|
||
selectedFriendIds.includes(friend.id) ? "bg-blue-50 border-blue-200" : "border-gray-200"
|
||
}`}
|
||
>
|
||
<Checkbox
|
||
id={`friend-${friend.id}`}
|
||
checked={selectedFriendIds.includes(friend.id)}
|
||
onCheckedChange={() => toggleFriendSelection(friend.id)}
|
||
/>
|
||
<img src={friend.avatar || "/placeholder.svg"} alt={friend.name} className="w-8 h-8 rounded-full" />
|
||
<div className="flex-1 min-w-0">
|
||
<div className="font-medium truncate">{friend.name}</div>
|
||
<div className="text-xs text-gray-500 truncate">{friend.region}</div>
|
||
</div>
|
||
{friend.tags.length > 0 && (
|
||
<div className="flex flex-wrap gap-1">
|
||
{friend.tags.map((tag) => (
|
||
<span key={tag} className="px-1.5 py-0.5 bg-gray-100 text-gray-700 rounded text-xs">
|
||
{tag}
|
||
</span>
|
||
))}
|
||
</div>
|
||
)}
|
||
</div>
|
||
))}
|
||
</div>
|
||
) : (
|
||
<div className="text-center py-8 text-gray-500">没有找到符合条件的好友</div>
|
||
)}
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div className="text-sm text-gray-500">已选择 {selectedFriendIds.length} 位好友</div>
|
||
</div>
|
||
)
|
||
}
|
||
|