This repository has been archived on 2024-10-29. You can view files and clone it, but cannot push or open issues or pull requests.
website/hooks/usePagination.ts

130 lines
3.7 KiB
TypeScript
Raw Normal View History

import { useState, useRef, useCallback } from 'react'
2022-08-31 20:44:33 +01:00
import type { AxiosInstance } from 'axios'
import type { FetchState } from 'react-component-form'
2022-08-31 20:44:33 +01:00
import type { CacheKey } from '../tools/cache'
import { getPaginationCache, savePaginationCache } from '../tools/cache'
export interface Query {
[key: string]: string
}
export type NextPageAsync = (query?: Query) => Promise<void>
export type NextPage = (query?: Query, callback?: () => void) => void
export interface UsePaginationOptions {
api: AxiosInstance
url: string
inverse?: boolean
cacheKey?: CacheKey
}
export interface PaginationItem {
id: number
}
2022-01-13 18:21:45 +01:00
export type SetItems<T> = React.Dispatch<React.SetStateAction<T[]>>
export interface UsePaginationResult<T> {
items: T[]
nextPage: NextPage
resetPagination: () => void
hasMore: boolean
2022-01-13 18:21:45 +01:00
setItems: SetItems<T>
}
export const usePagination = <T extends PaginationItem>(
options: UsePaginationOptions
): UsePaginationResult<T> => {
const { api, url, inverse = false, cacheKey } = options
const [items, setItems] = useState<T[]>([])
const [hasMore, setHasMore] = useState(true)
const fetchState = useRef<FetchState>('idle')
const afterId = useRef<number | null>(null)
const nextPageAsync: NextPageAsync = useCallback(
async (query) => {
if (fetchState.current !== 'idle') {
return
}
fetchState.current = 'loading'
const searchParameters = new URLSearchParams(query)
searchParameters.append('limit', '20')
if (afterId.current != null) {
searchParameters.append('after', afterId.current.toString())
}
const { data: newItems } = await api.get<T[]>(
`${url}?${searchParameters.toString()}`
)
2022-08-30 21:30:06 +02:00
if (!inverse) {
2023-01-11 17:39:09 +01:00
const endIndex = newItems.length - 1
const lastItem = newItems[endIndex]
2022-08-30 21:30:06 +02:00
afterId.current =
2023-01-11 17:39:09 +01:00
newItems.length > 0 && lastItem != null ? lastItem.id : null
2022-08-30 21:30:06 +02:00
} else {
2023-01-11 17:39:09 +01:00
afterId.current =
newItems.length > 0 && newItems[0] != null ? newItems[0].id : null
2022-08-30 21:30:06 +02:00
}
setItems((oldItems) => {
const updatedItems = inverse
? [...newItems, ...oldItems]
: [...oldItems, ...newItems]
2022-08-30 21:30:06 +02:00
const unique = updatedItems.reduce<T[]>((accumulator, item) => {
2022-08-31 20:44:33 +01:00
const isExisting = accumulator.some((itemSome) => {
return itemSome.id === item.id
})
2022-08-30 21:30:06 +02:00
if (!isExisting) {
accumulator.push(item)
}
return accumulator
}, [])
if (cacheKey != null) {
2022-08-30 21:30:06 +02:00
savePaginationCache(cacheKey, unique)
}
2022-08-30 21:30:06 +02:00
return unique
})
setHasMore(newItems.length > 0)
fetchState.current = 'idle'
},
[api, url, inverse, cacheKey]
)
const nextPage: NextPage = useCallback(
(query, callback) => {
nextPageAsync(query)
.then(() => {
if (callback != null) {
callback()
}
})
.catch((error) => {
console.error(error)
})
},
[nextPageAsync]
)
const resetPagination = useCallback((): void => {
if (cacheKey == null) {
afterId.current = null
setItems([])
} else {
2022-08-30 21:30:06 +02:00
fetchState.current = 'loading'
const newItems = getPaginationCache<T>(cacheKey)
setItems(newItems)
2022-08-30 21:30:06 +02:00
if (!inverse) {
2023-01-11 17:39:09 +01:00
const endIndex = newItems.length - 1
const lastItem = newItems[endIndex]
2022-08-30 21:30:06 +02:00
afterId.current =
2023-01-11 17:39:09 +01:00
newItems.length > 0 && lastItem != null ? lastItem.id : null
2022-08-30 21:30:06 +02:00
} else {
2023-01-11 17:39:09 +01:00
afterId.current =
newItems.length > 0 && newItems[0] != null ? newItems[0].id : null
2022-08-30 21:30:06 +02:00
}
fetchState.current = 'idle'
}
2022-08-30 21:30:06 +02:00
}, [cacheKey, inverse])
2022-01-13 18:21:45 +01:00
return { items, hasMore, nextPage, resetPagination, setItems }
}