chore: Move to Typescript (#2783)

This PR moves the entire project to Typescript. Due to the ~1000 ignores this will lead to a messy codebase for a while, but the churn is worth it – all of those ignore comments are places that were never type-safe previously.

closes #1282
This commit is contained in:
Tom Moor
2021-11-29 06:40:55 -08:00
committed by GitHub
parent 25ccfb5d04
commit 15b1069bcc
1017 changed files with 17410 additions and 54942 deletions

182
app/menus/UserMenu.tsx Normal file
View File

@@ -0,0 +1,182 @@
import { observer } from "mobx-react";
import * as React from "react";
import { useTranslation } from "react-i18next";
import { useMenuState } from "reakit/Menu";
import User from "~/models/User";
import ContextMenu from "~/components/ContextMenu";
import OverflowMenuButton from "~/components/ContextMenu/OverflowMenuButton";
import Template from "~/components/ContextMenu/Template";
import useStores from "~/hooks/useStores";
type Props = {
user: User;
};
function UserMenu({ user }: Props) {
const { users, policies } = useStores();
const { t } = useTranslation();
const menu = useMenuState({
modal: true,
});
const can = policies.abilities(user.id);
const handlePromote = React.useCallback(
(ev: React.SyntheticEvent) => {
ev.preventDefault();
if (
!window.confirm(
t(
"Are you sure you want to make {{ userName }} an admin? Admins can modify team and billing information.",
{
userName: user.name,
}
)
)
) {
return;
}
users.promote(user);
},
[users, user, t]
);
const handleMember = React.useCallback(
(ev: React.SyntheticEvent) => {
ev.preventDefault();
if (
!window.confirm(
t("Are you sure you want to make {{ userName }} a member?", {
userName: user.name,
})
)
) {
return;
}
users.demote(user, "member");
},
[users, user, t]
);
const handleViewer = React.useCallback(
(ev: React.SyntheticEvent) => {
ev.preventDefault();
if (
!window.confirm(
t(
"Are you sure you want to make {{ userName }} a read-only viewer? They will not be able to edit any content",
{
userName: user.name,
}
)
)
) {
return;
}
users.demote(user, "viewer");
},
[users, user, t]
);
const handleSuspend = React.useCallback(
(ev: React.SyntheticEvent) => {
ev.preventDefault();
if (
!window.confirm(
t(
"Are you sure you want to suspend this account? Suspended users will be prevented from logging in."
)
)
) {
return;
}
users.suspend(user);
},
[users, user, t]
);
const handleRevoke = React.useCallback(
(ev: React.SyntheticEvent) => {
ev.preventDefault();
users.delete(user, {
confirmation: true,
});
},
[users, user]
);
const handleActivate = React.useCallback(
(ev: React.SyntheticEvent) => {
ev.preventDefault();
users.activate(user);
},
[users, user]
);
return (
<>
<OverflowMenuButton aria-label={t("Show menu")} {...menu} />
<ContextMenu {...menu} aria-label={t("User options")}>
<Template
{...menu}
items={[
{
type: "button",
title: t("Make {{ userName }} a member", {
userName: user.name,
}),
onClick: handleMember,
visible: can.demote && user.role !== "member",
},
{
type: "button",
title: t("Make {{ userName }} a viewer", {
userName: user.name,
}),
onClick: handleViewer,
visible: can.demote && user.role !== "viewer",
},
{
type: "button",
title: t("Make {{ userName }} an admin…", {
userName: user.name,
}),
onClick: handlePromote,
visible: can.promote && user.role !== "admin",
},
{
type: "separator",
},
{
type: "button",
title: `${t("Revoke invite")}`,
onClick: handleRevoke,
visible: user.isInvited,
},
{
type: "button",
title: t("Activate account"),
onClick: handleActivate,
visible: !user.isInvited && user.isSuspended,
},
{
type: "button",
title: `${t("Suspend account")}`,
onClick: handleSuspend,
visible: !user.isInvited && !user.isSuspended,
},
]}
/>
</ContextMenu>
</>
);
}
export default observer(UserMenu);