超管后台 - 关联设备列表增加设备状态和微信状态筛选、还有数量排序

This commit is contained in:
柳清爽
2025-04-29 11:30:44 +08:00
parent 12e5b93a38
commit cd864763ee

View File

@@ -8,12 +8,14 @@ import { Button } from "@/components/ui/button"
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs"
import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow } from "@/components/ui/table"
import { ArrowLeft, Edit } from "lucide-react"
import { ArrowLeft, Edit, ArrowUp, ArrowDown, ArrowUpDown } from "lucide-react"
import { toast } from "sonner"
import { use } from "react"
import Image from "next/image"
import { Badge } from "@/components/ui/badge"
import { apiRequest } from '@/lib/api-utils'
import { Loader2 } from "lucide-react"
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"
interface ProjectProfile {
id: number
@@ -69,6 +71,9 @@ export default function ProjectDetailPage({ params }: ProjectDetailPageProps) {
const [devices, setDevices] = useState<Device[]>([])
const [subUsers, setSubUsers] = useState<SubUser[]>([])
const [activeTab, setActiveTab] = useState("overview")
const [sortOrder, setSortOrder] = useState<'asc' | 'desc' | null>(null)
const [deviceStatusFilter, setDeviceStatusFilter] = useState<'all' | 'online' | 'offline'>('all')
const [wechatStatusFilter, setWechatStatusFilter] = useState<'all' | 'loggedIn' | 'loggedOut' | 'notLogged'>('all')
const fetchProject = async () => {
try {
@@ -92,17 +97,48 @@ export default function ProjectDetailPage({ params }: ProjectDetailPageProps) {
}, [id])
const fetchDevices = async () => {
if (activeTab !== "devices") return
setIsDevicesLoading(true)
try {
setIsDevicesLoading(true)
const result = await apiRequest(`/company/devices?companyId=${id}`)
if (result.code === 200 && result.data) {
setDevices(result.data)
if (result.code === 200) {
let filteredDevices = result.data
// 应用设备状态筛选
if (deviceStatusFilter !== 'all') {
filteredDevices = filteredDevices.filter(device =>
deviceStatusFilter === 'online' ? device.alive === 1 : device.alive !== 1
)
}
// 应用微信状态筛选
if (wechatStatusFilter !== 'all') {
filteredDevices = filteredDevices.filter(device => {
if (wechatStatusFilter === 'loggedIn') return device.wAlive === 1
if (wechatStatusFilter === 'loggedOut') return device.wAlive === 0
return device.wAlive === -1
})
}
// 应用排序
if (sortOrder) {
filteredDevices = [...filteredDevices].sort((a, b) => {
const aCount = a.friendCount || 0
const bCount = b.friendCount || 0
return sortOrder === 'asc' ? aCount - bCount : bCount - aCount
})
}
setDevices(filteredDevices)
} else {
toast.error(result.msg || "获取设备列表失败")
setDevices([])
}
} catch (error) {
console.error("获取设备列表失败:", error)
toast.error("网络错误,请稍后再试")
toast.error("网络错误,请稍后重试")
setDevices([])
} finally {
setIsDevicesLoading(false)
}
@@ -110,7 +146,7 @@ export default function ProjectDetailPage({ params }: ProjectDetailPageProps) {
useEffect(() => {
fetchDevices()
}, [activeTab, id])
}, [activeTab, id, sortOrder, deviceStatusFilter, wechatStatusFilter])
useEffect(() => {
const fetchSubUsers = async () => {
@@ -133,6 +169,14 @@ export default function ProjectDetailPage({ params }: ProjectDetailPageProps) {
fetchSubUsers()
}, [id])
const handleSort = () => {
setSortOrder(prev => {
if (prev === null) return 'desc'
if (prev === 'desc') return 'asc'
return null
})
}
if (isLoading) {
return <div className="flex items-center justify-center min-h-screen">...</div>
}
@@ -226,67 +270,111 @@ export default function ProjectDetailPage({ params }: ProjectDetailPageProps) {
</TabsContent>
<TabsContent value="devices">
<Card>
<CardHeader>
<CardTitle></CardTitle>
<CardDescription></CardDescription>
</CardHeader>
<CardContent>
{isDevicesLoading ? (
<div className="flex items-center justify-center py-8">...</div>
) : devices.length === 0 ? (
<div className="flex items-center justify-center py-8 text-muted-foreground"></div>
) : (
<>
<Table>
<TableHeader>
<TableRow>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead>IMEI</TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead className="text-right"></TableHead>
<div className="space-y-4">
<div className="flex justify-between items-center">
<h2 className="text-lg font-semibold"></h2>
<div className="flex items-center gap-4">
<div className="flex items-center gap-2">
<span className="text-sm text-muted-foreground"></span>
<Select
value={deviceStatusFilter}
onValueChange={(value: 'all' | 'online' | 'offline') => setDeviceStatusFilter(value)}
>
<SelectTrigger className="w-[120px]">
<SelectValue placeholder="选择状态" />
</SelectTrigger>
<SelectContent>
<SelectItem value="all"></SelectItem>
<SelectItem value="online">线</SelectItem>
<SelectItem value="offline">线</SelectItem>
</SelectContent>
</Select>
</div>
<div className="flex items-center gap-2">
<span className="text-sm text-muted-foreground"></span>
<Select
value={wechatStatusFilter}
onValueChange={(value: 'all' | 'loggedIn' | 'loggedOut' | 'notLogged') => setWechatStatusFilter(value)}
>
<SelectTrigger className="w-[120px]">
<SelectValue placeholder="选择状态" />
</SelectTrigger>
<SelectContent>
<SelectItem value="all"></SelectItem>
<SelectItem value="loggedIn"></SelectItem>
<SelectItem value="loggedOut"></SelectItem>
<SelectItem value="notLogged"></SelectItem>
</SelectContent>
</Select>
</div>
<Button
variant="outline"
size="sm"
onClick={handleSort}
className="flex items-center gap-2"
>
<span></span>
{sortOrder === 'asc' && <ArrowUp className="h-4 w-4" />}
{sortOrder === 'desc' && <ArrowDown className="h-4 w-4" />}
{sortOrder === null && <ArrowUpDown className="h-4 w-4" />}
</Button>
</div>
</div>
{isDevicesLoading ? (
<div className="flex justify-center items-center h-32">
<Loader2 className="h-8 w-8 animate-spin" />
</div>
) : devices.length > 0 ? (
<div className="border rounded-lg">
<Table>
<TableHeader>
<TableRow>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead>IMEI</TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
</TableRow>
</TableHeader>
<TableBody>
{devices.map((device) => (
<TableRow key={device.id}>
<TableCell>{device.memo}</TableCell>
<TableCell>{device.model}</TableCell>
<TableCell>{device.brand}</TableCell>
<TableCell>{device.imei}</TableCell>
<TableCell>
<Badge variant={device.alive === 1 ? "success" : "destructive"}>
{device.alive === 1 ? "在线" : "离线"}
</Badge>
</TableCell>
<TableCell>
<Badge
variant={
device.wAlive === 1
? "success"
: device.wAlive === 0
? "destructive"
: "secondary"
}
>
{device.wAlive === 1 ? "已登录" : device.wAlive === 0 ? "已登出" : "未登录微信"}
</Badge>
</TableCell>
<TableCell>{device.friendCount || 0}</TableCell>
</TableRow>
</TableHeader>
<TableBody>
{devices.map((device) => (
<TableRow key={device.id}>
<TableCell className="font-medium">{device.memo}</TableCell>
<TableCell>{device.model}</TableCell>
<TableCell>{device.brand}</TableCell>
<TableCell>{device.imei}</TableCell>
<TableCell>
<Badge variant={device.alive === 1 ? "success" : "destructive"}>
{device.alive === 1 ? "在线" : "离线"}
</Badge>
</TableCell>
<TableCell>
<Badge
variant={
device.wAlive === 1
? "success"
: device.wAlive === 0
? "destructive"
: "secondary"
}
>
{device.wAlive === 1 ? "已登录" : device.wAlive === 0 ? "已登出" : "未登录微信"}
</Badge>
</TableCell>
<TableCell className="text-right">{device.friendCount || 0}</TableCell>
</TableRow>
))}
</TableBody>
</Table>
<div className="mt-4 text-sm text-muted-foreground">
{devices.length}
</div>
</>
)}
</CardContent>
</Card>
))}
</TableBody>
</Table>
</div>
) : (
<div className="text-center py-8 text-muted-foreground">
</div>
)}
</div>
</TabsContent>
<TabsContent value="accounts">