fix(components): add skeleton placeholders to dashboard

This commit is contained in:
diced 2021-09-06 14:54:25 -07:00
parent 5c980c21e5
commit 606821a2c0
No known key found for this signature in database
GPG key ID: 85AB64C74535D76E

View file

@ -1,6 +1,5 @@
import React, { useEffect, useState } from 'react'; import React, { useEffect, useState } from 'react';
import { import {
Paper,
Table, Table,
TableBody, TableBody,
TableCell, TableCell,
@ -11,12 +10,12 @@ import {
Button, Button,
ButtonGroup, ButtonGroup,
Typography, Typography,
Grid Grid,
Skeleton
} from '@material-ui/core'; } from '@material-ui/core';
import Link from 'components/Link'; import Link from 'components/Link';
import Card from 'components/Card'; import Card from 'components/Card';
import Backdrop from 'components/Backdrop';
import useFetch from 'lib/hooks/useFetch'; import useFetch from 'lib/hooks/useFetch';
import { useStoreSelector } from 'lib/redux/store'; import { useStoreSelector } from 'lib/redux/store';
@ -88,18 +87,13 @@ export default function Dashboard() {
const [images, setImages] = useState([]); const [images, setImages] = useState([]);
const [page, setPage] = useState(0); const [page, setPage] = useState(0);
const [stats, setStats] = useState(null); const [stats, setStats] = useState(null);
const [apiLoading, setApiLoading] = useState(true);
const [rowsPerPage, setRowsPerPage] = useState(10); const [rowsPerPage, setRowsPerPage] = useState(10);
const updateImages = async () => { const updateImages = async () => {
setApiLoading(true);
const imgs = await useFetch('/api/user/images'); const imgs = await useFetch('/api/user/images');
const stts = await useFetch('/api/stats'); const stts = await useFetch('/api/stats');
setImages(imgs); setImages(imgs);
setStats(stts); setStats(stts);
setApiLoading(false);
}; };
const handleChangePage = (event, newPage) => { const handleChangePage = (event, newPage) => {
@ -122,36 +116,31 @@ export default function Dashboard() {
return ( return (
<> <>
<Backdrop open={apiLoading} />
<Typography variant='h4'>Welcome back {user?.username}</Typography> <Typography variant='h4'>Welcome back {user?.username}</Typography>
<Typography color='GrayText' pb={2}>You have <b>{images.length}</b> images</Typography> <Typography color='GrayText' pb={2}>You have <b>{images.length ? images.length : '...'}</b> images</Typography>
<Typography variant='h4'>Stats</Typography> <Typography variant='h4'>Stats</Typography>
{stats && (
<Grid container spacing={4} py={2}> <Grid container spacing={4} py={2}>
<Grid item xs={12} sm={4}> <Grid item xs={12} sm={4}>
<Card name='Size' sx={{ height: '100%' }}> <Card name='Size' sx={{ height: '100%' }}>
<StatText>{stats.size}</StatText> <StatText>{stats ? stats.size : <Skeleton variant='text' />}</StatText>
<Typography variant='h3'>Average Size</Typography> <Typography variant='h3'>Average Size</Typography>
<StatText>{bytesToRead(stats.size_num / stats.count)}</StatText> <StatText>{stats ? bytesToRead(stats.size_num / stats.count) : <Skeleton variant='text' />}</StatText>
</Card> </Card>
</Grid> </Grid>
<Grid item xs={12} sm={4}> <Grid item xs={12} sm={4}>
<Card name='Images' sx={{ height: '100%' }}> <Card name='Images' sx={{ height: '100%' }}>
<StatText>{stats.count}</StatText> <StatText>{stats ? stats.count : <Skeleton variant='text' />}</StatText>
<Typography variant='h3'>Views</Typography> <Typography variant='h3'>Views</Typography>
<StatText>{stats.views_count} ({isNaN(stats.views_count / stats.count) ? '0' : stats.views_count / stats.count})</StatText> <StatText>{stats ? `${stats.views_count} (${isNaN(stats.views_count / stats.count) ? '0' : stats.views_count / stats.count})` : <Skeleton variant='text' />}</StatText>
</Card> </Card>
</Grid> </Grid>
<Grid item xs={12} sm={4}> <Grid item xs={12} sm={4}>
<Card name='Users' sx={{ height: '100%' }}> <Card name='Users' sx={{ height: '100%' }}>
<StatText>{stats.count_users}</StatText> <StatText>{stats ? stats.count_users : <Skeleton variant='text' />}</StatText>
</Card> </Card>
</Grid> </Grid>
</Grid> </Grid><Card name='Images' sx={{ my: 2 }} elevation={0} variant='outlined'>
)}
<Card name='Images' sx={{ my: 2 }} elevation={0} variant='outlined'>
<Link href='/dashboard/images' pb={2}>View Gallery</Link> <Link href='/dashboard/images' pb={2}>View Gallery</Link>
<TableContainer sx={{ maxHeight: 440 }}> <TableContainer sx={{ maxHeight: 440 }}>
<Table size='small'> <Table size='small'>
@ -203,32 +192,24 @@ export default function Dashboard() {
rowsPerPage={rowsPerPage} rowsPerPage={rowsPerPage}
page={page} page={page}
onPageChange={handleChangePage} onPageChange={handleChangePage}
onRowsPerPageChange={handleChangeRowsPerPage} onRowsPerPageChange={handleChangeRowsPerPage} />
/>
</Card> </Card>
{stats && (
<>
<Card name='Images per User' sx={{ height: '100%', my: 2 }} elevation={0} variant='outlined'> <Card name='Images per User' sx={{ height: '100%', my: 2 }} elevation={0} variant='outlined'>
<StatTable <StatTable
columns={[ columns={[
{ id: 'username', name: 'Name' }, { id: 'username', name: 'Name' },
{ id: 'count', name: 'Images' } { id: 'count', name: 'Images' }
]} ]}
rows={stats.count_by_user} rows={stats ? stats.count_by_user : []} />
/>
</Card> </Card>
<Card name='Types' sx={{ height: '100%', my: 2 }} elevation={0} variant='outlined'> <Card name='Types' sx={{ height: '100%', my: 2 }} elevation={0} variant='outlined'>
<StatTable <StatTable
columns={[ columns={[
{ id: 'mimetype', name: 'Type' }, { id: 'mimetype', name: 'Type' },
{ id: 'count', name: 'Count' } { id: 'count', name: 'Count' }
]} ]}
rows={stats.types_count} rows={stats ? stats.types_count : []} />
/>
</Card> </Card>
</> </>
)}
</>
); );
} }