From 6a26ab81bc55f744598aa45e2a8a4b9f7d075b54 Mon Sep 17 00:00:00 2001 From: Parker Date: Fri, 22 Dec 2023 16:42:43 -0700 Subject: [PATCH 1/8] Center trot text in footer --- app/routes/_marketing+/index.tsx | 33 +++++++++++++++++--------------- 1 file changed, 18 insertions(+), 15 deletions(-) diff --git a/app/routes/_marketing+/index.tsx b/app/routes/_marketing+/index.tsx index 30355e9..9872605 100644 --- a/app/routes/_marketing+/index.tsx +++ b/app/routes/_marketing+/index.tsx @@ -23,8 +23,8 @@ export default function Index() { />
-
-

+
+

- -
Trot Track is built by: - - opportunity hack logo - +
+ + Trot Track + {' '} + is built by: + + opportunity hack logo +
From 4a76b05462e0935f6764aa86eaa5259ccef3d9df Mon Sep 17 00:00:00 2001 From: Parker Date: Tue, 26 Dec 2023 09:25:11 -0700 Subject: [PATCH 2/8] Add event volunteer statuses to agenda view --- app/components/EventAgenda.tsx | 52 ++++++++++++++++++++++++++++++++++ app/routes/calendar+/index.tsx | 47 +++++++++++++++++++----------- 2 files changed, 83 insertions(+), 16 deletions(-) create mode 100644 app/components/EventAgenda.tsx diff --git a/app/components/EventAgenda.tsx b/app/components/EventAgenda.tsx new file mode 100644 index 0000000..a6c2cdb --- /dev/null +++ b/app/components/EventAgenda.tsx @@ -0,0 +1,52 @@ +import { useUser } from '~/utils/user.ts' +import { volunteerTypes, type EventWithVolunteers } from '~/data.ts' + +type VolunteerTypes = typeof volunteerTypes +type VolunteerType = VolunteerTypes[number] + +interface PositionStatusProps { + event: EventWithVolunteers + volunteerType: VolunteerType +} + +function PositionStatus({ volunteerType, event }: PositionStatusProps) { + const user = useUser() + + const filled = + event[volunteerType.field].length >= event[volunteerType.reqField] + const registeredUsers = event[volunteerType.field].map(user => user.id) + const userIsRegistered = registeredUsers.includes(user.id) + const className = `flex gap-4 justify-between ${ + filled ? 'text-muted-foreground' : '' + }` + const volunteerTypeClass = `capitalize ${ + userIsRegistered ? 'before:content-["✅"] before:pr-1' : '' + }` + + if (event[volunteerType.reqField] > 0) + return ( +
+
{volunteerType.displayName}
+
+ {event[volunteerType.field].length} / {event[volunteerType.reqField]} +
+
+ ) +} + +export function EventAgenda({ event }: { event: EventWithVolunteers }) { + return ( +
+
{event.title}
+
+ {volunteerTypes.map(volunteerType => ( + + ))} +
+
+ ) +} diff --git a/app/routes/calendar+/index.tsx b/app/routes/calendar+/index.tsx index e63a7d5..8740263 100644 --- a/app/routes/calendar+/index.tsx +++ b/app/routes/calendar+/index.tsx @@ -15,7 +15,7 @@ import { type HorseData, type EventWithVolunteers, } from '~/data.ts' -import { useState } from 'react' +import { useMemo, useState } from 'react' import { prisma } from '~/utils/db.server.ts' import { requireUserId } from '~/utils/auth.server.ts' @@ -71,6 +71,7 @@ import { horseDateConflicts, renderHorseConflictMessage, } from '~/utils/cooldown-functions.ts' +import { EventAgenda } from '~/components/EventAgenda.tsx' const locales = { 'en-US': enUS, @@ -139,8 +140,12 @@ const instructorSchema = z const createEventSchema = z.object({ title: z.string().min(1, 'Title is required'), - dates: z.string().regex(new RegExp(/^(\d{4}-\d{2}-\d{2},?\s?)+$/g), 'Invalid dates'), - startTime: z.string().regex(new RegExp(/^\d{2}:\d{2}$/g), 'Invalid start time'), + dates: z + .string() + .regex(new RegExp(/^(\d{4}-\d{2}-\d{2},?\s?)+$/g), 'Invalid dates'), + startTime: z + .string() + .regex(new RegExp(/^\d{2}:\d{2}$/g), 'Invalid start time'), duration: z.coerce.number().gt(0), horses: z.array(horseSchema).optional(), instructor: instructorSchema, @@ -269,7 +274,6 @@ export default function Schedule() { const [filterFlag, setFilterFlag] = useState(false) - const eventsThatNeedHelp = events.filter((event: (typeof events)[number]) => { return ( event.cleaningCrewReq > event.cleaningCrew.length || @@ -284,10 +288,19 @@ export default function Schedule() { setRegisterOpen(!registerOpen) } + const components = useMemo( + () => ({ + agenda: { + event: EventAgenda, + }, + }), + [], + ) + return (
-

Calendar

-
+

Calendar

+
setFilterFlag(!filterFlag)} @@ -297,20 +310,22 @@ export default function Schedule() { Show only events that need more volunteers
- + {userIsAdmin ? ( - - ) : null} + + ) : null} -
+
`Cleaning Crew: ${event.cleaningCrew.length} / ${event.cleaningCrewReq}\nSidewalkers: ${event.sideWalkers.length} / ${event.sideWalkersReq}\nLesson Assistants: ${event.lessonAssistants.length} / ${event.lessonAssistantsReq}\nHorse Leaders: ${event.horseLeaders.length} / ${event.horseLeadersReq}`} + tooltipAccessor={event => + `Cleaning Crew: ${event.cleaningCrew.length} / ${event.cleaningCrewReq}\nSidewalkers: ${event.sideWalkers.length} / ${event.sideWalkersReq}\nLesson Assistants: ${event.lessonAssistants.length} / ${event.lessonAssistantsReq}\nHorse Leaders: ${event.horseLeaders.length} / ${event.horseLeadersReq}` + } startAccessor="start" endAccessor="end" onSelectEvent={handleSelectEvent} - style={{ + style={{ height: '95%', width: '95%', backgroundColor: 'white', @@ -318,17 +333,17 @@ export default function Schedule() { padding: 20, borderRadius: '1.5rem', }} + components={components} + defaultView="agenda" />
- + - -
) } @@ -557,7 +572,7 @@ function CreateEventDialog({ horses, instructors }: CreateEventDialogProps) { return ( -