revite/src/pages/settings/server/Bans.tsx

140 lines
4.5 KiB
TypeScript
Raw Normal View History

import { XCircle } from "@styled-icons/boxicons-regular";
2021-07-29 19:01:40 +01:00
import { observer } from "mobx-react-lite";
import { Virtuoso } from "react-virtuoso";
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
import styles from "./Panes.module.scss";
import { Text } from "preact-i18n";
import { useEffect, useMemo, useState } from "preact/hooks";
2021-06-19 22:37:12 +01:00
import UserIcon from "../../../components/common/user/UserIcon";
import IconButton from "../../../components/ui/IconButton";
import Preloader from "../../../components/ui/Preloader";
import { InputBox } from "@revoltchat/ui";
2021-07-05 11:23:23 +01:00
interface InnerProps {
ban: API.ServerBan;
users: Record<string, API.BannedUser>;
server: Server;
removeSelf: () => void;
}
const Inner = observer(({ ban, users, server, removeSelf }: InnerProps) => {
const [deleting, setDelete] = useState(false);
const user = users[ban._id.user];
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) => {
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(() => {
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
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 (
<div className={styles.userList}>
<InputBox
placeholder="Search for a specific user..."
value={query}
onChange={(e) => setQuery(e.currentTarget.value)}
palette="secondary"
/>
<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>
{typeof bans === "undefined" && <Preloader type="ring" />}
{bans && (
<div className={styles.virtual}>
<Virtuoso
totalCount={bans.length}
itemContent={(index) => (
<Inner
key={bans[index]._id.user}
server={server}
users={result!.users}
ban={bans[index]}
removeSelf={() => {
setData({
bans: result!.bans.filter(
(y) =>
y._id.user !==
bans[index]._id.user,
),
users: result!.users,
});
}}
/>
)}
/>
</div>
)}
2021-07-05 11:25:20 +01:00
</div>
);
2021-07-29 19:01:40 +01:00
});