import { useMutation, useQuery, useQueryClient } from '@tanstack/react-query'; import { KeyRound, Search, ShieldCheck, ShieldOff } from 'lucide-react-native'; import { router } from 'expo-router'; import { useCallback, useMemo, useState } from 'react'; import { FlatList, Pressable, RefreshControl, Text, TextInput, View } from 'react-native'; import { ListCard } from '@/src/components/list-card'; import { ScreenShell } from '@/src/components/screen-shell'; import { useDebouncedValue } from '@/src/hooks/use-debounced-value'; import { getAccount, getAccountTodayStats, getDashboardTrend, listAccounts, setAccountSchedulable, testAccount } from '@/src/services/admin'; import type { AdminAccount } from '@/src/types/admin'; function getDateRange() { const end = new Date(); const start = new Date(); start.setDate(end.getDate() - 6); const toDate = (value: Date) => value.toISOString().slice(0, 10); return { start_date: toDate(start), end_date: toDate(end), }; } function formatTime(value?: string | null) { if (!value) return '--'; const date = new Date(value); if (Number.isNaN(date.getTime())) return '--'; return `${date.getFullYear()}/${String(date.getMonth() + 1).padStart(2, '0')}/${String(date.getDate()).padStart(2, '0')} ${String(date.getHours()).padStart(2, '0')}:${String(date.getMinutes()).padStart(2, '0')}`; } function getAccountError(account: AdminAccount) { return Boolean(account.status === 'error' || account.error_message); } export default function AccountsScreen() { const [searchText, setSearchText] = useState(''); const [filter, setFilter] = useState<'all' | 'schedulable' | 'paused' | 'error'>('all'); const keyword = useDebouncedValue(searchText.trim(), 300); const queryClient = useQueryClient(); const range = getDateRange(); const accountsQuery = useQuery({ queryKey: ['accounts', keyword], queryFn: () => listAccounts(keyword), }); const toggleMutation = useMutation({ mutationFn: ({ accountId, schedulable }: { accountId: number; schedulable: boolean }) => setAccountSchedulable(accountId, schedulable), onSuccess: () => queryClient.invalidateQueries({ queryKey: ['accounts'] }), }); const items = accountsQuery.data?.items ?? []; const filteredItems = useMemo(() => { return items.filter((account) => { if (filter === 'schedulable') return account.schedulable !== false && !getAccountError(account); if (filter === 'paused') return account.schedulable === false && !getAccountError(account); if (filter === 'error') return getAccountError(account); return true; }); }, [filter, items]); const errorMessage = accountsQuery.error instanceof Error ? accountsQuery.error.message : ''; const summary = useMemo(() => { const total = items.length; const errors = items.filter(getAccountError).length; const paused = items.filter((item) => item.schedulable === false && !getAccountError(item)).length; const active = items.filter((item) => item.schedulable !== false && !getAccountError(item)).length; return { total, active, paused, errors }; }, [items]); const listHeader = useMemo( () => ( {([ ['all', `全部 ${summary.total}`], ['schedulable', `可调度 ${summary.active}`], ['paused', `暂停 ${summary.paused}`], ['error', `异常 ${summary.errors}`], ] as const).map(([key, label]) => { const active = filter === key; return ( setFilter(key)} className={active ? 'rounded-full bg-[#1d5f55] px-3 py-2' : 'rounded-full bg-[#e7dfcf] px-3 py-2'} > {label} ); })} ), [filter, summary.active, summary.errors, summary.paused, summary.total] ); const renderItem = useCallback( ({ item: account }: { item: (typeof filteredItems)[number] }) => { const isError = getAccountError(account); const statusText = isError ? '异常' : account.schedulable ? '可调度' : '暂停调度'; const groupsText = account.groups?.map((group) => group.name).filter(Boolean).slice(0, 3).join(' · '); return ( { void queryClient.prefetchQuery({ queryKey: ['account', account.id], queryFn: () => getAccount(account.id) }); void queryClient.prefetchQuery({ queryKey: ['account-today-stats', account.id], queryFn: () => getAccountTodayStats(account.id) }); void queryClient.prefetchQuery({ queryKey: ['account-trend', account.id, range.start_date, range.end_date], queryFn: () => getDashboardTrend({ ...range, granularity: 'day', account_id: account.id }), }); router.push(`/accounts/${account.id}`); }} > {account.schedulable && !isError ? : } {statusText} 最近使用 {formatTime(account.last_used_at || account.updated_at)} 并发 {account.current_concurrency ?? 0} / {account.concurrency ?? 0} 倍率 {(account.rate_multiplier ?? 1).toFixed(2)}x {groupsText ? 分组 {groupsText} : null} {account.error_message ? 异常信息:{account.error_message} : null} { event.stopPropagation(); testAccount(account.id).catch(() => undefined); }} > 测试 { event.stopPropagation(); toggleMutation.mutate({ accountId: account.id, schedulable: !account.schedulable, }); }} > {account.schedulable ? '暂停' : '恢复'} ); }, [filteredItems, queryClient, range.end_date, range.start_date, toggleMutation] ); const emptyState = useMemo( () => , [errorMessage] ); return ( 更接近网页后台的账号视图。} variant="minimal" scroll={false} > `${item.id}`} showsVerticalScrollIndicator={false} refreshControl={ void accountsQuery.refetch()} tintColor="#1d5f55" />} ListHeaderComponent={listHeader} ListEmptyComponent={emptyState} ItemSeparatorComponent={() => } keyboardShouldPersistTaps="handled" removeClippedSubviews initialNumToRender={8} maxToRenderPerBatch={8} windowSize={5} /> ); }