2024-10-01 16:06:02 +02:00
|
|
|
import { Button, Form, InputGroup, Table } from "react-bootstrap";
|
2024-09-30 21:40:28 +02:00
|
|
|
import { useTranslation } from "react-i18next";
|
2024-10-02 16:49:33 +02:00
|
|
|
import {
|
|
|
|
Form as RemixForm,
|
|
|
|
Link,
|
|
|
|
Outlet,
|
|
|
|
useActionData,
|
|
|
|
useFetcher,
|
|
|
|
useRouteLoaderData,
|
|
|
|
} from "@remix-run/react";
|
2024-09-30 21:40:28 +02:00
|
|
|
import { loader as settingsLoader } from "../settings/route";
|
2024-10-01 16:06:02 +02:00
|
|
|
import { loader as rootLoader } from "../../root";
|
2024-09-30 22:05:14 +02:00
|
|
|
import { DateTime } from "luxon";
|
2024-10-01 16:06:02 +02:00
|
|
|
import { defaultAvatarUrl, idTimestamp } from "~/lib/utils";
|
|
|
|
import { ExclamationTriangleFill, InfoCircleFill } from "react-bootstrap-icons";
|
|
|
|
import AvatarImage from "~/components/profile/AvatarImage";
|
|
|
|
import { ActionFunctionArgs, json } from "@remix-run/node";
|
|
|
|
import { type ApiError, ErrorCode, firstErrorFor } from "~/lib/api/error";
|
|
|
|
import serverRequest, { getToken } from "~/lib/request.server";
|
|
|
|
import { MeUser } from "~/lib/api/user";
|
|
|
|
import ErrorAlert from "~/components/ErrorAlert";
|
2024-09-30 22:05:14 +02:00
|
|
|
|
2024-10-01 16:06:02 +02:00
|
|
|
export const action = async ({ request }: ActionFunctionArgs) => {
|
|
|
|
const data = await request.formData();
|
|
|
|
const username = data.get("username") as string | null;
|
2024-09-30 22:05:14 +02:00
|
|
|
const token = getToken(request);
|
2024-10-01 16:06:02 +02:00
|
|
|
|
|
|
|
if (!username) {
|
|
|
|
return json({
|
|
|
|
error: {
|
|
|
|
status: 403,
|
|
|
|
code: ErrorCode.BadRequest,
|
|
|
|
message: "Invalid username",
|
|
|
|
} as ApiError,
|
|
|
|
user: null,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
const resp = await serverRequest<MeUser>("PATCH", "/users/@me", { body: { username }, token });
|
|
|
|
|
|
|
|
return json({ user: resp, error: null });
|
|
|
|
} catch (e) {
|
|
|
|
return json({ error: e as ApiError, user: null });
|
|
|
|
}
|
2024-09-30 22:05:14 +02:00
|
|
|
};
|
2024-09-30 21:40:28 +02:00
|
|
|
|
|
|
|
export default function SettingsIndex() {
|
|
|
|
const { user } = useRouteLoaderData<typeof settingsLoader>("routes/settings")!;
|
2024-10-01 16:06:02 +02:00
|
|
|
const actionData = useActionData<typeof action>();
|
|
|
|
const { meta } = useRouteLoaderData<typeof rootLoader>("root")!;
|
2024-09-30 21:40:28 +02:00
|
|
|
const { t } = useTranslation();
|
2024-09-30 22:05:14 +02:00
|
|
|
|
|
|
|
const createdAt = idTimestamp(user.id);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
2024-10-02 16:49:33 +02:00
|
|
|
<Outlet />
|
2024-10-01 16:06:02 +02:00
|
|
|
<div className="row">
|
|
|
|
<div className="col-md">
|
|
|
|
<RemixForm method="POST">
|
|
|
|
<Form as="div">
|
|
|
|
<Form.Group className="mb-3" controlId="username">
|
|
|
|
<Form.Label>{t("settings.general.username")}</Form.Label>
|
|
|
|
<InputGroup className="m-1 w-75">
|
2024-10-01 21:25:51 +02:00
|
|
|
<Form.Control defaultValue={user.username} name="username" type="text" required />
|
2024-10-01 16:06:02 +02:00
|
|
|
<Button variant="secondary" type="submit">
|
|
|
|
{t("settings.general.change-username")}
|
|
|
|
</Button>
|
|
|
|
</InputGroup>
|
|
|
|
</Form.Group>
|
|
|
|
</Form>
|
|
|
|
</RemixForm>
|
|
|
|
|
|
|
|
<p className="text-muted text-has-newline">
|
|
|
|
<InfoCircleFill /> {t("settings.general.username-change-hint")}
|
|
|
|
</p>
|
|
|
|
{actionData?.error && <UsernameUpdateError error={actionData.error} />}
|
|
|
|
</div>
|
|
|
|
<div className="col-md text-center">
|
|
|
|
<AvatarImage
|
|
|
|
src={user.avatar_url || defaultAvatarUrl}
|
|
|
|
width={200}
|
|
|
|
alt={t("user.avatar-alt", { username: user.username })}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div>
|
|
|
|
<h4>{t("settings.general.log-out-everywhere")}</h4>
|
2024-10-01 21:25:51 +02:00
|
|
|
<p>{t("settings.general.log-out-everywhere-hint")}</p>
|
2024-10-02 16:49:33 +02:00
|
|
|
{/* @ts-expect-error as=Link */}
|
|
|
|
<Button as={Link} variant="danger" to="/settings/force-log-out">
|
|
|
|
{t("settings.general.force-log-out-button")}
|
|
|
|
</Button>
|
2024-10-01 16:06:02 +02:00
|
|
|
</div>
|
2024-10-01 21:25:51 +02:00
|
|
|
<h4 className="mt-2">{t("settings.general.table-header")}</h4>
|
2024-10-01 16:06:02 +02:00
|
|
|
<Table striped bordered hover>
|
2024-09-30 22:05:14 +02:00
|
|
|
<tbody>
|
|
|
|
<tr>
|
|
|
|
<th scope="row">{t("settings.general.id")}</th>
|
|
|
|
<td>
|
|
|
|
<code>{user.id}</code>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th scope="row">{t("settings.general.created")}</th>
|
|
|
|
<td>{createdAt.toLocaleString(DateTime.DATETIME_MED)}</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th scope="row">{t("settings.general.member-count")}</th>
|
|
|
|
<td>
|
2024-10-01 16:06:02 +02:00
|
|
|
{user.members.length}/{meta.limits.member_count}
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th scope="row">{t("settings.general.member-list-hidden")}</th>
|
|
|
|
<td>{user.member_list_hidden ? t("yes") : t("no")}</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th scope="row">{t("settings.general.custom-preferences")}</th>
|
|
|
|
<td>
|
|
|
|
{Object.keys(user.custom_preferences).length}/{meta.limits.custom_preferences}
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th scope="row">{t("settings.general.role")}</th>
|
|
|
|
<td>
|
|
|
|
<code>{user.role}</code>
|
2024-09-30 22:05:14 +02:00
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
</tbody>
|
|
|
|
</Table>
|
|
|
|
</>
|
|
|
|
);
|
2024-09-30 21:40:28 +02:00
|
|
|
}
|
2024-10-01 16:06:02 +02:00
|
|
|
|
|
|
|
function UsernameUpdateError({ error }: { error: ApiError }) {
|
|
|
|
const { t } = useTranslation();
|
|
|
|
const usernameError = firstErrorFor(error, "username");
|
|
|
|
if (!usernameError) {
|
|
|
|
return <ErrorAlert error={error} />;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<p className="text-danger-emphasis text-has-newline">
|
|
|
|
<ExclamationTriangleFill />{" "}
|
|
|
|
{t("settings.general.username-update-error", { message: usernameError.message })}
|
|
|
|
</p>
|
|
|
|
);
|
|
|
|
}
|