import { useInfiniteScroll } from '@vueuse/core'; import { onMounted, Ref, ref, UnwrapRef, watch } from 'vue'; export async function usePaginate(getSingle: (page: number) => Promise): Promise { let hasMore = true; let page = 1; const result: T[] = []; while (hasMore) { // eslint-disable-next-line no-await-in-loop const singleRes = await getSingle(page); result.push(...singleRes); hasMore = singleRes.length !== 0; page += 1; } return result; } export function usePagination( _loadData: (page: number, arg: S) => Promise, isActive: () => boolean = () => true, { scrollElement: _scrollElement, each: _each }: { scrollElement?: Ref; each?: S[] } = {}, ) { const scrollElement = _scrollElement ?? ref(document.getElementById('scroll-component')); const page = ref(1); const pageSize = ref(0); const hasMore = ref(true); const data = ref([]) as Ref; const loading = ref(false); const each = ref(_each ?? []); async function loadData() { if (loading.value === true || hasMore.value === false) { return; } loading.value = true; const newData = (await _loadData(page.value, each.value?.[0] as S)) ?? []; hasMore.value = newData.length >= pageSize.value && newData.length > 0; if (newData.length > 0) { data.value.push(...newData); } // last page and each has more if (!hasMore.value && each.value.length > 0) { // use next each element each.value.shift(); page.value = 1; pageSize.value = 0; hasMore.value = each.value.length > 0; if (hasMore.value) { loading.value = false; await loadData(); } } pageSize.value = newData.length; loading.value = false; } onMounted(loadData); watch(page, loadData); function nextPage() { if (isActive() && !loading.value && hasMore.value) { page.value += 1; } } useInfiniteScroll(scrollElement, nextPage, { distance: 10 }); async function resetPage() { const _page = page.value; hasMore.value = true; data.value = []; each.value = (_each ?? []) as UnwrapRef; page.value = 1; if (_page === 1) { // we need to reload manually as the page is already 1, so changing won't trigger watcher await loadData(); } } return { resetPage, nextPage, data, hasMore, loading }; }