import { useRef, useState, useCallback, type RefObject, type ReactNode, useMemo, useEffect } from 'react'; import type { FlexProps } from '@chakra-ui/react'; import { Flex, Box, type BoxProps } from '@chakra-ui/react'; import MyIcon from '../components/common/Icon'; import type { IconNameType } from '../components/common/Icon/type'; import { useTranslation } from 'next-i18next'; import { useToast } from './useToast'; import { getErrText } from '@fastgpt/global/common/error/utils'; import { useBoolean, useCreation, useMemoizedFn, useRequest, useScroll, useThrottleEffect } from 'ahooks'; import { type PaginationProps, type PaginationResponse } from '../common/fetch/type'; import MyMenu from '../components/common/MyMenu'; import { useSystem } from './useSystem'; const thresholdVal = 200; export function usePagination( api: (data: PaginationProps) => Promise>, { defaultPageSize = 10, pageSizeOptions: defaultPageSizeOptions, params, defaultRequest = true, type = 'button', onChange, refreshDeps, scrollLoadType = 'bottom', EmptyTip, pollingInterval, pollingWhenHidden = false }: { defaultPageSize?: number; pageSizeOptions?: number[]; params?: DataT; defaultRequest?: boolean; type?: 'button' | 'scroll'; onChange?: (pageNum: number) => void; refreshDeps?: any[]; throttleWait?: number; scrollLoadType?: 'top' | 'bottom'; EmptyTip?: React.JSX.Element; pollingInterval?: number; pollingWhenHidden?: boolean; } ) { const { toast } = useToast(); const { isPc } = useSystem(); const { t } = useTranslation(); const [isLoading, { setTrue, setFalse }] = useBoolean(false); const [pageNum, setPageNum] = useState(1); const [pageSize, setPageSize] = useState(defaultPageSize); const pageSizeOptions = useCreation( () => defaultPageSizeOptions || [10, 20, 50, 100], [defaultPageSizeOptions] ); const [total, setTotal] = useState(0); const [data, setData] = useState([]); const totalDataLength = useMemo(() => Math.max(total, data.length), [total, data.length]); const isEmpty = total === 0 && !isLoading; const noMore = data.length >= totalDataLength; const fetchData = useMemoizedFn( async (num: number = pageNum, ScrollContainerRef?: RefObject) => { if (noMore && num !== 1) return; setTrue(); try { const res = await api({ pageNum: num, pageSize, ...params }); setPageNum(num); res.total !== undefined && setTotal(res.total); if (type === 'scroll') { if (scrollLoadType === 'top') { const prevHeight = ScrollContainerRef?.current?.scrollHeight || 0; const prevScrollTop = ScrollContainerRef?.current?.scrollTop || 0; // 使用 requestAnimationFrame 来调整滚动位置 function adjustScrollPosition() { requestAnimationFrame( ScrollContainerRef?.current ? () => { if (ScrollContainerRef?.current) { const newHeight = ScrollContainerRef.current.scrollHeight; const heightDiff = newHeight - prevHeight; ScrollContainerRef.current.scrollTop = prevScrollTop + heightDiff; } } : adjustScrollPosition ); } setData((prevData) => (num === 1 ? res.list : [...res.list, ...prevData])); adjustScrollPosition(); } else { setData((prevData) => (num === 1 ? res.list : [...prevData, ...res.list])); } } else { setData(res.list); } onChange?.(num); } catch (error: any) { if (error.code !== 'ERR_CANCELED') { toast({ title: getErrText(error, t('common:core.chat.error.data_error')), status: 'error' }); } } setFalse(); } ); // Button pagination const Pagination = useCallback(() => { const maxPage = Math.ceil(totalDataLength / pageSize); const IconButton = ({ icon, isDisabled, onClick, ...props }: { icon: IconNameType; isDisabled?: boolean; onClick: () => void; } & FlexProps) => { isDisabled = isDisabled || isLoading; return ( ); }; return ( {isPc && {t('common:total_num', { num: totalDataLength })}} {isPc && ( fetchData(1)} /> )} fetchData(pageNum - 1)} /> {pageNum} / {maxPage} fetchData(pageNum + 1)} /> {isPc && ( fetchData(maxPage)} /> )} {isPc && ( ({ label: `${item}`, isActive: pageSize === item, onClick: () => setPageSize(item) })) } ]} Button={ {pageSize} / {t('common:page')} } /> )} ); }, [totalDataLength, isPc, pageSize, t, pageNum, pageSizeOptions, isLoading, fetchData]); // Scroll pagination const DefaultRef = useRef(null); const ScrollData = useMemoizedFn( ({ children, ScrollContainerRef, ...props }: { children: ReactNode; ScrollContainerRef?: RefObject; } & BoxProps) => { const ref = ScrollContainerRef || DefaultRef; const loadText = (() => { if (isLoading) return t('common:is_requesting'); if (noMore) return t('common:request_end'); return t('common:request_more'); })(); const scroll = useScroll(ref); // Watch scroll position useThrottleEffect( () => { if (!ref?.current || type !== 'scroll' || noMore || isLoading) return; const { scrollTop, scrollHeight, clientHeight } = ref.current; if ( (scrollLoadType === 'bottom' && scrollTop + clientHeight >= scrollHeight - thresholdVal) || (scrollLoadType === 'top' && scrollTop < thresholdVal) ) { fetchData(pageNum + 1, ref); } }, [scroll, isLoading], { wait: 50 } ); return ( {scrollLoadType === 'top' && total > 0 && isLoading && ( {t('common:is_requesting')} )} {children} {scrollLoadType === 'bottom' && !isEmpty && ( { if (loadText !== t('common:request_more')) return; fetchData(pageNum + 1); }} > {loadText} )} {isEmpty && EmptyTip} ); } ); // Reload data const { runAsync: refresh } = useRequest( async () => { defaultRequest && fetchData(1); }, { manual: false, refreshDeps, throttleWait: 100 } ); useEffect(() => { data.length > 0 && fetchData(); }, [pageSize]); useRequest( async () => { if (!pollingInterval) return; await fetchData(pageNum); }, { pollingInterval, pollingWhenHidden, manual: false, refreshDeps: [pollingInterval] } ); return { pageNum, setPageNum, pageSize, total: totalDataLength, data, setData, isLoading, Pagination, ScrollData, getData: fetchData, refresh }; }