You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
overseerr/src/components/UserEdit/index.tsx

187 lines
6.1 KiB

import React, { useState, useEffect } from 'react';
import { useRouter } from 'next/router';
import LoadingSpinner from '../Common/LoadingSpinner';
import { useUser } from '../../hooks/useUser';
import Button from '../Common/Button';
import { useIntl, defineMessages, FormattedMessage } from 'react-intl';
import axios from 'axios';
import { useToasts } from 'react-toast-notifications';
import Header from '../Common/Header';
import PermissionEdit from '../PermissionEdit';
import { Field, Form, Formik } from 'formik';
import * as Yup from 'yup';
import { UserType } from '../../../server/constants/user';
export const messages = defineMessages({
edituser: 'Edit User',
plexUsername: 'Plex Username',
username: 'Display Name',
avatar: 'Avatar',
email: 'Email',
permissions: 'Permissions',
save: 'Save Changes',
saving: 'Saving…',
usersaved: 'User saved!',
userfail: 'Something went wrong while saving the user.',
});
const UserEdit: React.FC = () => {
const router = useRouter();
const intl = useIntl();
const { addToast } = useToasts();
const { user: currentUser } = useUser();
const { user, error, revalidate } = useUser({
id: Number(router.query.userId),
});
const [currentPermission, setCurrentPermission] = useState(0);
useEffect(() => {
if (currentPermission !== user?.permissions ?? 0) {
setCurrentPermission(user?.permissions ?? 0);
}
// We know what we are doing here.
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [user]);
if (!user && !error) {
return <LoadingSpinner />;
}
const UserEditSchema = Yup.object().shape({
username: Yup.string(),
});
return (
<Formik
initialValues={{
plexUsername: user?.plexUsername,
username: user?.username,
email: user?.email,
}}
validationSchema={UserEditSchema}
onSubmit={async (values) => {
try {
await axios.put(`/api/v1/user/${user?.id}`, {
permissions: currentPermission,
email: user?.email,
username: values.username,
});
addToast(intl.formatMessage(messages.usersaved), {
appearance: 'success',
autoDismiss: true,
});
} catch (e) {
addToast(intl.formatMessage(messages.userfail), {
appearance: 'error',
autoDismiss: true,
});
throw new Error(
`Something went wrong while saving the user: ${e.message}`
);
} finally {
revalidate();
}
}}
>
{({ isSubmitting, handleSubmit }) => (
<Form>
<div>
<div className="flex flex-col justify-between sm:flex-row">
<Header>
<FormattedMessage {...messages.edituser} />
</Header>
</div>
{user?.userType === UserType.PLEX && (
<div className="form-row">
<label htmlFor="plexUsername" className="text-label">
{intl.formatMessage(messages.plexUsername)}
</label>
<div className="form-input">
<div className="flex max-w-lg rounded-md shadow-sm">
<Field
id="plexUsername"
name="plexUsername"
type="text"
readOnly
/>
</div>
</div>
</div>
)}
<div className="form-row">
<label htmlFor="username" className="text-label">
{intl.formatMessage(messages.username)}
</label>
<div className="form-input">
<div className="flex max-w-lg rounded-md shadow-sm">
<Field id="username" name="username" type="text" />
</div>
</div>
</div>
<div className="form-row">
<label htmlFor="email" className="text-label">
<FormattedMessage {...messages.email} />
</label>
<div className="form-input">
<div className="flex max-w-lg rounded-md shadow-sm">
<Field id="email" name="email" type="text" readOnly />
</div>
</div>
</div>
<div className="form-row">
<span className="text-label">
<FormattedMessage {...messages.avatar} />
</span>
<div className="form-input">
<div className="flex max-w-lg rounded-md shadow-sm">
<img
className="w-40 h-40 rounded-full"
src={user?.avatar}
alt=""
/>
</div>
</div>
</div>
</div>
<div role="group" aria-labelledby="group-label" className="group">
<div className="form-row">
<span id="group-label" className="group-label">
<FormattedMessage {...messages.permissions} />
</span>
<div className="form-input">
<div className="max-w-lg">
<PermissionEdit
user={currentUser}
currentPermission={currentPermission}
onUpdate={(newPermission) =>
setCurrentPermission(newPermission)
}
/>
</div>
</div>
</div>
</div>
<div className="actions">
<div className="flex justify-end">
<span className="inline-flex ml-3 rounded-md shadow-sm">
<Button
buttonType="primary"
type="submit"
disabled={isSubmitting}
onClick={() => handleSubmit}
>
{isSubmitting
? intl.formatMessage(messages.saving)
: intl.formatMessage(messages.save)}
</Button>
</span>
</div>
</div>
</Form>
)}
</Formik>
);
};
export default UserEdit;