2021-07-31 15:57:43 +00:00
|
|
|
import type { BlitzPage, GetServerSideProps } from "blitz";
|
|
|
|
import { Routes, getSession, useRouter, useMutation } from "blitz";
|
|
|
|
import { useEffect } from "react";
|
|
|
|
import { useForm } from "react-hook-form";
|
|
|
|
import clsx from "clsx";
|
2021-08-08 04:34:29 +00:00
|
|
|
import twilio from "twilio";
|
2021-07-31 14:33:18 +00:00
|
|
|
|
2021-07-31 15:57:43 +00:00
|
|
|
import db from "../../../../db";
|
|
|
|
import OnboardingLayout from "../../components/onboarding-layout";
|
|
|
|
import setPhoneNumber from "../../mutations/set-phone-number";
|
2021-07-31 14:33:18 +00:00
|
|
|
|
|
|
|
type PhoneNumber = {
|
2021-07-31 15:57:43 +00:00
|
|
|
phoneNumber: string;
|
|
|
|
sid: string;
|
|
|
|
};
|
2021-07-31 14:33:18 +00:00
|
|
|
|
|
|
|
type Props = {
|
2021-07-31 15:57:43 +00:00
|
|
|
availablePhoneNumbers: PhoneNumber[];
|
|
|
|
};
|
2021-07-31 14:33:18 +00:00
|
|
|
|
|
|
|
type Form = {
|
2021-07-31 15:57:43 +00:00
|
|
|
phoneNumberSid: string;
|
|
|
|
};
|
2021-07-31 14:33:18 +00:00
|
|
|
|
|
|
|
const StepThree: BlitzPage<Props> = ({ availablePhoneNumbers }) => {
|
|
|
|
const {
|
|
|
|
register,
|
|
|
|
handleSubmit,
|
|
|
|
setValue,
|
|
|
|
formState: { isSubmitting },
|
2021-07-31 15:57:43 +00:00
|
|
|
} = useForm<Form>();
|
|
|
|
const router = useRouter();
|
|
|
|
const [setPhoneNumberMutation] = useMutation(setPhoneNumber);
|
2021-07-31 14:33:18 +00:00
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (availablePhoneNumbers[0]) {
|
2021-07-31 15:57:43 +00:00
|
|
|
setValue("phoneNumberSid", availablePhoneNumbers[0].sid);
|
2021-07-31 14:33:18 +00:00
|
|
|
}
|
2021-07-31 15:57:43 +00:00
|
|
|
});
|
2021-07-31 14:33:18 +00:00
|
|
|
|
|
|
|
const onSubmit = handleSubmit(async ({ phoneNumberSid }) => {
|
|
|
|
if (isSubmitting) {
|
2021-07-31 15:57:43 +00:00
|
|
|
return;
|
2021-07-31 14:33:18 +00:00
|
|
|
}
|
|
|
|
|
2021-07-31 15:57:43 +00:00
|
|
|
await setPhoneNumberMutation({ phoneNumberSid });
|
|
|
|
await router.push(Routes.Messages());
|
|
|
|
});
|
2021-07-31 14:33:18 +00:00
|
|
|
|
|
|
|
return (
|
2021-08-01 03:05:40 +00:00
|
|
|
<div className="flex flex-col space-y-4 items-center">
|
|
|
|
<form onSubmit={onSubmit}>
|
|
|
|
<label htmlFor="phoneNumberSid" className="block text-sm font-medium text-gray-700">
|
|
|
|
Phone number
|
|
|
|
</label>
|
|
|
|
<select
|
|
|
|
id="phoneNumberSid"
|
|
|
|
className="mt-1 block w-full pl-3 pr-10 py-2 text-base border-gray-300 focus:outline-none focus:ring-primary-500 focus:border-primary-500 sm:text-sm rounded-md"
|
|
|
|
{...register("phoneNumberSid")}
|
|
|
|
>
|
|
|
|
{availablePhoneNumbers.map(({ sid, phoneNumber }) => (
|
|
|
|
<option value={sid} key={sid}>
|
|
|
|
{phoneNumber}
|
|
|
|
</option>
|
|
|
|
))}
|
|
|
|
</select>
|
2021-07-31 14:33:18 +00:00
|
|
|
|
2021-08-01 03:05:40 +00:00
|
|
|
<button
|
|
|
|
type="submit"
|
|
|
|
className={clsx(
|
|
|
|
"max-w-[240px] mt-6 mx-auto w-full inline-flex justify-center rounded-md border border-transparent shadow-sm px-4 py-2 text-base font-medium text-white focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-primary-500 sm:text-sm",
|
|
|
|
!isSubmitting && "bg-primary-600 hover:bg-primary-700",
|
2021-08-01 12:04:04 +00:00
|
|
|
isSubmitting && "bg-primary-400 cursor-not-allowed",
|
2021-08-01 03:05:40 +00:00
|
|
|
)}
|
|
|
|
>
|
|
|
|
Save
|
|
|
|
</button>
|
|
|
|
</form>
|
|
|
|
</div>
|
2021-07-31 15:57:43 +00:00
|
|
|
);
|
|
|
|
};
|
2021-07-31 14:33:18 +00:00
|
|
|
|
2021-08-01 03:05:40 +00:00
|
|
|
StepThree.getLayout = (page) => (
|
|
|
|
<OnboardingLayout currentStep={3} previous={{ href: Routes.StepTwo().pathname, label: "Back" }}>
|
|
|
|
{page}
|
|
|
|
</OnboardingLayout>
|
|
|
|
);
|
|
|
|
|
|
|
|
StepThree.authenticate = { redirectTo: Routes.SignIn() };
|
2021-07-31 14:33:18 +00:00
|
|
|
|
|
|
|
export const getServerSideProps: GetServerSideProps<Props> = async ({ req, res }) => {
|
2021-07-31 15:57:43 +00:00
|
|
|
const session = await getSession(req, res);
|
2021-07-31 17:22:48 +00:00
|
|
|
if (!session.userId) {
|
|
|
|
await session.$revoke();
|
|
|
|
return {
|
|
|
|
redirect: {
|
2021-08-26 19:17:46 +00:00
|
|
|
destination: Routes.LandingPage().pathname,
|
2021-07-31 17:22:48 +00:00
|
|
|
permanent: false,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2021-08-05 17:07:15 +00:00
|
|
|
const phoneNumber = await db.phoneNumber.findFirst({ where: { organizationId: session.orgId } });
|
2021-07-31 17:22:48 +00:00
|
|
|
if (phoneNumber) {
|
|
|
|
return {
|
|
|
|
redirect: {
|
|
|
|
destination: Routes.Messages().pathname,
|
|
|
|
permanent: false,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2021-08-05 17:07:15 +00:00
|
|
|
const organization = await db.organization.findFirst({ where: { id: session.orgId } });
|
|
|
|
if (!organization) {
|
2021-07-31 14:33:18 +00:00
|
|
|
return {
|
|
|
|
redirect: {
|
|
|
|
destination: Routes.StepOne().pathname,
|
|
|
|
permanent: false,
|
|
|
|
},
|
2021-07-31 15:57:43 +00:00
|
|
|
};
|
2021-07-31 14:33:18 +00:00
|
|
|
}
|
|
|
|
|
2021-08-05 17:07:15 +00:00
|
|
|
if (!organization.twilioAccountSid || !organization.twilioAuthToken) {
|
2021-07-31 14:33:18 +00:00
|
|
|
return {
|
|
|
|
redirect: {
|
|
|
|
destination: Routes.StepTwo().pathname,
|
|
|
|
permanent: false,
|
|
|
|
},
|
2021-07-31 15:57:43 +00:00
|
|
|
};
|
2021-07-31 14:33:18 +00:00
|
|
|
}
|
|
|
|
|
2021-08-08 04:34:29 +00:00
|
|
|
const incomingPhoneNumbers = await twilio(
|
|
|
|
organization.twilioAccountSid,
|
|
|
|
organization.twilioAuthToken,
|
|
|
|
).incomingPhoneNumbers.list();
|
2021-07-31 15:57:43 +00:00
|
|
|
const phoneNumbers = incomingPhoneNumbers.map(({ phoneNumber, sid }) => ({ phoneNumber, sid }));
|
2021-07-31 14:33:18 +00:00
|
|
|
|
|
|
|
return {
|
|
|
|
props: {
|
|
|
|
availablePhoneNumbers: phoneNumbers,
|
|
|
|
},
|
2021-07-31 15:57:43 +00:00
|
|
|
};
|
|
|
|
};
|
2021-07-31 14:33:18 +00:00
|
|
|
|
2021-07-31 15:57:43 +00:00
|
|
|
export default StepThree;
|