Press n or j to go to the next uncovered block, b, p or k for the previous block.
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 | 1x 1x 1x 1x 1x 1x 1x 1x 1x 1x 1x 14x 5x 16x 16x 16x 16x 16x 2x 16x 16x 2x 1x 1x 7x | import { styled, Typography, useMediaQuery, useTheme } from "@mui/material";
import { useMutation, useQueryClient } from "@tanstack/react-query";
import Alert from "components/Alert";
import Button from "components/Button";
import TextField from "components/TextField";
import { accountInfoQueryKey } from "features/queryKeys";
import { Empty } from "google-protobuf/google/protobuf/empty_pb";
import { RpcError } from "grpc-web";
import { useTranslation } from "i18n";
import { AUTH, GLOBAL } from "i18n/namespaces";
import { useForm } from "react-hook-form";
import { service } from "service";
import { theme } from "theme";
interface ChangePasswordVariables {
oldPassword: string;
newPassword: string;
}
interface ChangePasswordFormData extends ChangePasswordVariables {
passwordConfirmation: string;
}
interface ChangePasswordProps {
className?: string;
}
const StyledForm = styled("form")(() => ({
marginBottom: theme.spacing(2),
"& > * + *": {
marginBlockStart: theme.spacing(1),
},
}));
export default function ChangePassword({ className }: ChangePasswordProps) {
const { t } = useTranslation([AUTH, GLOBAL]);
const theme = useTheme();
const isMdOrWider = useMediaQuery(theme.breakpoints.up("md"));
const {
getValues,
handleSubmit,
reset: resetForm,
register,
formState: { errors },
} = useForm<ChangePasswordFormData>({
mode: "onBlur",
});
const onSubmit = handleSubmit(({ oldPassword, newPassword }) => {
changePassword({ oldPassword, newPassword });
});
const queryClient = useQueryClient();
const {
error: changePasswordError,
isPending: isChangePasswordLoading,
isSuccess: isChangePasswordSuccess,
mutate: changePassword,
} = useMutation<Empty, RpcError, ChangePasswordVariables>({
mutationFn: ({ oldPassword, newPassword }) =>
service.account.changePassword(oldPassword, newPassword),
onSuccess: () => {
queryClient.invalidateQueries({
queryKey: [accountInfoQueryKey],
});
resetForm();
},
});
return (
<div className={className}>
<Typography variant="h2">
{t("auth:change_password_form.title")}
</Typography>
{changePasswordError && (
<Alert severity="error">{changePasswordError.message}</Alert>
)}
{isChangePasswordSuccess && (
<Alert severity="success">
{t("auth:change_password_form.password_changed_success")}
</Alert>
)}
<StyledForm onSubmit={onSubmit}>
<TextField
{...register("oldPassword", { required: true })}
id="oldPassword"
label={t("auth:change_password_form.old_password")}
type="password"
fullWidth={!isMdOrWider}
/>
<TextField
id="newPassword"
{...register("newPassword", { required: true })}
label={t("auth:change_password_form.new_password")}
name="newPassword"
type="password"
fullWidth={!isMdOrWider}
/>
<TextField
id="passwordConfirmation"
{...register("passwordConfirmation", {
validate: (value) =>
value === getValues("newPassword") ||
t("auth:change_password_form.password_mismatch_error"),
})}
label={t("auth:change_password_form.confirm_password")}
fullWidth={!isMdOrWider}
type="password"
helperText={errors.passwordConfirmation?.message}
/>
<Button
fullWidth={!isMdOrWider}
loading={isChangePasswordLoading}
type="submit"
>
{t("global:submit")}
</Button>
</StyledForm>
</div>
);
}
|