Files
users/components/WechatGroupSelector.tsx
v0 2408d50cb0 refactor: overhaul UI for streamlined user experience
Redesign navigation, home overview, user portrait, and valuation pages
with improved functionality and responsive design.

Co-authored-by: null <4804959+fnvtk@users.noreply.github.com>
2025-07-18 13:47:12 +00:00

111 lines
3.8 KiB
TypeScript
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

"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"
import { Search } from "lucide-react"
import { Checkbox } from "@/components/ui/checkbox"
interface WechatGroup {
id: string
name: string
memberCount: number
avatar: string
owner: string
customer: string
}
interface WechatGroupSelectorProps {
open: boolean
onOpenChange: (open: boolean) => void
selectedGroups: WechatGroup[]
onSelect: (groups: WechatGroup[]) => void
}
export function WechatGroupSelector({ open, onOpenChange, selectedGroups, onSelect }: WechatGroupSelectorProps) {
const [searchQuery, setSearchQuery] = useState("")
const [groups, setGroups] = useState<WechatGroup[]>([])
const [loading, setLoading] = useState(false)
useEffect(() => {
if (open) {
fetchGroups()
}
}, [open])
const fetchGroups = async () => {
setLoading(true)
// 模拟从API获取群聊列表
await new Promise((resolve) => setTimeout(resolve, 1000))
const mockGroups = Array.from({ length: 10 }, (_, i) => ({
id: `group-${i}`,
name: `群聊${i + 1}`,
memberCount: Math.floor(Math.random() * 400) + 100,
avatar: `/placeholder.svg?height=40&width=40&text=群${i + 1}`,
owner: `群主${i + 1}`,
customer: `客户${i + 1}`,
}))
setGroups(mockGroups)
setLoading(false)
}
const filteredGroups = groups.filter((group) => group.name.toLowerCase().includes(searchQuery.toLowerCase()))
return (
<Dialog open={open} onOpenChange={onOpenChange}>
<DialogContent className="max-w-md">
<DialogHeader>
<DialogTitle></DialogTitle>
</DialogHeader>
<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"
/>
</div>
<div className="mt-4 space-y-2 max-h-[400px] overflow-y-auto">
{loading ? (
<div className="text-center py-4">...</div>
) : filteredGroups.length === 0 ? (
<div className="text-center py-4"></div>
) : (
filteredGroups.map((group) => (
<div key={group.id} className="flex items-center space-x-3 p-2 hover:bg-gray-100 rounded-lg">
<Checkbox
checked={selectedGroups.some((g) => g.id === group.id)}
onCheckedChange={(checked) => {
if (checked) {
onSelect([...selectedGroups, group])
} else {
onSelect(selectedGroups.filter((g) => g.id !== group.id))
}
}}
/>
<img src={group.avatar || "/placeholder.svg"} alt={group.name} className="w-10 h-10 rounded-lg" />
<div className="flex-1">
<div className="font-medium">{group.name}</div>
<div className="text-sm text-gray-500">
<div>{group.owner}</div>
<div>{group.customer}</div>
<div>{group.memberCount}</div>
</div>
</div>
</div>
))
)}
</div>
<div className="flex justify-end space-x-2 mt-4">
<Button variant="outline" onClick={() => onOpenChange(false)}>
</Button>
<Button onClick={() => onOpenChange(false)}></Button>
</div>
</DialogContent>
</Dialog>
)
}