2023-02-06 08:33:47 +00:00
|
|
|
/*
|
|
|
|
GoToSocial
|
2023-03-12 17:49:06 +00:00
|
|
|
Copyright (C) GoToSocial Authors admin@gotosocial.org
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-or-later
|
2023-02-06 08:33:47 +00:00
|
|
|
|
|
|
|
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 <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2024-06-18 16:18:00 +00:00
|
|
|
import React from "react";
|
|
|
|
import { useLocation, useParams } from "wouter";
|
2024-04-24 10:12:47 +00:00
|
|
|
import FormWithData from "../../../lib/form/form-with-data";
|
|
|
|
import BackButton from "../../../components/back-button";
|
|
|
|
import { useValue, useTextInput } from "../../../lib/form";
|
|
|
|
import useFormSubmit from "../../../lib/form/submit";
|
|
|
|
import { TextArea } from "../../../components/form/inputs";
|
|
|
|
import MutationButton from "../../../components/form/mutation-button";
|
2024-05-01 13:11:22 +00:00
|
|
|
import Username from "../../../components/username";
|
2024-04-24 10:12:47 +00:00
|
|
|
import { useGetReportQuery, useResolveReportMutation } from "../../../lib/query/admin/reports";
|
|
|
|
import { useBaseUrl } from "../../../lib/navigation/util";
|
2024-11-06 14:55:00 +00:00
|
|
|
import type { AdminReport } from "../../../lib/types/report";
|
2024-06-18 16:18:00 +00:00
|
|
|
import { yesOrNo } from "../../../lib/util";
|
|
|
|
import { Status } from "../../../components/status";
|
2023-02-06 08:33:47 +00:00
|
|
|
|
2024-04-13 11:25:10 +00:00
|
|
|
export default function ReportDetail({ }) {
|
2024-06-18 16:18:00 +00:00
|
|
|
const params: { reportId: string } = useParams();
|
2023-03-29 10:18:45 +00:00
|
|
|
const baseUrl = useBaseUrl();
|
2024-11-06 14:55:00 +00:00
|
|
|
const backLocation: string = history.state?.backLocation ?? `~${baseUrl}`;
|
2024-04-24 10:12:47 +00:00
|
|
|
|
|
|
|
return (
|
2024-06-18 16:18:00 +00:00
|
|
|
<div className="report-detail">
|
|
|
|
<h1><BackButton to={backLocation}/> Report Details</h1>
|
2024-04-24 10:12:47 +00:00
|
|
|
<FormWithData
|
|
|
|
dataQuery={useGetReportQuery}
|
|
|
|
queryArg={params.reportId}
|
|
|
|
DataForm={ReportDetailForm}
|
2024-06-18 16:18:00 +00:00
|
|
|
{...{ backLocation: backLocation }}
|
2024-04-24 10:12:47 +00:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
);
|
2024-04-13 11:25:10 +00:00
|
|
|
}
|
2023-02-06 08:33:47 +00:00
|
|
|
|
2024-06-18 16:18:00 +00:00
|
|
|
function ReportDetailForm({ data: report }: { data: AdminReport }) {
|
|
|
|
const [ location ] = useLocation();
|
|
|
|
const baseUrl = useBaseUrl();
|
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<ReportBasicInfo
|
|
|
|
report={report}
|
|
|
|
baseUrl={baseUrl}
|
|
|
|
location={location}
|
|
|
|
/>
|
|
|
|
|
|
|
|
{ report.action_taken
|
|
|
|
&& <ReportHistory
|
|
|
|
report={report}
|
|
|
|
baseUrl={baseUrl}
|
|
|
|
location={location}
|
|
|
|
/>
|
|
|
|
}
|
|
|
|
|
|
|
|
{ report.statuses &&
|
|
|
|
<ReportStatuses report={report} />
|
|
|
|
}
|
|
|
|
|
|
|
|
{ !report.action_taken &&
|
|
|
|
<ReportActionForm report={report} />
|
|
|
|
}
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
interface ReportSectionProps {
|
|
|
|
report: AdminReport;
|
|
|
|
baseUrl: string;
|
|
|
|
location: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
function ReportBasicInfo({ report, baseUrl, location }: ReportSectionProps) {
|
2023-02-06 08:33:47 +00:00
|
|
|
const from = report.account;
|
|
|
|
const target = report.target_account;
|
2024-06-18 16:18:00 +00:00
|
|
|
const comment = report.comment;
|
|
|
|
const status = report.action_taken ? "Resolved" : "Unresolved";
|
|
|
|
const created = new Date(report.created_at).toLocaleString();
|
2023-02-06 08:33:47 +00:00
|
|
|
|
|
|
|
return (
|
2024-06-18 16:18:00 +00:00
|
|
|
<dl className="info-list overview">
|
|
|
|
<div className="info-list-entry">
|
|
|
|
<dt>Reported account</dt>
|
|
|
|
<dd>
|
|
|
|
<Username
|
|
|
|
account={target}
|
|
|
|
linkTo={`~/settings/moderation/accounts/${target.id}`}
|
|
|
|
backLocation={`~${baseUrl}${location}`}
|
|
|
|
/>
|
|
|
|
</dd>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className="info-list-entry">
|
|
|
|
<dt>Reported by</dt>
|
|
|
|
<dd>
|
|
|
|
<Username
|
|
|
|
account={from}
|
|
|
|
linkTo={`~/settings/moderation/accounts/${from.id}`}
|
|
|
|
backLocation={`~${baseUrl}${location}`}
|
|
|
|
/>
|
|
|
|
</dd>
|
2023-02-06 08:33:47 +00:00
|
|
|
</div>
|
|
|
|
|
2024-06-18 16:18:00 +00:00
|
|
|
<div className="info-list-entry">
|
|
|
|
<dt>Status</dt>
|
|
|
|
<dd>
|
|
|
|
{ report.action_taken
|
|
|
|
? <>{status}</>
|
|
|
|
: <b>{status}</b>
|
|
|
|
}
|
|
|
|
</dd>
|
|
|
|
</div>
|
2023-02-06 08:33:47 +00:00
|
|
|
|
2024-06-18 16:18:00 +00:00
|
|
|
<div className="info-list-entry">
|
|
|
|
<dt>Reason</dt>
|
|
|
|
<dd>
|
|
|
|
{ comment.length > 0
|
|
|
|
? <>{comment}</>
|
|
|
|
: <i>none provided</i>
|
|
|
|
}
|
|
|
|
</dd>
|
|
|
|
</div>
|
2023-02-06 08:33:47 +00:00
|
|
|
|
2024-06-18 16:18:00 +00:00
|
|
|
<div className="info-list-entry">
|
|
|
|
<dt>Created</dt>
|
|
|
|
<dd>
|
|
|
|
<time dateTime={report.created_at}>{created}</time>
|
|
|
|
</dd>
|
|
|
|
</div>
|
2023-02-06 08:33:47 +00:00
|
|
|
|
2024-06-18 16:18:00 +00:00
|
|
|
<div className="info-list-entry">
|
|
|
|
<dt>Category</dt>
|
|
|
|
<dd>{ report.category }</dd>
|
|
|
|
</div>
|
2023-02-06 08:33:47 +00:00
|
|
|
|
2024-06-18 16:18:00 +00:00
|
|
|
<div className="info-list-entry">
|
|
|
|
<dt>Forwarded</dt>
|
|
|
|
<dd>{ yesOrNo(report.forwarded) }</dd>
|
2023-02-06 08:33:47 +00:00
|
|
|
</div>
|
2024-06-18 16:18:00 +00:00
|
|
|
</dl>
|
|
|
|
);
|
|
|
|
}
|
2023-02-06 08:33:47 +00:00
|
|
|
|
2024-06-18 16:18:00 +00:00
|
|
|
function ReportHistory({ report, baseUrl, location }: ReportSectionProps) {
|
|
|
|
const handled_by = report.action_taken_by_account;
|
|
|
|
if (!handled_by) {
|
|
|
|
throw "report handled by action_taken_by_account undefined";
|
|
|
|
}
|
|
|
|
|
|
|
|
const handled = report.action_taken_at ? new Date(report.action_taken_at).toLocaleString() : "never";
|
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<h3>Moderation History</h3>
|
|
|
|
<dl className="info-list">
|
|
|
|
<div className="info-list-entry">
|
|
|
|
<dt>Handled by</dt>
|
|
|
|
<dd>
|
|
|
|
<Username
|
|
|
|
account={handled_by}
|
|
|
|
linkTo={`~/settings/moderation/accounts/${handled_by.id}`}
|
|
|
|
backLocation={`~${baseUrl}${location}`}
|
|
|
|
/>
|
|
|
|
</dd>
|
2023-02-06 08:33:47 +00:00
|
|
|
</div>
|
2024-06-18 16:18:00 +00:00
|
|
|
|
|
|
|
<div className="info-list-entry">
|
|
|
|
<dt>Handled</dt>
|
|
|
|
<dd>
|
|
|
|
<time dateTime={report.action_taken_at}>{handled}</time>
|
|
|
|
</dd>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className="info-list-entry">
|
|
|
|
<dt>Comment</dt>
|
|
|
|
<dd>{ report.action_taken_comment ?? "none"}</dd>
|
|
|
|
</div>
|
|
|
|
</dl>
|
|
|
|
</>
|
2023-02-06 08:33:47 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function ReportActionForm({ report }) {
|
|
|
|
const form = {
|
|
|
|
id: useValue("id", report.id),
|
2024-11-06 14:55:00 +00:00
|
|
|
comment: useTextInput("action_taken_comment"),
|
2023-02-06 08:33:47 +00:00
|
|
|
};
|
|
|
|
|
2023-10-17 10:46:06 +00:00
|
|
|
const [submit, result] = useFormSubmit(form, useResolveReportMutation(), { changedOnly: false });
|
2023-02-06 08:33:47 +00:00
|
|
|
|
|
|
|
return (
|
2024-06-18 16:18:00 +00:00
|
|
|
<form onSubmit={submit}>
|
|
|
|
<h3>Resolve this report</h3>
|
|
|
|
<>
|
2023-02-06 08:33:47 +00:00
|
|
|
An optional comment can be included while resolving this report.
|
2024-06-18 16:18:00 +00:00
|
|
|
This is useful for providing an explanation about what action was
|
|
|
|
taken (if any) before the report was marked as resolved.
|
|
|
|
<br />
|
2024-07-08 07:38:27 +00:00
|
|
|
<div className="info">
|
|
|
|
<i className="fa fa-fw fa-exclamation-triangle" aria-hidden="true"></i>
|
|
|
|
<b>
|
|
|
|
If the report was created by a local account, then any
|
|
|
|
comment made here will be emailed to that account's user!
|
|
|
|
</b>
|
|
|
|
</div>
|
2024-06-18 16:18:00 +00:00
|
|
|
</>
|
2023-02-06 08:33:47 +00:00
|
|
|
<TextArea
|
|
|
|
field={form.comment}
|
|
|
|
label="Comment"
|
2024-07-08 07:38:27 +00:00
|
|
|
autoCapitalize="sentences"
|
2023-02-06 08:33:47 +00:00
|
|
|
/>
|
2024-04-13 11:25:10 +00:00
|
|
|
<MutationButton
|
|
|
|
disabled={false}
|
|
|
|
label="Resolve"
|
|
|
|
result={result}
|
|
|
|
/>
|
2023-02-06 08:33:47 +00:00
|
|
|
</form>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-06-18 16:18:00 +00:00
|
|
|
function ReportStatuses({ report }: { report: AdminReport }) {
|
|
|
|
if (report.statuses.length === 0) {
|
|
|
|
return null;
|
2023-02-06 08:33:47 +00:00
|
|
|
}
|
2024-06-18 16:18:00 +00:00
|
|
|
|
2023-02-06 08:33:47 +00:00
|
|
|
return (
|
2024-06-18 16:18:00 +00:00
|
|
|
<div className="report-statuses">
|
|
|
|
<h3>Reported Statuses</h3>
|
|
|
|
<ul className="thread">
|
|
|
|
{ report.statuses.map((status) => {
|
|
|
|
return (
|
|
|
|
<Status
|
|
|
|
key={status.id}
|
|
|
|
status={status}
|
2024-11-06 14:55:00 +00:00
|
|
|
/>
|
2024-06-18 16:18:00 +00:00
|
|
|
);
|
|
|
|
})}
|
|
|
|
</ul>
|
2023-02-06 08:33:47 +00:00
|
|
|
</div>
|
|
|
|
);
|
2024-04-13 11:25:10 +00:00
|
|
|
}
|