diff options
| -rw-r--r-- | web/source/settings/admin/reports/detail.jsx | 234 | ||||
| -rw-r--r-- | web/source/settings/admin/reports/index.jsx | 112 | ||||
| -rw-r--r-- | web/source/settings/admin/reports/username.jsx | 54 | ||||
| -rw-r--r-- | web/source/settings/index.js | 1 | ||||
| -rw-r--r-- | web/source/settings/lib/query/admin/index.js | 3 | ||||
| -rw-r--r-- | web/source/settings/lib/query/admin/reports.js | 52 | ||||
| -rw-r--r-- | web/source/settings/lib/query/base.js | 2 | ||||
| -rw-r--r-- | web/source/settings/style.css | 119 | 
8 files changed, 575 insertions, 2 deletions
| diff --git a/web/source/settings/admin/reports/detail.jsx b/web/source/settings/admin/reports/detail.jsx new file mode 100644 index 000000000..20548a720 --- /dev/null +++ b/web/source/settings/admin/reports/detail.jsx @@ -0,0 +1,234 @@ +/* +	GoToSocial +	Copyright (C) 2021-2023 GoToSocial Authors admin@gotosocial.org + +	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/>. +*/ + +"use strict"; + +const React = require("react"); +const { useRoute, Redirect } = require("wouter"); + +const query = require("../../lib/query"); + +const FormWithData = require("../../lib/form/form-with-data"); +const BackButton = require("../../components/back-button"); + +const { useValue, useTextInput } = require("../../lib/form"); +const useFormSubmit = require("../../lib/form/submit"); + +const { TextArea } = require("../../components/form/inputs"); + +const MutationButton = require("../../components/form/mutation-button"); +const Username = require("./username"); + +module.exports = function ReportDetail({ baseUrl }) { +	let [_match, params] = useRoute(`${baseUrl}/:reportId`); +	if (params?.reportId == undefined) { +		return <Redirect to={baseUrl} />; +	} else { +		return ( +			<div className="report-detail"> +				<h1> +					<BackButton to={baseUrl} /> Report Details +				</h1> +				<FormWithData +					dataQuery={query.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> +				<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, query.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 label="Resolve" result={result} /> +		</form> +	); +} + +function ReportedToot({ toot }) { +	const account = toot.account; + +	return ( +		<div className="toot expanded"> +			<div className="contentgrid"> +				<span className="avatar"> +					<img src={account.avatar} alt="" /> +				</span> +				<span className="displayname">{account.display_name.trim().length > 0 ? account.display_name : account.username}</span> +				<span className="username">@{account.username}</span> +				<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} /> +				} +			</div> +			<div className="toot-info"> +				<a +					href={toot.url} +					target="_blank" +					rel="noreferrer" +				>{new Date(toot.created_at).toLocaleString()}</a> +			</div> +		</div> +	); +} + +function TootCW({ note, content }) { +	const [visible, setVisible] = React.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} +							size={m.meta?.original} +							type={m.type} +						/> +					</a> +				</div> +			))} +		</div> +	); +}
\ No newline at end of file diff --git a/web/source/settings/admin/reports/index.jsx b/web/source/settings/admin/reports/index.jsx new file mode 100644 index 000000000..61483b0d6 --- /dev/null +++ b/web/source/settings/admin/reports/index.jsx @@ -0,0 +1,112 @@ +/* +	GoToSocial +	Copyright (C) 2021-2023 GoToSocial Authors admin@gotosocial.org + +	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/>. +*/ + +"use strict"; + +const React = require("react"); +const { Link, Switch, Route } = require("wouter"); + +const query = require("../../lib/query"); + +const FormWithData = require("../../lib/form/form-with-data"); + +const ReportDetail = require("./detail"); +const Username = require("./username"); + +const baseUrl = "/settings/admin/reports"; + +module.exports = function Reports() { +	return ( +		<div className="reports"> +			<Switch> +				<Route path={`${baseUrl}/:reportId`}> +					<ReportDetail baseUrl={baseUrl} /> +				</Route> +				<ReportOverview baseUrl={baseUrl} /> +			</Switch> +		</div> +	); +}; + +function ReportOverview({ _baseUrl }) { +	return ( +		<> +			<h1>Reports</h1> +			<div> +				<div className="info"> +					<i className="fa fa-fw fa-exclamation-triangle" aria-hidden="true"></i> +					<p> +						<b>This interface is currently very limited</b>, only providing a basic overview. <br /> +						Work is in progress on a more full-fledged moderation experience. +					</p> +				</div> +				<p> +					Here you can view and resolve reports made to your instance, originating from local and remote users. +				</p> +			</div> +			<FormWithData +				dataQuery={query.useListReportsQuery} +				DataForm={ReportsList} +			/> +		</> +	); +} + +function ReportsList({ data: reports }) { +	return ( +		<div className="list"> +			{reports.map((report) => ( +				<ReportEntry key={report.id} report={report} /> +			))} +		</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={`${baseUrl}/${report.id}`}> +			<a className={`report entry${report.action_taken ? " resolved" : ""}`}> +				<div className="byline"> +					<div className="users"> +						<Username user={from} link={false} /> reported <Username user={target} link={false} /> +					</div> +					<h3 className="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> +			</a> +		</Link> +	); +}
\ No newline at end of file diff --git a/web/source/settings/admin/reports/username.jsx b/web/source/settings/admin/reports/username.jsx new file mode 100644 index 000000000..5eeea4310 --- /dev/null +++ b/web/source/settings/admin/reports/username.jsx @@ -0,0 +1,54 @@ +/* +	GoToSocial +	Copyright (C) 2021-2023 GoToSocial Authors admin@gotosocial.org + +	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/>. +*/ + +"use strict"; + +const React = require("react"); + +module.exports = 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 = "span"; +	let href = null; + +	if (link) { +		Element = "a"; +		href = user.account.url; +	} + +	return ( +		<Element className={className} href={href} target="_blank" rel="noreferrer" > +			@{user.account.acct} +			<i className={`fa fa-fw ${icon.fa}`} aria-hidden="true" title={icon.info} /> +			<span className="sr-only">{icon.info}</span> +		</Element> +	); +};
\ No newline at end of file diff --git a/web/source/settings/index.js b/web/source/settings/index.js index 6de72581e..812fff6b4 100644 --- a/web/source/settings/index.js +++ b/web/source/settings/index.js @@ -43,6 +43,7 @@ const nav = {  		"Instance Settings": require("./admin/settings.js"),  		"Actions": require("./admin/actions"),  		"Federation": require("./admin/federation"), +		"Reports": require("./admin/reports")  	},  	"Custom Emoji": {  		adminOnly: true, diff --git a/web/source/settings/lib/query/admin/index.js b/web/source/settings/lib/query/admin/index.js index 33e14521e..c348a2701 100644 --- a/web/source/settings/lib/query/admin/index.js +++ b/web/source/settings/lib/query/admin/index.js @@ -78,7 +78,8 @@ const endpoints = (build) => ({  		})  	}),  	...require("./import-export")(build), -	...require("./custom-emoji")(build) +	...require("./custom-emoji")(build), +	...require("./reports")(build)  });  module.exports = base.injectEndpoints({ endpoints });
\ No newline at end of file diff --git a/web/source/settings/lib/query/admin/reports.js b/web/source/settings/lib/query/admin/reports.js new file mode 100644 index 000000000..96fec8ae2 --- /dev/null +++ b/web/source/settings/lib/query/admin/reports.js @@ -0,0 +1,52 @@ +/* +	GoToSocial +	Copyright (C) 2021-2023 GoToSocial Authors admin@gotosocial.org + +	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/>. +*/ + +"use strict"; + +module.exports = (build) => ({ +	listReports: build.query({ +		query: (params = {}) => ({ +			url: "/api/v1/admin/reports", +			params: { +				limit: 100, +				...params +			} +		}), +		providesTags: ["Reports"] +	}), + +	getReport: build.query({ +		query: (id) => ({ +			url: `/api/v1/admin/reports/${id}` +		}), +		providesTags: (res, error, id) => [{ type: "Reports", id }] +	}), + +	resolveReport: build.mutation({ +		query: (formData) => ({ +			url: `/api/v1/admin/reports/${formData.id}/resolve`, +			method: "POST", +			asForm: true, +			body: formData +		}), +		invalidatesTags: (res) => +			res +				? [{ type: "Reports", id: "LIST" }, { type: "Reports", id: res.id }] +				: [{ type: "Reports", id: "LIST" }] +	}) +});
\ No newline at end of file diff --git a/web/source/settings/lib/query/base.js b/web/source/settings/lib/query/base.js index d9d62d356..4020616f2 100644 --- a/web/source/settings/lib/query/base.js +++ b/web/source/settings/lib/query/base.js @@ -72,7 +72,7 @@ function instanceBasedQuery(args, api, extraOptions) {  module.exports = createApi({  	reducerPath: "api",  	baseQuery: instanceBasedQuery, -	tagTypes: ["Auth", "Emoji"], +	tagTypes: ["Auth", "Emoji", "Reports"],  	endpoints: (build) => ({  		instance: build.query({  			query: () => ({ diff --git a/web/source/settings/style.css b/web/source/settings/style.css index 7affd8269..17ccec6d3 100644 --- a/web/source/settings/style.css +++ b/web/source/settings/style.css @@ -663,6 +663,10 @@ span.form-info {  	a {  		color: $info-link;  	} + +	p { +		margin-top: 0; +	}  }  button.with-icon { @@ -805,6 +809,121 @@ button.with-padding {  	}  } +.reports { +	p { +		margin: 0; +	} + +	.report { +		display: flex; +		flex-direction: column; +		gap: 0.5rem; +		margin: 0.5rem 0; + +		text-decoration: none; +		color: $fg; + +		padding: 1rem; + +		border: none; +		border-left: 0.3rem solid $border-accent; + +		.byline { +			display: grid; +			grid-template-columns: 1fr auto; + +			.status { +				color: $border-accent; +			} +		} + +		.details { +			display: grid; +			grid-template-columns: auto 1fr; +			gap: 0.2rem 0.5rem; +			padding: 0.5rem; + +			justify-items: start; +		} + +		h3 { +			margin: 0; +		} + +		&.resolved { +			color: $fg-reduced; +			border-left: 0.4rem solid $bg; + +			.byline .status { +				color: $fg-reduced; +			} +			 +			.user { +				opacity: 0.8; +			} +		} + +		&.detail { +			border: none; +			padding: 0; +		} +	} + +	.report.detail { +		display: flex; +		flex-direction: column; +		margin-top: 1rem; +		gap: 1rem; + +		.info-block { +			padding: 0.5rem; +			background: $gray2; +		} + +		.info { +			display: block; +		} + +		.reported-toots { +			margin-top: 0.5rem; +		} + +		.toot .toot-info { +			padding: 0.5rem; +			background: $toot-info-bg; + +			a { +				color: $fg-reduced; +			} + +			&:last-child { +				border-bottom-left-radius: $br; +				border-bottom-right-radius: $br; +			} +		} +	} + +	.user { +		background: $fg-accent; +		color: $bg; +		border-radius: $br; +		padding: 0.1rem 0.2rem; +		margin: 0 0.1rem; +		font-weight: bold; +		text-decoration: none; + +		&.suspended { +			background: $bg-accent; +			color: $fg; +			text-decoration: line-through; +		} + +		&.local { +			background: $green1; +		} +	} +} +  [role="button"] {  	cursor: pointer;  } | 
