in superset-frontend/src/features/users/UserListModal.tsx [37:240]
function UserListModal({
show,
onHide,
onSave,
roles,
isEditMode = false,
user,
}: UserModalProps) {
const { addDangerToast, addSuccessToast } = useToasts();
const handleFormSubmit = async (values: FormValues) => {
const handleError = async (err: any, action: 'create' | 'update') => {
let errorMessage =
action === 'create'
? t('Error while adding user!')
: t('Error while updating user!');
if (err.status === 422) {
const errorData = await err.json();
const detail = errorData?.message || '';
if (detail.includes('duplicate key value')) {
if (detail.includes('ab_user_username_key')) {
errorMessage = t(
'This username is already taken. Please choose another one.',
);
} else if (detail.includes('ab_user_email_key')) {
errorMessage = t(
'This email is already associated with an account.',
);
}
}
}
addDangerToast(errorMessage);
throw err;
};
if (isEditMode) {
if (!user) {
throw new Error('User is required in edit mode');
}
try {
await updateUser(user.id, values);
addSuccessToast(t('User was successfully updated!'));
} catch (err) {
await handleError(err, 'update');
}
} else {
try {
await createUser(values);
addSuccessToast(t('User was successfully created!'));
} catch (err) {
await handleError(err, 'create');
}
}
};
const requiredFields = isEditMode
? ['first_name', 'last_name', 'username', 'email', 'roles']
: [
'first_name',
'last_name',
'username',
'email',
'password',
'roles',
'confirmPassword',
];
const initialValues = {
...user,
roles: user?.roles.map(role => role.id) || [],
};
return (
<FormModal
show={show}
onHide={onHide}
title={isEditMode ? t('Edit User') : t('Add User')}
onSave={onSave}
formSubmitHandler={handleFormSubmit}
requiredFields={requiredFields}
initialValues={initialValues}
>
{(form: FormInstance) => (
<>
<FormItem
name="first_name"
label={t('First name')}
rules={[{ required: true, message: t('First name is required') }]}
>
<Input
name="first_name"
placeholder={t("Enter the user's first name")}
/>
</FormItem>
<FormItem
name="last_name"
label={t('Last name')}
rules={[{ required: true, message: t('Last name is required') }]}
>
<Input
name="last_name"
placeholder={t("Enter the user's last name")}
/>
</FormItem>
<FormItem
name="username"
label={t('Username')}
rules={[{ required: true, message: t('Username is required') }]}
>
<Input
name="username"
placeholder={t("Enter the user's username")}
/>
</FormItem>
<FormItem
name="active"
label={t('Is active?')}
valuePropName="checked"
>
<Checkbox
onChange={checked => {
form.setFieldsValue({ isActive: checked });
}}
/>
</FormItem>
<FormItem
name="email"
label={t('Email')}
rules={[
{ required: true, message: t('Email is required') },
{
type: 'email',
message: t('Please enter a valid email address'),
},
]}
>
<Input name="email" placeholder={t("Enter the user's email")} />
</FormItem>
<FormItem
name="roles"
label={t('Roles')}
rules={[{ required: true, message: t('Role is required') }]}
>
<Select
name="roles"
mode="multiple"
placeholder={t('Select roles')}
options={roles.map(role => ({
value: role.id,
label: role.name,
}))}
getPopupContainer={trigger =>
trigger.closest('.antd5-modal-content')
}
/>
</FormItem>
{!isEditMode && (
<>
<FormItem
name="password"
label={t('Password')}
rules={[{ required: true, message: t('Password is required') }]}
>
<Input.Password
name="password"
placeholder="Enter the user's password"
/>
</FormItem>
<FormItem
name="confirmPassword"
label={t('Confirm Password')}
dependencies={['password']}
rules={[
{
required: true,
message: t('Please confirm your password'),
},
({ getFieldValue }) => ({
validator(_, value) {
if (!value || getFieldValue('password') === value) {
return Promise.resolve();
}
return Promise.reject(
new Error(t('Passwords do not match!')),
);
},
}),
]}
>
<Input.Password
name="confirmPassword"
placeholder={t("Confirm the user's password")}
/>
</FormItem>
</>
)}
</>
)}
</FormModal>
);
}