mirror of
https://github.com/ajnart/homarr.git
synced 2025-11-08 14:35:49 +01:00
138 lines
3.6 KiB
TypeScript
138 lines
3.6 KiB
TypeScript
import {
|
|
Alert,
|
|
Center,
|
|
Code,
|
|
Group,
|
|
Pagination,
|
|
Skeleton,
|
|
Stack,
|
|
Table,
|
|
Text,
|
|
Title,
|
|
Tooltip,
|
|
} from '@mantine/core';
|
|
import { useElementSize } from '@mantine/hooks';
|
|
import { IconAlertCircle } from '@tabler/icons-react';
|
|
import dayjs from 'dayjs';
|
|
import duration from 'dayjs/plugin/duration';
|
|
import { useTranslation } from 'next-i18next';
|
|
import { FunctionComponent, useState } from 'react';
|
|
|
|
import { useGetUsenetHistory } from '../dashDot/api';
|
|
import { parseDuration } from '~/tools/client/parseDuration';
|
|
import { humanFileSize } from '~/tools/humanFileSize';
|
|
|
|
dayjs.extend(duration);
|
|
|
|
interface UsenetHistoryListProps {
|
|
appId: string;
|
|
}
|
|
|
|
const PAGE_SIZE = 13;
|
|
|
|
export const UsenetHistoryList: FunctionComponent<UsenetHistoryListProps> = ({ appId }) => {
|
|
const [page, setPage] = useState(1);
|
|
const { t } = useTranslation(['modules/usenet', 'common']);
|
|
|
|
const { ref, width, height } = useElementSize();
|
|
const durationBreakpoint = 400;
|
|
const { data, isLoading, isError, error } = useGetUsenetHistory({
|
|
limit: PAGE_SIZE,
|
|
offset: (page - 1) * PAGE_SIZE,
|
|
appId,
|
|
});
|
|
const totalPages = Math.ceil((data?.total || 1) / PAGE_SIZE);
|
|
|
|
if (isLoading) {
|
|
return (
|
|
<>
|
|
<Skeleton height={40} mt={10} />
|
|
<Skeleton height={40} mt={10} />
|
|
<Skeleton height={40} mt={10} />
|
|
</>
|
|
);
|
|
}
|
|
|
|
if (isError) {
|
|
return (
|
|
<Group position="center">
|
|
<Alert
|
|
icon={<IconAlertCircle size={16} />}
|
|
my="lg"
|
|
title={t('modules/usenet:history.error.title')}
|
|
color="red"
|
|
radius="md"
|
|
>
|
|
{t('modules/usenet:history.error.message')}
|
|
<Code mt="sm" block>
|
|
{error.message}
|
|
</Code>
|
|
</Alert>
|
|
</Group>
|
|
);
|
|
}
|
|
|
|
if (!data || data.items.length <= 0) {
|
|
return (
|
|
<Center style={{ display: 'flex', alignItems: 'center', justifyContent: 'center' }}>
|
|
<Title order={3}>{t('modules/usenet:history.empty')}</Title>
|
|
</Center>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<Stack justify="space-around" spacing="xs">
|
|
<Table highlightOnHover style={{ tableLayout: 'fixed' }} ref={ref}>
|
|
<thead>
|
|
<tr>
|
|
<th>{t('modules/usenet:history.header.name')}</th>
|
|
<th style={{ width: 100 }}>{t('modules/usenet:history.header.size')}</th>
|
|
{durationBreakpoint < width ? (
|
|
<th style={{ width: 200 }}>{t('modules/usenet:history.header.duration')}</th>
|
|
) : null}
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{data.items.map((history) => (
|
|
<tr key={history.id}>
|
|
<td>
|
|
<Tooltip position="top" label={history.name}>
|
|
<Text
|
|
size="xs"
|
|
style={{
|
|
whiteSpace: 'nowrap',
|
|
overflow: 'hidden',
|
|
textOverflow: 'ellipsis',
|
|
}}
|
|
>
|
|
{history.name}
|
|
</Text>
|
|
</Tooltip>
|
|
</td>
|
|
<td>
|
|
<Text size="xs">{humanFileSize(history.size)}</Text>
|
|
</td>
|
|
{durationBreakpoint < width ? (
|
|
<td>
|
|
<Text size="xs">{parseDuration(history.time, t)}</Text>
|
|
</td>
|
|
) : null}
|
|
</tr>
|
|
))}
|
|
</tbody>
|
|
</Table>
|
|
{totalPages > 1 && (
|
|
<Pagination
|
|
noWrap
|
|
size="sm"
|
|
position="center"
|
|
mt="md"
|
|
total={totalPages}
|
|
value={page}
|
|
onChange={setPage}
|
|
/>
|
|
)}
|
|
</Stack>
|
|
);
|
|
};
|