shellphone.app/app/routes/__app/messages.$recipient.tsx

96 lines
3.1 KiB
TypeScript
Raw Normal View History

2022-05-14 10:22:06 +00:00
import { Suspense } from "react";
import type { LoaderFunction, MetaFunction } from "@remix-run/node";
2022-05-19 22:55:02 +00:00
import { Link, useNavigate, useParams } from "@remix-run/react";
import { json, useLoaderData } from "superjson-remix";
2022-05-19 22:55:02 +00:00
import { IoCall, IoChevronBack } from "react-icons/io5";
import { parsePhoneNumber } from "awesome-phonenumber";
2022-05-14 10:22:06 +00:00
import { type Message, Prisma } from "@prisma/client";
import Conversation from "~/features/messages/components/conversation";
import { getSeoMeta } from "~/utils/seo";
import db from "~/utils/db.server";
import { requireLoggedIn } from "~/utils/auth.server";
2022-05-19 22:55:02 +00:00
import newMessageAction from "~/features/messages/actions/messages.$recipient";
2022-05-14 10:22:06 +00:00
export const meta: MetaFunction = ({ params }) => {
const recipient = decodeURIComponent(params.recipient ?? "");
return {
...getSeoMeta({
title: `Messages with ${recipient}`,
}),
};
};
2022-05-19 22:55:02 +00:00
export const action = newMessageAction;
2022-05-14 10:22:06 +00:00
type ConversationType = {
recipient: string;
formattedPhoneNumber: string;
messages: Message[];
};
export type ConversationLoaderData = {
conversation: ConversationType;
};
export const loader: LoaderFunction = async ({ request, params }) => {
const { organizations } = await requireLoggedIn(request);
const recipient = decodeURIComponent(params.recipient ?? "");
const conversation = await getConversation(recipient);
return json<ConversationLoaderData>({ conversation });
async function getConversation(recipient: string): Promise<ConversationType> {
const organizationId = organizations[0].id;
const phoneNumber = await db.phoneNumber.findUnique({
where: { organizationId_isCurrent: { organizationId, isCurrent: true } },
2022-05-14 10:22:06 +00:00
});
if (!phoneNumber || phoneNumber.isFetchingMessages) {
throw new Error("unreachable");
2022-05-14 10:22:06 +00:00
}
const formattedPhoneNumber = parsePhoneNumber(recipient).getNumber("international");
const messages = await db.message.findMany({
where: {
phoneNumberId: phoneNumber.id,
2022-05-14 10:22:06 +00:00
OR: [{ from: recipient }, { to: recipient }],
},
orderBy: { sentAt: Prisma.SortOrder.asc },
2022-05-14 10:22:06 +00:00
});
return {
recipient,
formattedPhoneNumber,
messages,
};
}
};
export default function ConversationPage() {
const navigate = useNavigate();
const params = useParams<{ recipient: string }>();
const recipient = decodeURIComponent(params.recipient ?? "");
const { conversation } = useLoaderData<ConversationLoaderData>();
return (
2022-05-19 22:55:02 +00:00
<section className="h-full">
<header className="absolute top-0 w-screen h-12 backdrop-filter backdrop-blur-sm bg-white bg-opacity-75 border-b items-center flex justify-between">
<span className="pl-2 cursor-pointer" onClick={() => navigate(-1)}>
<IoChevronBack className="h-6 w-6" />
2022-05-14 10:22:06 +00:00
</span>
2022-05-19 22:55:02 +00:00
<strong className="absolute right-0 left-0 text-center pointer-events-none">
{conversation?.formattedPhoneNumber ?? recipient}
</strong>
<Link to={`/outgoing-call/${encodeURI(recipient)}`} className="pr-2">
<IoCall className="h-6 w-6" />
</Link>
2022-05-14 10:22:06 +00:00
</header>
{/*<Suspense fallback={<div className="pt-12">Loading messages with {recipient}</div>}>*/}
<Conversation />
{/*</Suspense>*/}
2022-05-19 22:55:02 +00:00
</section>
2022-05-14 10:22:06 +00:00
);
}
2022-05-19 23:05:06 +00:00
export const handle = { hideFooter: true };