summaryrefslogtreecommitdiff
path: root/web/source/settings/views/moderation/reports
diff options
context:
space:
mode:
Diffstat (limited to 'web/source/settings/views/moderation/reports')
-rw-r--r--web/source/settings/views/moderation/reports/detail.tsx243
-rw-r--r--web/source/settings/views/moderation/reports/overview.tsx99
-rw-r--r--web/source/settings/views/moderation/reports/username.tsx54
3 files changed, 396 insertions, 0 deletions
diff --git a/web/source/settings/views/moderation/reports/detail.tsx b/web/source/settings/views/moderation/reports/detail.tsx
new file mode 100644
index 000000000..9bb2de6b2
--- /dev/null
+++ b/web/source/settings/views/moderation/reports/detail.tsx
@@ -0,0 +1,243 @@
+/*
+ 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 <http://www.gnu.org/licenses/>.
+*/
+
+import React, { useState } from "react";
+import { useParams } from "wouter";
+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";
+import Username from "./username";
+import { useGetReportQuery, useResolveReportMutation } from "../../../lib/query/admin/reports";
+import { useBaseUrl } from "../../../lib/navigation/util";
+
+export default function ReportDetail({ }) {
+ const baseUrl = useBaseUrl();
+ const params = useParams();
+
+ return (
+ <div className="reports">
+ <h1><BackButton to={`~${baseUrl}`}/> Report Details</h1>
+ <FormWithData
+ dataQuery={useGetReportQuery}
+ queryArg={params.reportId}
+ DataForm={ReportDetailForm}
+ />
+ </div>
+ );
+}
+
+function ReportDetailForm({ data: report }) {
+ const from = report.account;
+ const target = report.target_account;
+
+ return (
+ <div className="report detail">
+ <div className="usernames">
+ <Username user={from} /> reported <Username user={target} />
+ </div>
+
+ {report.action_taken &&
+ <div className="info">
+ <h3>Resolved by @{report.action_taken_by_account.account.acct}</h3>
+ <span className="timestamp">at {new Date(report.action_taken_at).toLocaleString()}</span>
+ <br />
+ <b>Comment: </b><span>{report.action_taken_comment}</span>
+ </div>
+ }
+
+ <div className="info-block">
+ <h3>Report info:</h3>
+ <div className="details">
+ <b>Created: </b>
+ <span>{new Date(report.created_at).toLocaleString()}</span>
+
+ <b>Forwarded: </b> <span>{report.forwarded ? "Yes" : "No"}</span>
+ <b>Category: </b> <span>{report.category}</span>
+
+ <b>Reason: </b>
+ {report.comment.length > 0
+ ? <p>{report.comment}</p>
+ : <i className="no-comment">none provided</i>
+ }
+
+ </div>
+ </div>
+
+ {!report.action_taken && <ReportActionForm report={report} />}
+
+ {
+ report.statuses.length > 0 &&
+ <div className="info-block">
+ <h3>Reported toots ({report.statuses.length}):</h3>
+ <div className="reported-toots">
+ {report.statuses.map((status) => (
+ <ReportedToot key={status.id} toot={status} />
+ ))}
+ </div>
+ </div>
+ }
+ </div>
+ );
+}
+
+function ReportActionForm({ report }) {
+ const form = {
+ id: useValue("id", report.id),
+ comment: useTextInput("action_taken_comment")
+ };
+
+ const [submit, result] = useFormSubmit(form, useResolveReportMutation(), { changedOnly: false });
+
+ return (
+ <form onSubmit={submit} className="info-block">
+ <h3>Resolving this report</h3>
+ <p>
+ An optional comment can be included while resolving this report.
+ Useful for providing an explanation about what action was taken (if any) before the report was marked as resolved.<br />
+ <b>This will be visible to the user that created the report!</b>
+ </p>
+ <TextArea
+ field={form.comment}
+ label="Comment"
+ />
+ <MutationButton
+ disabled={false}
+ label="Resolve"
+ result={result}
+ />
+ </form>
+ );
+}
+
+function ReportedToot({ toot }) {
+ const account = toot.account;
+
+ return (
+ <article className="status expanded">
+ <header className="status-header">
+ <address>
+ <a style={{margin: 0}}>
+ <img className="avatar" src={account.avatar} alt="" />
+ <dl className="author-strap">
+ <dt className="sr-only">Display name</dt>
+ <dd className="displayname text-cutoff">
+ {account.display_name.trim().length > 0 ? account.display_name : account.username}
+ </dd>
+ <dt className="sr-only">Username</dt>
+ <dd className="username text-cutoff">@{account.username}</dd>
+ </dl>
+ </a>
+ </address>
+ </header>
+ <section className="status-body">
+ <div className="text">
+ <div className="content">
+ {toot.spoiler_text?.length > 0
+ ? <TootCW content={toot.content} note={toot.spoiler_text} />
+ : toot.content
+ }
+ </div>
+ </div>
+ {toot.media_attachments?.length > 0 &&
+ <TootMedia media={toot.media_attachments} sensitive={toot.sensitive} />
+ }
+ </section>
+ <aside className="status-info">
+ <dl className="status-stats">
+ <div className="stats-grouping">
+ <div className="stats-item published-at text-cutoff">
+ <dt className="sr-only">Published</dt>
+ <dd>
+ <time dateTime={toot.created_at}>{new Date(toot.created_at).toLocaleString()}</time>
+ </dd>
+ </div>
+ </div>
+ </dl>
+ </aside>
+ </article>
+ );
+}
+
+function TootCW({ note, content }) {
+ const [visible, setVisible] = useState(false);
+
+ function toggleVisible() {
+ setVisible(!visible);
+ }
+
+ return (
+ <>
+ <div className="spoiler">
+ <span>{note}</span>
+ <label className="button spoiler-label" onClick={toggleVisible}>Show {visible ? "less" : "more"}</label>
+ </div>
+ {visible && content}
+ </>
+ );
+}
+
+function TootMedia({ media, sensitive }) {
+ let classes = (media.length % 2 == 0) ? "even" : "odd";
+ if (media.length == 1) {
+ classes += " single";
+ }
+
+ return (
+ <div className={`media photoswipe-gallery ${classes}`}>
+ {media.map((m) => (
+ <div key={m.id} className="media-wrapper">
+ {sensitive && <>
+ <input id={`sensitiveMedia-${m.id}`} type="checkbox" className="sensitive-checkbox hidden" />
+ <div className="sensitive">
+ <div className="open">
+ <label htmlFor={`sensitiveMedia-${m.id}`} className="button" role="button" tabIndex={0}>
+ <i className="fa fa-eye-slash" title="Hide sensitive media"></i>
+ </label>
+ </div>
+ <div className="closed" title={m.description}>
+ <label htmlFor={`sensitiveMedia-${m.id}`} className="button" role="button" tabIndex={0}>
+ Show sensitive media
+ </label>
+ </div>
+ </div>
+ </>}
+ <a
+ href={m.url}
+ title={m.description}
+ target="_blank"
+ rel="noreferrer"
+ data-cropped="true"
+ data-pswp-width={`${m.meta?.original.width}px`}
+ data-pswp-height={`${m.meta?.original.height}px`}
+ >
+ <img
+ alt={m.description}
+ src={m.url}
+ // thumb={m.preview_url}
+ sizes={m.meta?.original}
+ />
+ </a>
+ </div>
+ ))}
+ </div>
+ );
+}
diff --git a/web/source/settings/views/moderation/reports/overview.tsx b/web/source/settings/views/moderation/reports/overview.tsx
new file mode 100644
index 000000000..ca8fc185c
--- /dev/null
+++ b/web/source/settings/views/moderation/reports/overview.tsx
@@ -0,0 +1,99 @@
+/*
+ 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 <http://www.gnu.org/licenses/>.
+*/
+
+import React from "react";
+import { Link } from "wouter";
+
+import FormWithData from "../../../lib/form/form-with-data";
+
+import Username from "./username";
+import { useListReportsQuery } from "../../../lib/query/admin/reports";
+
+export function ReportOverview({ }) {
+ return (
+ <FormWithData
+ dataQuery={useListReportsQuery}
+ DataForm={ReportsList}
+ />
+ );
+}
+
+function ReportsList({ data: reports }) {
+ return (
+ <div className="reports">
+ <div className="form-section-docs">
+ <h1>Reports</h1>
+ <p>
+ Here you can view and resolve reports made to your
+ instance, originating from local and remote users.
+ </p>
+ <a
+ href="https://docs.gotosocial.org/en/latest/admin/settings/#reports"
+ target="_blank"
+ className="docslink"
+ rel="noreferrer"
+ >
+ Learn more about this (opens in a new tab)
+ </a>
+ </div>
+ <div className="list">
+ {reports.map((report) => (
+ <ReportEntry key={report.id} report={report} />
+ ))}
+ </div>
+ </div>
+ );
+}
+
+function ReportEntry({ report }) {
+ const from = report.account;
+ const target = report.target_account;
+
+ let comment = report.comment.length > 200
+ ? report.comment.slice(0, 200) + "..."
+ : report.comment;
+
+ return (
+ <Link
+ to={`/${report.id}`}
+ className="nounderline"
+ >
+ <div className={`report entry${report.action_taken ? " resolved" : ""}`}>
+ <div className="byline">
+ <div className="usernames">
+ <Username user={from} link={false} /> reported <Username user={target} link={false} />
+ </div>
+ <h3 className="report-status">
+ {report.action_taken ? "Resolved" : "Open"}
+ </h3>
+ </div>
+ <div className="details">
+ <b>Created: </b>
+ <span>{new Date(report.created_at).toLocaleString()}</span>
+
+ <b>Reason: </b>
+ {comment.length > 0
+ ? <p>{comment}</p>
+ : <i className="no-comment">none provided</i>
+ }
+ </div>
+ </div>
+ </Link>
+ );
+}
diff --git a/web/source/settings/views/moderation/reports/username.tsx b/web/source/settings/views/moderation/reports/username.tsx
new file mode 100644
index 000000000..6fba0b804
--- /dev/null
+++ b/web/source/settings/views/moderation/reports/username.tsx
@@ -0,0 +1,54 @@
+/*
+ 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 <http://www.gnu.org/licenses/>.
+*/
+
+import React from "react";
+import { Link } from "wouter";
+
+export default function Username({ user, link = true }) {
+ let className = "user";
+ let isLocal = user.domain == null;
+
+ if (user.suspended) {
+ className += " suspended";
+ }
+
+ if (isLocal) {
+ className += " local";
+ }
+
+ let icon = isLocal
+ ? { fa: "fa-home", info: "Local user" }
+ : { fa: "fa-external-link-square", info: "Remote user" };
+
+ let Element: any = "div";
+ let href: any = null;
+
+ if (link) {
+ Element = Link;
+ href = `/settings/admin/accounts/${user.id}`;
+ }
+
+ return (
+ <Element className={className} to={href}>
+ <span className="acct">@{user.account.acct}</span>
+ <i className={`fa fa-fw ${icon.fa}`} aria-hidden="true" title={icon.info} />
+ <span className="sr-only">{icon.info}</span>
+ </Element>
+ );
+}