feat(dm): messages per month

This commit is contained in:
Samuel 2024-12-15 19:42:37 +01:00
parent 2fedbdc884
commit d9268a4991
No known key found for this signature in database
2 changed files with 123 additions and 13 deletions

View file

@ -177,7 +177,7 @@ const threadSentMessagesPerPersonOverviewQueryRaw = (threadId: number) =>
.selectFrom("message") .selectFrom("message")
.select((eb) => [ .select((eb) => [
"from_recipient_id", "from_recipient_id",
sql<Date>`DATE(datetime(message.date_sent / 1000, 'unixepoch'))`.as( sql<string>`DATE(datetime(message.date_sent / 1000, 'unixepoch'))`.as(
"message_date" "message_date"
), ),
eb.fn.countAll().as("message_count"), eb.fn.countAll().as("message_count"),
@ -200,6 +200,27 @@ export const dmSentMessagesPerPersonOverviewQuery = cached(
threadSentMessagesPerPersonOverviewQueryRaw threadSentMessagesPerPersonOverviewQueryRaw
); );
const threadSentMessagesOverviewQueryRaw = (threadId: number) =>
kyselyDb()
.selectFrom("message")
.select([
"from_recipient_id",
sql<Date>`datetime(date_sent / 1000, 'unixepoch')`.as("message_datetime"),
])
.orderBy(["message_datetime"])
.where((eb) =>
eb.and([
eb("body", "is not", null),
eb("body", "!=", ""),
eb("thread_id", "=", threadId),
])
)
.execute();
export const threadSentMessagesOverviewQuery = cached(
threadSentMessagesOverviewQueryRaw
);
const threadMostUsedWordsQueryRaw = (threadId: number, limit = 10) => const threadMostUsedWordsQueryRaw = (threadId: number, limit = 10) =>
kyselyDb() kyselyDb()
.withRecursive("words", (eb) => { .withRecursive("words", (eb) => {

View file

@ -1,9 +1,9 @@
import { type Accessor, type Component, createResource, Show } from "solid-js"; import { type Accessor, type Component, createEffect, createResource, Show } from "solid-js";
import type { RouteSectionProps } from "@solidjs/router"; import type { RouteSectionProps } from "@solidjs/router";
import { type ChartData } from "chart.js"; import { type ChartData } from "chart.js";
import { LineChart, WordCloudChart } from "~/components/ui/charts"; import { LineChart, RadarChart, WordCloudChart } from "~/components/ui/charts";
import { import {
dmOverviewQuery, dmOverviewQuery,
@ -11,6 +11,7 @@ import {
dmSentMessagesPerPersonOverviewQuery, dmSentMessagesPerPersonOverviewQuery,
SELF_ID, SELF_ID,
threadMostUsedWordsQuery, threadMostUsedWordsQuery,
threadSentMessagesOverviewQuery,
} from "~/db"; } from "~/db";
import { getNameFromRecipient } from "~/lib/get-name-from-recipient"; import { getNameFromRecipient } from "~/lib/get-name-from-recipient";
import { Heading } from "~/components/ui/heading"; import { Heading } from "~/components/ui/heading";
@ -19,6 +20,29 @@ import { Flex } from "~/components/ui/flex";
import { CalendarArrowUp, CalendarArrowDown, CalendarClock, MessagesSquare } from "lucide-solid"; import { CalendarArrowUp, CalendarArrowDown, CalendarClock, MessagesSquare } from "lucide-solid";
import { getDistanceBetweenDatesInDays } from "~/lib/date"; import { getDistanceBetweenDatesInDays } from "~/lib/date";
type MonthIndex = 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12;
const monthNames: Record<MonthIndex, string> = {
1: "January",
2: "February",
3: "March",
4: "April",
5: "May",
6: "June",
7: "July",
8: "August",
9: "September",
10: "October",
11: "November",
12: "December",
};
const initialMonthMap = Object.fromEntries(
Array(12)
.fill(0)
.map((_value, index) => [index + 1, 0]),
) as Record<MonthIndex, number>;
export const DmId: Component<RouteSectionProps> = (props) => { export const DmId: Component<RouteSectionProps> = (props) => {
const dmId = () => Number(props.params.dmid); const dmId = () => Number(props.params.dmid);
@ -52,16 +76,46 @@ export const DmId: Component<RouteSectionProps> = (props) => {
const [dmMessagesPerPerson] = createResource(() => dmSentMessagesPerPersonOverviewQuery(dmId())); const [dmMessagesPerPerson] = createResource(() => dmSentMessagesPerPersonOverviewQuery(dmId()));
const [dmMessagesOverview] = createResource(async () => {
const dmMessageOverview = await threadSentMessagesOverviewQuery(dmId());
if (dmMessageOverview) {
return dmMessageOverview.map((row) => {
return {
messageDate: new Date(row.message_datetime),
recipientId: row.from_recipient_id,
};
});
}
});
const dmMessagesPerMonth = () => {
const currentDmMessagesOverview = dmMessagesOverview();
if (currentDmMessagesOverview) {
return currentDmMessagesOverview.reduce<Record<MonthIndex, number>>(
(prev, curr) => {
const month = curr.messageDate.getMonth() as MonthIndex;
prev[month as MonthIndex] += 1;
return prev;
},
{ ...initialMonthMap },
);
}
};
// maps all the message counts to dates // maps all the message counts to dates
const dmMessages = () => { const dmMessagesPerDateOverview = () => {
return dmMessagesPerPerson()?.reduce< return dmMessagesPerPerson()?.reduce<
{ {
rawDate: string;
date: Date; date: Date;
totalMessages: number; totalMessages: number;
[recipientId: number]: number; [recipientId: number]: number;
}[] }[]
>((prev, curr) => { >((prev, curr) => {
const existingDate = prev.find(({ date }) => date === curr.message_date); const existingDate = prev.find(({ rawDate }) => rawDate === curr.message_date);
if (existingDate) { if (existingDate) {
existingDate[curr.from_recipient_id] = curr.message_count; existingDate[curr.from_recipient_id] = curr.message_count;
@ -69,7 +123,8 @@ export const DmId: Component<RouteSectionProps> = (props) => {
existingDate.totalMessages += curr.message_count; existingDate.totalMessages += curr.message_count;
} else { } else {
prev.push({ prev.push({
date: curr.message_date, rawDate: curr.message_date,
date: new Date(curr.message_date),
totalMessages: curr.message_count, totalMessages: curr.message_count,
[curr.from_recipient_id]: curr.message_count, [curr.from_recipient_id]: curr.message_count,
}); });
@ -128,12 +183,12 @@ export const DmId: Component<RouteSectionProps> = (props) => {
}; };
const dateChartData: Accessor<ChartData<"line"> | undefined> = () => { const dateChartData: Accessor<ChartData<"line"> | undefined> = () => {
const currentDmMessages = dmMessages(); const currentDmMessages = dmMessagesPerDateOverview();
const currentRecipients = recipients(); const currentRecipients = recipients();
if (currentDmMessages) { if (currentDmMessages) {
return { return {
labels: currentDmMessages.map((row) => row.date), labels: currentDmMessages.map((row) => row.date.toDateString()),
datasets: [ datasets: [
{ {
label: "Total number of messages", label: "Total number of messages",
@ -162,6 +217,22 @@ export const DmId: Component<RouteSectionProps> = (props) => {
} }
}; };
const monthChartData: Accessor<ChartData<"radar"> | undefined> = () => {
const currentMessagesPerMonth = dmMessagesPerMonth();
if (currentMessagesPerMonth) {
return {
labels: Object.values(monthNames),
datasets: [
{
label: "Number of messages",
data: Object.values(currentMessagesPerMonth),
},
],
};
}
};
return ( return (
<div class="flex flex-col items-center"> <div class="flex flex-col items-center">
<Heading level={1}>DM with {dmPartner()?.name}</Heading> <Heading level={1}>DM with {dmPartner()?.name}</Heading>
@ -198,33 +269,33 @@ export const DmId: Component<RouteSectionProps> = (props) => {
<Grid cols={1} colsMd={2} class="my-12 min-w-[35rem] gap-y-8 text-sm"> <Grid cols={1} colsMd={2} class="my-12 min-w-[35rem] gap-y-8 text-sm">
<Flex flexDirection="row" justifyContent="evenly" class="bg-amber-200 p-2 text-amber-900"> <Flex flexDirection="row" justifyContent="evenly" class="bg-amber-200 p-2 text-amber-900">
<Flex alignItems="center" justifyContent="center" class="min-w-16"> <Flex alignItems="center" justifyContent="center" class="min-w-16">
<CalendarArrowDown /> <CalendarArrowDown class="h-8 w-8" />
</Flex> </Flex>
<Flex flexDirection="col" justifyContent="around" class="flex-1"> <Flex flexDirection="col" justifyContent="around" class="flex-1">
<span>Your first message is from</span> <span>Your first message is from</span>
<Show when={!dmOverview.loading && dmOverview()}> <Show when={!dmOverview.loading && dmOverview()}>
{(currentDmOverview) => ( {(currentDmOverview) => (
<span class="font-semibold text-2xl">{currentDmOverview().firstMessageDate.toLocaleDateString()}</span> <span class="font-semibold text-2xl">{currentDmOverview().firstMessageDate.toDateString()}</span>
)} )}
</Show> </Show>
</Flex> </Flex>
</Flex> </Flex>
<Flex flexDirection="row" justifyContent="evenly" class="bg-emerald-200 p-2 text-emerald-900"> <Flex flexDirection="row" justifyContent="evenly" class="bg-emerald-200 p-2 text-emerald-900">
<Flex alignItems="center" justifyContent="center" class="min-w-16"> <Flex alignItems="center" justifyContent="center" class="min-w-16">
<CalendarArrowUp /> <CalendarArrowUp class="h-8 w-8" />
</Flex> </Flex>
<Flex flexDirection="col" justifyContent="around" class="flex-1"> <Flex flexDirection="col" justifyContent="around" class="flex-1">
<span>Your last message is from</span> <span>Your last message is from</span>
<Show when={!dmOverview.loading && dmOverview()}> <Show when={!dmOverview.loading && dmOverview()}>
{(currentDmOverview) => ( {(currentDmOverview) => (
<span class="font-semibold text-2xl">{currentDmOverview().lastMessageDate.toLocaleDateString()}</span> <span class="font-semibold text-2xl">{currentDmOverview().lastMessageDate.toDateString()}</span>
)} )}
</Show> </Show>
</Flex> </Flex>
</Flex> </Flex>
<Flex flexDirection="row" justifyContent="evenly" class="bg-blue-200 p-2 text-blue-900"> <Flex flexDirection="row" justifyContent="evenly" class="bg-blue-200 p-2 text-blue-900">
<Flex alignItems="center" justifyContent="center" class="min-w-16"> <Flex alignItems="center" justifyContent="center" class="min-w-16">
<CalendarClock /> <CalendarClock class="h-8 w-8" />
</Flex> </Flex>
<Flex flexDirection="col" justifyContent="around" class="flex-1"> <Flex flexDirection="col" justifyContent="around" class="flex-1">
<span>You have been chatting for</span> <span>You have been chatting for</span>
@ -256,6 +327,24 @@ export const DmId: Component<RouteSectionProps> = (props) => {
</Flex> </Flex>
</Flex> </Flex>
</Grid> </Grid>
<Heading level={2}>Messages per</Heading>
<div>
<Heading level={3}>Month</Heading>
<Grid cols={1} colsMd={2}>
<Show when={monthChartData()}>
{(currentMonthChartData) => (
<RadarChart
title="Month"
options={{
normalized: true,
}}
data={currentMonthChartData()}
class="max-h-96"
/>
)}
</Show>
</Grid>
</div>
<Heading level={2}>Word cloud</Heading> <Heading level={2}>Word cloud</Heading>
<Show when={mostUsedWordChartData()}> <Show when={mostUsedWordChartData()}>
{(currentMostUsedWordChartData) => ( {(currentMostUsedWordChartData) => (