Files
sub2api-mobile/app/(tabs)/groups.tsx

87 lines
2.9 KiB
TypeScript
Raw Normal View History

2026-03-07 18:12:39 +08:00
import { useQuery } from '@tanstack/react-query';
import { FolderKanban, Layers3, Search } from 'lucide-react-native';
import { useCallback, useMemo, useState } from 'react';
import { FlatList, Text, TextInput, View } from 'react-native';
2026-03-07 18:12:39 +08:00
import { ListCard } from '@/src/components/list-card';
import { ScreenShell } from '@/src/components/screen-shell';
import { useDebouncedValue } from '@/src/hooks/use-debounced-value';
2026-03-07 18:12:39 +08:00
import { listGroups } from '@/src/services/admin';
export default function GroupsScreen() {
const [searchText, setSearchText] = useState('');
const keyword = useDebouncedValue(searchText.trim(), 300);
2026-03-07 18:12:39 +08:00
const groupsQuery = useQuery({
queryKey: ['groups', keyword],
queryFn: () => listGroups(keyword),
});
const items = groupsQuery.data?.items ?? [];
const errorMessage = groupsQuery.error instanceof Error ? groupsQuery.error.message : '';
const listHeader = useMemo(
() => (
<View className="pb-4">
<View className="flex-row items-center rounded-[24px] bg-[#fbf8f2] px-4 py-3">
<Search color="#7d7468" size={18} />
<TextInput
defaultValue=""
onChangeText={setSearchText}
placeholder="搜索分组名称"
placeholderTextColor="#9b9081"
className="ml-3 flex-1 text-base text-[#16181a]"
/>
</View>
2026-03-07 18:12:39 +08:00
</View>
),
[]
);
const renderItem = useCallback(
({ item: group }: { item: (typeof items)[number] }) => (
<ListCard
title={group.name}
meta={`${group.platform} · 倍率 ${group.rate_multiplier ?? 1} · ${group.subscription_type || 'standard'}`}
badge={group.status || 'active'}
icon={FolderKanban}
>
<View className="flex-row items-center gap-2">
<Layers3 color="#7d7468" size={14} />
<Text className="text-sm text-[#7d7468]">
{group.account_count ?? 0} · {group.is_exclusive ? '独占分组' : '共享分组'}
</Text>
</View>
</ListCard>
),
[]
);
const emptyState = useMemo(
() => <ListCard title="暂无分组" meta={errorMessage || '连上 Sub2API 后,这里会展示分组列表。'} icon={FolderKanban} />,
[errorMessage]
);
2026-03-07 18:12:39 +08:00
return (
<ScreenShell
title="分组管理"
subtitle=""
titleAside={<Text className="text-[11px] text-[#a2988a]"></Text>}
variant="minimal"
scroll={false}
>
<FlatList
data={items}
renderItem={renderItem}
keyExtractor={(item) => `${item.id}`}
showsVerticalScrollIndicator={false}
ListHeaderComponent={listHeader}
ListEmptyComponent={emptyState}
ItemSeparatorComponent={() => <View className="h-4" />}
keyboardShouldPersistTaps="handled"
removeClippedSubviews
initialNumToRender={8}
maxToRenderPerBatch={8}
windowSize={5}
/>
2026-03-07 18:12:39 +08:00
</ScreenShell>
);
}