Track which account is interested in which events

Store a list of ids of events and slots that accounts have marked as
being interested in, and show aggeregate counts in the schedule.
This commit is contained in:
Hornwitser 2025-03-07 20:15:41 +01:00
parent ca51c07065
commit db9a12250e
5 changed files with 137 additions and 0 deletions

View file

@ -2,10 +2,35 @@
<section class="event">
<h3>{{ event.name }}</h3>
<p>{{ event.description ?? "No description provided" }}</p>
<p v-if="event.interested">
{{ event.interested }} interested
</p>
<p v-if="session">
<button
class="interested"
:class="{ active: interestedIds.has(event.id) }"
@click="toggle(event.id, event.slots.map(slot => slot.id))"
>
{{ interestedIds.has(event.id) ? "✔ interested" : "🔔 interested?" }}
</button>
</p>
<h4>Timeslots</h4>
<ul>
<li v-for="slot in event.slots" :key="slot.id">
{{ slot.start }} - {{ slot.end }}
<button
v-if="session && event.slots.length > 1"
class="interested"
:disabled="interestedIds.has(event.id)"
:class="{ active: interestedIds.has(event.id) || interestedIds.has(slot.id) }"
@click="toggle(slot.id)"
>
{{ interestedIds.has(event.id) || interestedIds.has(slot.id) ? "✔ interested" : "🔔 interested?" }}
</button>
<template v-if="slot.interested">
({{ slot.interested }} interested)
</template>
</li>
</ul>
</section>
@ -17,6 +42,27 @@ import type { ScheduleEvent } from '~/shared/types/schedule';
defineProps<{
event: ScheduleEvent
}>()
const { data: session, refresh: refreshSession } = useAccountSession();
const interestedIds = computed(() => new Set(session.value?.account.interestedIds ?? []));
async function toggle(id: string, slotIds?: string[]) {
let newIds = [...session.value!.account.interestedIds ?? []];
if (interestedIds.value.has(id)) {
newIds = newIds.filter(newId => newId !== id);
} else {
newIds.push(id);
if (slotIds) {
const filterIds = new Set(slotIds);
newIds = newIds.filter(newId => !filterIds.has(newId));
}
}
await $fetch("/api/account", {
method: "PATCH",
body: { interestedIds: newIds },
})
await refreshSession();
}
</script>
<style scoped>
@ -31,4 +77,11 @@ defineProps<{
.event + .event {
margin-block-start: 0.5rem;
}
button {
padding-inline: 0.2em;
}
button.active {
color: color-mix(in oklab, var(--foreground), green 50%);
}
</style>

View file

@ -0,0 +1,46 @@
import { Account } from "~/shared/types/account";
import { readAccounts, readSchedule, writeAccounts, writeSchedule } from "~/server/database";
import { broadcastUpdate } from "~/server/streams";
export default defineEventHandler(async (event) => {
const session = await requireAccountSession(event);
const body: Pick<Account, "interestedIds"> = await readBody(event);
if (
!(body.interestedIds instanceof Array)
|| !body.interestedIds.every(id => typeof id === "string")
) {
throw createError({
status: 400,
message: "Invalid interestedIds",
});
}
const accounts = await readAccounts();
const sessionAccount = accounts.find(account => account.id === session.accountId);
if (!sessionAccount) {
throw Error("Account does not exist");
}
if (body.interestedIds.length) {
sessionAccount.interestedIds = body.interestedIds;
} else {
delete sessionAccount.interestedIds;
}
await writeAccounts(accounts);
const counts = new Map();
for (const account of accounts)
if (account.interestedIds)
for (const id of account.interestedIds)
counts.set(id, (counts.get(id) ?? 0) + 1);
const schedule = await readSchedule();
for (const event of schedule.events) {
event.interested = counts.get(event.id);
for (const slot of event.slots) {
slot.interested = counts.get(slot.id);
}
}
await writeSchedule(schedule);
broadcastUpdate(schedule);
})

View file

@ -175,5 +175,40 @@ export function generateDemoAccounts(): Account[] {
type: (["regular", "crew", "admin"] as const)[Math.floor(random() ** 5 * 3)],
});
}
// These have a much higher probability of being in someone's interested list.
const desiredEvent = ["opening", "closing", "fursuit-games"];
for (const account of accounts) {
const interestedIds: string[] = [];
for (const id of desiredEvent) {
if (random() < 0.5) {
interestedIds.push(id);
}
}
const eventsToAdd = Math.floor(random() * 10);
while (interestedIds.length < eventsToAdd) {
const event = events[Math.floor(random() * events.length)];
const eventId = toId(event.name);
if (interestedIds.some(id => id.replace(/-\d+$/, "") === eventId)) {
continue;
}
if (event.slots.length === 1 || random() < 0.8) {
interestedIds.push(toId(event.name))
} else {
for (const index of event.slots.map((_, index) => index)) {
if (random() < 0.5) {
interestedIds.push(toId(`${toId(event.name)}-${index}`));
}
}
}
}
if (interestedIds.length) {
account.interestedIds = interestedIds;
}
}
return accounts;
}

View file

@ -3,6 +3,7 @@ export interface Account {
type: "anonymous" | "regular" | "crew" | "admin",
/** Name of the account. Not present on anonymous accounts */
name?: string,
interestedIds?: string[],
}
export interface Subscription {

View file

@ -4,6 +4,7 @@ export interface ScheduleEvent {
host?: string,
cancelled?: boolean,
description?: string,
interested?: number,
slots: TimeSlot[],
}
@ -18,6 +19,7 @@ export interface TimeSlot {
start: string,
end: string,
locations: string[],
interested?: number,
}
export interface Schedule {