2021-07-24 17:01:50 +01:00
|
|
|
import { XCircle } from "@styled-icons/boxicons-regular";
|
2021-07-29 19:01:40 +01:00
|
|
|
import { observer } from "mobx-react-lite";
|
2021-09-25 14:43:28 +01:00
|
|
|
import { Virtuoso } from "react-virtuoso";
|
2022-04-09 15:47:04 +01:00
|
|
|
import { API } from "revolt.js";
|
2022-04-27 20:18:51 +01:00
|
|
|
import { Server } from "revolt.js";
|
2021-07-05 11:23:23 +01:00
|
|
|
|
2021-07-24 17:01:50 +01:00
|
|
|
import styles from "./Panes.module.scss";
|
|
|
|
import { Text } from "preact-i18n";
|
2022-05-08 10:40:45 +01:00
|
|
|
import { useEffect, useMemo, useState } from "preact/hooks";
|
2021-06-19 22:37:12 +01:00
|
|
|
|
2021-07-24 17:01:50 +01:00
|
|
|
import UserIcon from "../../../components/common/user/UserIcon";
|
|
|
|
import IconButton from "../../../components/ui/IconButton";
|
|
|
|
import Preloader from "../../../components/ui/Preloader";
|
2022-05-08 10:40:45 +01:00
|
|
|
import { InputBox } from "@revoltchat/ui";
|
2021-07-05 11:23:23 +01:00
|
|
|
|
2021-09-25 14:43:28 +01:00
|
|
|
interface InnerProps {
|
2022-04-09 15:47:04 +01:00
|
|
|
ban: API.ServerBan;
|
2022-05-08 10:40:45 +01:00
|
|
|
users: Record<string, API.BannedUser>;
|
2021-09-25 14:43:28 +01:00
|
|
|
server: Server;
|
|
|
|
removeSelf: () => void;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Inner = observer(({ ban, users, server, removeSelf }: InnerProps) => {
|
|
|
|
const [deleting, setDelete] = useState(false);
|
2022-05-08 10:40:45 +01:00
|
|
|
const user = users[ban._id.user];
|
2021-09-25 14:43:28 +01:00
|
|
|
|
|
|
|
return (
|
|
|
|
<div className={styles.ban} data-deleting={deleting}>
|
|
|
|
<span>
|
|
|
|
<UserIcon attachment={user?.avatar ?? undefined} size={24} />{" "}
|
|
|
|
{user?.username}
|
|
|
|
</span>
|
|
|
|
<div className={styles.reason}>
|
|
|
|
{ban.reason ?? (
|
|
|
|
<Text id="app.settings.server_pages.bans.no_reason" />
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
<IconButton
|
|
|
|
onClick={() => {
|
|
|
|
setDelete(true);
|
|
|
|
server.unbanUser(ban._id.user).then(removeSelf);
|
|
|
|
}}
|
|
|
|
disabled={deleting}>
|
|
|
|
<XCircle size={24} />
|
|
|
|
</IconButton>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2021-06-19 22:37:12 +01:00
|
|
|
interface Props {
|
2021-07-29 19:01:40 +01:00
|
|
|
server: Server;
|
2021-06-19 22:37:12 +01:00
|
|
|
}
|
|
|
|
|
2021-07-29 19:01:40 +01:00
|
|
|
export const Bans = observer(({ server }: Props) => {
|
2022-05-08 10:40:45 +01:00
|
|
|
const [query, setQuery] = useState("");
|
|
|
|
const [result, setData] = useState<
|
|
|
|
| {
|
|
|
|
users: Record<string, API.BannedUser>;
|
|
|
|
bans: API.BanListResult["bans"];
|
|
|
|
}
|
|
|
|
| undefined
|
|
|
|
>(undefined);
|
2021-06-19 22:37:12 +01:00
|
|
|
|
2021-07-05 11:25:20 +01:00
|
|
|
useEffect(() => {
|
2022-05-08 10:40:45 +01:00
|
|
|
server
|
|
|
|
.fetchBans()
|
|
|
|
.then((data) => {
|
|
|
|
const users: Record<string, API.BannedUser> = {};
|
|
|
|
for (const user of data.users) {
|
|
|
|
users[user._id] = user;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
users,
|
|
|
|
bans: data.bans,
|
|
|
|
};
|
|
|
|
})
|
|
|
|
.then(setData);
|
2021-08-05 14:47:00 +01:00
|
|
|
}, [server, setData]);
|
2021-06-19 22:37:12 +01:00
|
|
|
|
2022-05-08 10:40:45 +01:00
|
|
|
const bans = useMemo(() => {
|
|
|
|
if (!result) return;
|
|
|
|
|
|
|
|
if (query)
|
|
|
|
return result.bans.filter(({ _id }) =>
|
|
|
|
result.users[_id.user]?.username.includes(query),
|
|
|
|
);
|
|
|
|
|
|
|
|
return result.bans;
|
|
|
|
}, [query, result]);
|
|
|
|
|
2021-07-05 11:25:20 +01:00
|
|
|
return (
|
2021-07-24 17:01:50 +01:00
|
|
|
<div className={styles.userList}>
|
2022-05-08 10:40:45 +01:00
|
|
|
<InputBox
|
|
|
|
placeholder="Search for a specific user..."
|
|
|
|
value={query}
|
|
|
|
onChange={(e) => setQuery(e.currentTarget.value)}
|
|
|
|
palette="secondary"
|
|
|
|
/>
|
2021-07-24 17:01:50 +01:00
|
|
|
<div className={styles.subtitle}>
|
|
|
|
<span>
|
|
|
|
<Text id="app.settings.server_pages.bans.user" />
|
|
|
|
</span>
|
|
|
|
<span class={styles.reason}>
|
|
|
|
<Text id="app.settings.server_pages.bans.reason" />
|
|
|
|
</span>
|
|
|
|
<span>
|
|
|
|
<Text id="app.settings.server_pages.bans.revoke" />
|
|
|
|
</span>
|
|
|
|
</div>
|
2022-05-08 10:40:45 +01:00
|
|
|
{typeof bans === "undefined" && <Preloader type="ring" />}
|
|
|
|
{bans && (
|
2021-09-25 14:43:28 +01:00
|
|
|
<div className={styles.virtual}>
|
|
|
|
<Virtuoso
|
2022-05-08 10:40:45 +01:00
|
|
|
totalCount={bans.length}
|
2021-09-25 14:43:28 +01:00
|
|
|
itemContent={(index) => (
|
|
|
|
<Inner
|
2022-05-08 10:40:45 +01:00
|
|
|
key={bans[index]._id.user}
|
2021-09-25 14:43:28 +01:00
|
|
|
server={server}
|
2022-05-08 10:40:45 +01:00
|
|
|
users={result!.users}
|
|
|
|
ban={bans[index]}
|
2021-09-25 14:43:28 +01:00
|
|
|
removeSelf={() => {
|
|
|
|
setData({
|
2022-05-08 10:40:45 +01:00
|
|
|
bans: result!.bans.filter(
|
2021-09-25 14:43:28 +01:00
|
|
|
(y) =>
|
|
|
|
y._id.user !==
|
2022-05-08 10:40:45 +01:00
|
|
|
bans[index]._id.user,
|
2021-09-25 14:43:28 +01:00
|
|
|
),
|
2022-05-08 10:40:45 +01:00
|
|
|
users: result!.users,
|
2021-09-25 14:43:28 +01:00
|
|
|
});
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
)}
|
2021-07-05 11:25:20 +01:00
|
|
|
</div>
|
|
|
|
);
|
2021-07-29 19:01:40 +01:00
|
|
|
});
|