Skip to content

Commit

Permalink
Added ability to delete reports.
Browse files Browse the repository at this point in the history
  • Loading branch information
mikecao committed Jul 30, 2023
1 parent ad710cc commit 95d8245
Show file tree
Hide file tree
Showing 8 changed files with 158 additions and 39 deletions.
29 changes: 29 additions & 0 deletions components/common/ConfirmDeleteForm.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
import { useState } from 'react';
import { Button, LoadingButton, Form, FormButtons } from 'react-basics';
import useMessages from 'hooks/useMessages';

export function ConfirmDeleteForm({ name, onConfirm, onClose }) {
const [loading, setLoading] = useState(false);
const { formatMessage, labels, messages, FormattedMessage } = useMessages();

const handleConfirm = () => {
setLoading(true);
onConfirm();
};

return (
<Form>
<p>
<FormattedMessage {...messages.confirmDelete} values={{ target: <b>{name}</b> }} />
</p>
<FormButtons flex>
<LoadingButton loading={loading} onClick={handleConfirm} variant="danger">
{formatMessage(labels.delete)}
</LoadingButton>
<Button onClick={onClose}>{formatMessage(labels.cancel)}</Button>
</FormButtons>
</Form>
);
}

export default ConfirmDeleteForm;
12 changes: 12 additions & 0 deletions components/common/LinkButton.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
import Link from 'next/link';
import { Icon, Icons, Text } from 'react-basics';
import styles from './LinkButton.module.css';

export default function LinkButton({ href, icon, children }) {
return (
<Link className={styles.button} href={href}>
<Icon>{icon || <Icons.ArrowRight />}</Icon>
<Text>{children}</Text>
</Link>
);
}
28 changes: 28 additions & 0 deletions components/common/LinkButton.module.css
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
.button {
display: flex;
align-items: center;
align-self: flex-start;
white-space: nowrap;
gap: var(--size200);
font-family: inherit;
color: var(--base900);
background: var(--base100);
border: 1px solid transparent;
border-radius: var(--border-radius);
min-height: var(--base-height);
padding: 0 var(--size600);
position: relative;
cursor: pointer;
}

.button:hover {
background: var(--base200);
}

.button:active {
background: var(--base300);
}

.button:visited {
color: var(--base900);
}
57 changes: 38 additions & 19 deletions components/pages/reports/ReportsTable.js
Original file line number Diff line number Diff line change
@@ -1,9 +1,12 @@
import Link from 'next/link';
import { Button, Text, Icon, Icons } from 'react-basics';
import { useState } from 'react';
import { Flexbox, Icon, Icons, Text, Button, Modal } from 'react-basics';
import LinkButton from 'components/common/LinkButton';
import SettingsTable from 'components/common/SettingsTable';
import useMessages from 'hooks/useMessages';
import ConfirmDeleteForm from 'components/common/ConfirmDeleteForm';
import { useMessages } from 'hooks';

export function ReportsTable({ data = [] }) {
export function ReportsTable({ data = [], onDelete = () => {} }) {
const [report, setReport] = useState(null);
const { formatMessage, labels } = useMessages();

const columns = [
Expand All @@ -13,23 +16,39 @@ export function ReportsTable({ data = [] }) {
{ name: 'action', label: ' ' },
];

const handleConfirm = () => {
onDelete(report.id);
};

return (
<SettingsTable columns={columns} data={data}>
{row => {
const { id } = row;
<>
<SettingsTable columns={columns} data={data}>
{row => {
const { id } = row;

return (
<Link href={`/reports/${id}`}>
<Button>
<Icon>
<Icons.ArrowRight />
</Icon>
<Text>{formatMessage(labels.view)}</Text>
</Button>
</Link>
);
}}
</SettingsTable>
return (
<Flexbox gap={10}>
<LinkButton href={`/reports/${id}`}>{formatMessage(labels.view)}</LinkButton>
<Button onClick={() => setReport(row)}>
<Icon>
<Icons.Trash />
</Icon>
<Text>{formatMessage(labels.delete)}</Text>
</Button>
</Flexbox>
);
}}
</SettingsTable>
{report && (
<Modal>
<ConfirmDeleteForm
name={report.name}
onConfirm={handleConfirm}
onClose={() => setReport(null)}
/>
</Modal>
)}
</>
);
}

Expand Down
8 changes: 6 additions & 2 deletions components/pages/websites/WebsiteReportsPage.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,11 @@ import WebsiteHeader from './WebsiteHeader';

export function WebsiteReportsPage({ websiteId }) {
const { formatMessage, labels } = useMessages();
const { reports, error, isLoading } = useReports(websiteId);
const { reports, error, isLoading, deleteReport } = useReports(websiteId);

const handleDelete = async id => {
await deleteReport(id);
};

return (
<Page loading={isLoading} error={error}>
Expand All @@ -22,7 +26,7 @@ export function WebsiteReportsPage({ websiteId }) {
</Button>
</Link>
</Flexbox>
<ReportsTable websiteId={websiteId} data={reports} />
<ReportsTable data={reports} onDelete={handleDelete} />
</Page>
);
}
Expand Down
19 changes: 16 additions & 3 deletions hooks/useReports.js
Original file line number Diff line number Diff line change
@@ -1,10 +1,23 @@
import { useState } from 'react';
import useApi from './useApi';

export function useReports(websiteId) {
const { get, useQuery } = useApi();
const { data, error, isLoading } = useQuery(['reports'], () => get(`/reports`, { websiteId }));
const [modified, setModified] = useState(Date.now());
const { get, useQuery, del, useMutation } = useApi();
const { mutate } = useMutation(reportId => del(`/reports/${reportId}`));
const { data, error, isLoading } = useQuery(['reports:website', { websiteId, modified }], () =>
get(`/reports`, { websiteId }),
);

return { reports: data, error, isLoading };
const deleteReport = id => {
mutate(id, {
onSuccess: () => {
setModified(Date.now());
},
});
};

return { reports: data, error, isLoading, deleteReport };
}

export default useReports;
4 changes: 4 additions & 0 deletions lib/auth.ts
Original file line number Diff line number Diff line change
Expand Up @@ -105,6 +105,10 @@ export async function canUpdateReport({ user }: Auth, report: Report) {
return user.id == report.userId;
}

export async function canDeleteReport(auth: Auth, report: Report) {
return canUpdateReport(auth, report);
}

export async function canCreateTeam({ user }: Auth) {
if (user.isAdmin) {
return true;
Expand Down
40 changes: 25 additions & 15 deletions pages/api/reports/[id].ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
import { canUpdateReport, canViewReport } from 'lib/auth';
import { canUpdateReport, canViewReport, canDeleteReport } from 'lib/auth';
import { useAuth, useCors } from 'lib/middleware';
import { NextApiRequestQueryBody } from 'lib/types';
import { NextApiResponse } from 'next';
import { methodNotAllowed, ok, unauthorized } from 'next-basics';
import { getReportById, updateReport } from 'queries';
import { getReportById, updateReport, deleteReport } from 'queries';

export interface ReportRequestQuery {
id: string;
Expand All @@ -24,31 +24,29 @@ export default async (
await useCors(req, res);
await useAuth(req, res);

if (req.method === 'GET') {
const { id: reportId } = req.query;
const { id: reportId } = req.query;
const {
user: { id: userId },
} = req.auth;

const data = await getReportById(reportId);
if (req.method === 'GET') {
const report = await getReportById(reportId);

if (!(await canViewReport(req.auth, data))) {
if (!(await canViewReport(req.auth, report))) {
return unauthorized(res);
}

data.parameters = JSON.parse(data.parameters);
report.parameters = JSON.parse(report.parameters);

return ok(res, data);
return ok(res, report);
}

if (req.method === 'POST') {
const { id: reportId } = req.query;
const {
user: { id: userId },
} = req.auth;

const { websiteId, type, name, description, parameters } = req.body;

const data = await getReportById(reportId);
const report = await getReportById(reportId);

if (!(await canUpdateReport(req.auth, data))) {
if (!(await canUpdateReport(req.auth, report))) {
return unauthorized(res);
}

Expand All @@ -64,5 +62,17 @@ export default async (
return ok(res, result);
}

if (req.method === 'DELETE') {
const report = await getReportById(reportId);

if (!(await canDeleteReport(req.auth, report))) {
return unauthorized(res);
}

await deleteReport(reportId);

return ok(res);
}

return methodNotAllowed(res);
};

0 comments on commit 95d8245

Please sign in to comment.