# Introduction Avoid having multiple `isDefined` definition across our pacakges Also avoid importing `isDefined` from `twenty-ui` which exposes a huge barrel for a such little util function ## In a nutshell Removed own `isDefined.ts` definition from `twenty-ui` `twenty-front` and `twenty-server` to move it to `twenty-shared`. Updated imports for each packages, and added explicit dependencies to `twenty-shared` if not already in place Related PR https://github.com/twentyhq/twenty/pull/9941
147 lines
4.2 KiB
TypeScript
147 lines
4.2 KiB
TypeScript
import styled from '@emotion/styled';
|
|
import { zodResolver } from '@hookform/resolvers/zod';
|
|
import { useEffect } from 'react';
|
|
import { Controller, useForm } from 'react-hook-form';
|
|
import { Button, IconSend } from 'twenty-ui';
|
|
import { z } from 'zod';
|
|
|
|
import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/SnackBar';
|
|
import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar';
|
|
import { TextInput } from '@/ui/input/components/TextInput';
|
|
import { sanitizeEmailList } from '@/workspace/utils/sanitizeEmailList';
|
|
import { isDefined } from 'twenty-shared';
|
|
import { useCreateWorkspaceInvitation } from '../../workspace-invitation/hooks/useCreateWorkspaceInvitation';
|
|
|
|
const StyledContainer = styled.div`
|
|
display: flex;
|
|
flex-direction: row;
|
|
padding-bottom: ${({ theme }) => theme.spacing(3)};
|
|
`;
|
|
|
|
const StyledLinkContainer = styled.div`
|
|
flex: 1;
|
|
margin-right: ${({ theme }) => theme.spacing(2)};
|
|
`;
|
|
|
|
const emailValidationSchema = (email: string) =>
|
|
z.string().email(`Invalid email '${email}'`);
|
|
|
|
const validationSchema = () =>
|
|
z
|
|
.object({
|
|
emails: z.string().superRefine((value, ctx) => {
|
|
if (!value.length) {
|
|
return;
|
|
}
|
|
const emails = sanitizeEmailList(value.split(','));
|
|
if (emails.length === 0) {
|
|
ctx.addIssue({
|
|
code: z.ZodIssueCode.invalid_string,
|
|
message: 'Emails should not be empty',
|
|
validation: 'email',
|
|
});
|
|
}
|
|
const invalidEmails: string[] = [];
|
|
for (const email of emails) {
|
|
const result = emailValidationSchema(email).safeParse(email);
|
|
if (!result.success) {
|
|
invalidEmails.push(email);
|
|
}
|
|
}
|
|
if (invalidEmails.length > 0) {
|
|
ctx.addIssue({
|
|
code: z.ZodIssueCode.invalid_string,
|
|
message:
|
|
invalidEmails.length > 1
|
|
? 'Emails "' + invalidEmails.join('", "') + '" are invalid'
|
|
: 'Email "' + invalidEmails.join('", "') + '" is invalid',
|
|
validation: 'email',
|
|
});
|
|
}
|
|
}),
|
|
})
|
|
.required();
|
|
|
|
type FormInput = {
|
|
emails: string;
|
|
};
|
|
|
|
export const WorkspaceInviteTeam = () => {
|
|
const { enqueueSnackBar } = useSnackBar();
|
|
const { sendInvitation } = useCreateWorkspaceInvitation();
|
|
|
|
const { reset, handleSubmit, control, formState, watch } = useForm<FormInput>(
|
|
{
|
|
mode: 'onSubmit',
|
|
resolver: zodResolver(validationSchema()),
|
|
defaultValues: {
|
|
emails: '',
|
|
},
|
|
},
|
|
);
|
|
const isEmailsEmpty = !watch('emails');
|
|
|
|
const submit = handleSubmit(async ({ emails }) => {
|
|
const emailsList = sanitizeEmailList(emails.split(','));
|
|
const { data } = await sendInvitation({ emails: emailsList });
|
|
if (isDefined(data) && data.sendInvitations.result.length > 0) {
|
|
enqueueSnackBar(
|
|
`${data.sendInvitations.result.length} invitations sent`,
|
|
{
|
|
variant: SnackBarVariant.Success,
|
|
duration: 2000,
|
|
},
|
|
);
|
|
return;
|
|
}
|
|
if (isDefined(data) && !data.sendInvitations.success) {
|
|
data.sendInvitations.errors.forEach((error) => {
|
|
enqueueSnackBar(error, {
|
|
variant: SnackBarVariant.Error,
|
|
duration: 5000,
|
|
});
|
|
});
|
|
}
|
|
});
|
|
|
|
const { isSubmitSuccessful, errors } = formState;
|
|
|
|
useEffect(() => {
|
|
if (isSubmitSuccessful) {
|
|
reset();
|
|
}
|
|
}, [isSubmitSuccessful, reset]);
|
|
|
|
return (
|
|
<form onSubmit={submit}>
|
|
<StyledContainer>
|
|
<StyledLinkContainer>
|
|
<Controller
|
|
name="emails"
|
|
control={control}
|
|
render={({ field: { value, onChange }, fieldState: { error } }) => {
|
|
return (
|
|
<TextInput
|
|
placeholder="tim@apple.com, jony.ive@apple.dev"
|
|
value={value}
|
|
onChange={onChange}
|
|
error={error?.message}
|
|
fullWidth
|
|
/>
|
|
);
|
|
}}
|
|
/>
|
|
</StyledLinkContainer>
|
|
<Button
|
|
Icon={IconSend}
|
|
variant="primary"
|
|
accent="blue"
|
|
title="Invite"
|
|
type="submit"
|
|
disabled={isEmailsEmpty || !!errors.emails}
|
|
/>
|
|
</StyledContainer>
|
|
</form>
|
|
);
|
|
};
|