/* GoToSocial Copyright (C) GoToSocial Authors admin@gotosocial.org SPDX-License-Identifier: AGPL-3.0-or-later This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with this program. If not, see . */ import type { ReactNode } from "react"; import React from "react"; import { useLocation } from "wouter"; import { Error } from "./error"; import type { SerializedError } from "@reduxjs/toolkit"; import type { FetchBaseQueryError } from "@reduxjs/toolkit/query"; import type { Links } from "parse-link-header"; import Loading from "./loading"; export interface PageableListProps { isSuccess: boolean; items?: T[]; itemToEntry: (_item: T) => ReactNode; isLoading: boolean; isFetching: boolean; isError: boolean; error: FetchBaseQueryError | SerializedError | undefined; emptyMessage: ReactNode; prevNextLinks?: Links | null | undefined; } export function PageableList({ isLoading, isFetching, isSuccess, items, itemToEntry, isError, error, emptyMessage, prevNextLinks, }: PageableListProps) { const [ location, setLocation ] = useLocation(); if (!(isSuccess || isError)) { // Hasn't been called yet. return null; } if (isLoading || isFetching) { return ; } if (error) { return ; } // Map response to items if possible. let content: ReactNode; if (items == undefined || items.length == 0) { content = {emptyMessage}; } else { content = (
{items.map(item => itemToEntry(item))}
); } // If it's possible to page to next and previous // pages, instantiate button handlers for this. let prevClick: (() => void) | undefined; let nextClick: (() => void) | undefined; if (prevNextLinks) { const prev = prevNextLinks["prev"]; if (prev) { const prevUrl = new URL(prev.url); const prevParams = prevUrl.search; prevClick = () => { setLocation(location + prevParams.toString()); }; } const next = prevNextLinks["next"]; if (next) { const nextUrl = new URL(next.url); const nextParams = nextUrl.search; nextClick = () => { setLocation(location + nextParams.toString()); }; } } return (
{ content } { prevNextLinks &&
{ prevClick && } { nextClick && }
}
); }