mirror of
https://github.com/clash-verge-rev/clash-verge-rev.git
synced 2026-04-13 05:20:28 +08:00
104 lines
2.8 KiB
TypeScript
104 lines
2.8 KiB
TypeScript
import { Box } from '@mui/material'
|
|
import { useCallback, useEffect, useMemo, useRef, useState } from 'react'
|
|
import { useTranslation } from 'react-i18next'
|
|
|
|
import {
|
|
BaseEmpty,
|
|
BasePage,
|
|
BaseSearchBox,
|
|
VirtualList,
|
|
type VirtualListHandle,
|
|
} from '@/components/base'
|
|
import { ScrollTopButton } from '@/components/layout/scroll-top-button'
|
|
import { ProviderButton } from '@/components/rule/provider-button'
|
|
import RuleItem from '@/components/rule/rule-item'
|
|
import { useVisibility } from '@/hooks/use-visibility'
|
|
import { useAppData } from '@/providers/app-data-context'
|
|
|
|
const RulesPage = () => {
|
|
const { t } = useTranslation()
|
|
const { rules = [], refreshRules, refreshRuleProviders } = useAppData()
|
|
const [match, setMatch] = useState(() => (_: string) => true)
|
|
const virtuosoRef = useRef<VirtualListHandle>(null)
|
|
const [showScrollTop, setShowScrollTop] = useState(false)
|
|
const pageVisible = useVisibility()
|
|
|
|
// 在组件挂载时和页面获得焦点时刷新规则数据
|
|
useEffect(() => {
|
|
refreshRules()
|
|
refreshRuleProviders()
|
|
|
|
if (pageVisible) {
|
|
refreshRules()
|
|
refreshRuleProviders()
|
|
}
|
|
}, [refreshRules, refreshRuleProviders, pageVisible])
|
|
|
|
const filteredRules = useMemo(() => {
|
|
const rulesWithLineNo = rules.map((item, index) => ({
|
|
...item,
|
|
// UI-only derived data; keep app context/SWR data immutable
|
|
lineNo: index + 1,
|
|
}))
|
|
|
|
return rulesWithLineNo.filter((item) => match(item.payload ?? ''))
|
|
}, [rules, match])
|
|
|
|
const handleScroll = useCallback((e: Event) => {
|
|
setShowScrollTop((e.target as HTMLElement).scrollTop > 100)
|
|
}, [])
|
|
|
|
const scrollToTop = () => {
|
|
virtuosoRef.current?.scrollTo({ top: 0, behavior: 'smooth' })
|
|
}
|
|
|
|
return (
|
|
<BasePage
|
|
full
|
|
title={t('rules.page.title')}
|
|
contentStyle={{
|
|
height: '100%',
|
|
display: 'flex',
|
|
flexDirection: 'column',
|
|
overflow: 'auto',
|
|
}}
|
|
header={
|
|
<Box display="flex" alignItems="center" gap={1}>
|
|
<ProviderButton />
|
|
</Box>
|
|
}
|
|
>
|
|
<Box
|
|
sx={{
|
|
pt: 1,
|
|
mb: 0.5,
|
|
mx: '10px',
|
|
height: '36px',
|
|
display: 'flex',
|
|
alignItems: 'center',
|
|
}}
|
|
>
|
|
<BaseSearchBox onSearch={(match) => setMatch(() => match)} />
|
|
</Box>
|
|
|
|
{filteredRules && filteredRules.length > 0 ? (
|
|
<>
|
|
<VirtualList
|
|
ref={virtuosoRef}
|
|
count={filteredRules.length}
|
|
estimateSize={40}
|
|
renderItem={(i) => <RuleItem value={filteredRules[i]} />}
|
|
style={{ flex: 1 }}
|
|
onScroll={handleScroll}
|
|
/>
|
|
<ScrollTopButton onClick={scrollToTop} show={showScrollTop} />
|
|
</>
|
|
) : (
|
|
<BaseEmpty />
|
|
)}
|
|
</BasePage>
|
|
)
|
|
}
|
|
|
|
export default RulesPage
|