diff --git a/app/(protected)/my-appointments/components/appointmentDateTimePicker/appointmentDateTimePicker.tsx b/app/(protected)/my-appointments/components/appointmentDateTimePicker/appointmentDateTimePicker.tsx index cd76854..c5641e6 100644 --- a/app/(protected)/my-appointments/components/appointmentDateTimePicker/appointmentDateTimePicker.tsx +++ b/app/(protected)/my-appointments/components/appointmentDateTimePicker/appointmentDateTimePicker.tsx @@ -7,7 +7,7 @@ import { mutate } from "swr"; import AppointmentTimeslots from "./appointmentTimeslots"; interface AppointmentDateTimePickerProps { - availableDates: string[]; + availableDates?: string[]; } function AppointmentDateTimePicker({ diff --git a/app/(protected)/my-appointments/components/appointmentDateTimePicker/appointmentTimeslots.tsx b/app/(protected)/my-appointments/components/appointmentDateTimePicker/appointmentTimeslots.tsx index 4e4edde..7d4e7dc 100644 --- a/app/(protected)/my-appointments/components/appointmentDateTimePicker/appointmentTimeslots.tsx +++ b/app/(protected)/my-appointments/components/appointmentDateTimePicker/appointmentTimeslots.tsx @@ -1,6 +1,8 @@ import { Badge } from "@/components/atoms/badge"; import { FormControl } from "@/components/molecules/form"; import { TimeslotSchema } from "@/lib/formSchema"; +import { formatToDisplayTime } from "@/lib/utils"; +import { format } from "date-fns"; import { useState } from "react"; import { useController } from "react-hook-form"; import useSWR from "swr"; @@ -27,7 +29,6 @@ function AppointmentTimeslots({ currentDate }: AppointmentTimeslotsProps) { e: React.SyntheticEvent, timeslot: z.infer, ) => { - // getValues('timeslot').id field.onChange(timeslot); if (selected !== timeslot.id) { return setSelected(timeslot.id); @@ -52,7 +53,7 @@ function AppointmentTimeslots({ currentDate }: AppointmentTimeslotsProps) { className="shrink flex-grow-0" key={timeslot.id} selected={selected === timeslot.id} - label={timeslot.time} + label={formatToDisplayTime(timeslot.time)} onClick={(e) => handleSelect(e, timeslot)} /> )) diff --git a/app/(protected)/my-appointments/components/appointmentForm/appointmentForm.tsx b/app/(protected)/my-appointments/components/appointmentForm/appointmentForm.tsx index ddbb317..4510dc7 100644 --- a/app/(protected)/my-appointments/components/appointmentForm/appointmentForm.tsx +++ b/app/(protected)/my-appointments/components/appointmentForm/appointmentForm.tsx @@ -17,7 +17,7 @@ import { } from "@/components/molecules/form"; import { useToast } from "@/components/molecules/toast"; import { FormSchema } from "@/lib/formSchema"; -import { cn } from "@/lib/utils"; +import { cn, formatToDisplayDate, formatToDisplayTime } from "@/lib/utils"; import AppointmentDateTimePicker from "../appointmentDateTimePicker/appointmentDateTimePicker"; /** @@ -42,10 +42,10 @@ async function sendEmail(formValues: z.infer) { const res = await fetch("/api/email", { method: "POST", headers: { "Content-type": "application/json" }, - // This is a mock data. Replace with proper form values later. + body: JSON.stringify({ - date: new Date(timeslot.date).toUTCString().slice(0, 16), - time: format(new Date(`${timeslot.date} ${timeslot.time}`), "p"), + date: formatToDisplayDate(timeslot.date), + time: formatToDisplayTime(timeslot.time), location, comment, }), @@ -71,8 +71,6 @@ interface AppointmentFormProps { onClose?: () => void; } -const AVAILABLE_DATES = ["2024-05-06", "2024-05-02", "2024-05-13"]; - /** * A client side form component that handles both creating and editing appointments. */ @@ -143,9 +141,7 @@ export const AppointmentForm = ({ Pick your appointment date - + diff --git a/app/(protected)/my-appointments/components/appointments/upcomingAppointments.tsx b/app/(protected)/my-appointments/components/appointments/upcomingAppointments.tsx index 84476e5..178ea21 100644 --- a/app/(protected)/my-appointments/components/appointments/upcomingAppointments.tsx +++ b/app/(protected)/my-appointments/components/appointments/upcomingAppointments.tsx @@ -8,6 +8,7 @@ import CancelDialog from "../cancelDialog/cancelDialog"; import { getKindeServerSession } from "@kinde-oss/kinde-auth-nextjs/server"; import { getAppointments } from "@/lib/sanity/client"; import { format } from "date-fns"; +import { formatToDisplayDate, formatToDisplayTime } from "@/lib/utils"; export default async function UpcomingAppointments() { const { getUser } = getKindeServerSession(); @@ -33,11 +34,11 @@ export default async function UpcomingAppointments() {

- {new Date(date).toUTCString().slice(0, 16)} + {formatToDisplayDate(date)}

- {format(new Date(`${date} ${time}`), "p")} + {formatToDisplayTime(time)}

{[address1, address2, city, state, zipCode] diff --git a/app/api/timeslots/[date]/route.ts b/app/api/timeslots/[date]/route.ts index a5ea925..21384a1 100644 --- a/app/api/timeslots/[date]/route.ts +++ b/app/api/timeslots/[date]/route.ts @@ -1,4 +1,4 @@ -import { getAvailableTimeSlot } from "@/lib/sanity/client"; +import { getAvailableTimeSlots } from "@/lib/sanity/client"; import { formatISO } from "date-fns"; import { NextRequest } from "next/server"; @@ -12,6 +12,6 @@ export async function GET( representation: "date", }); - const res = await getAvailableTimeSlot(formattedDate); + const res = await getAvailableTimeSlots(formattedDate); return Response.json(res, { status: 200 }); } diff --git a/lib/sanity/client.ts b/lib/sanity/client.ts index 7f84d4c..19829b5 100644 --- a/lib/sanity/client.ts +++ b/lib/sanity/client.ts @@ -46,16 +46,17 @@ export const getAvailableDate = async () => { return Object.keys(distinctDates); }; -export const getAvailableTimeSlot = async (date: string) => { +export const getAvailableTimeSlots = async (date: string) => { const timeSlots = await client.fetch( `*[_type=='timeslot' - && date=='${date}' && !(_id in *[_type=='appointment'].timeslot._ref) - ]{ - "id": _id, - date, - "time": duration.start - }`, + && date == '${date}' + && dateTime(time) >= dateTime(now()) + ]{ + "id": _id, + date, + time, + }`, {}, { cache: "no-store" }, ); @@ -79,12 +80,12 @@ export const getAppointments = async (userId = "") => { const res = await client.fetch( `*[_type=='appointment' && customer->_id == '${userId}' - && timeslot->date >= now() + && dateTime(timeslot->time) >= dateTime(now()) ]{ "id":_id, "timeslotId":timeslot->_id, "date":timeslot->date, - "time":timeslot->duration.start, + "time":timeslot->time, address1, address2, city, diff --git a/lib/sanity/sanity.types.ts b/lib/sanity/sanity.types.ts index 766cce3..ec74b44 100644 --- a/lib/sanity/sanity.types.ts +++ b/lib/sanity/sanity.types.ts @@ -262,18 +262,19 @@ export type AVAILABLE_DATE_QUERYResult = Array<{ id: string; date: string; }>; +// Source: ./groq/groq.ts // Variable: AVAILABLE_TIMESLOT_QUERY -// Query: *[_type=='timeslot' && date=='' && !(_id in *[_type=='appointment'].timeslot._ref)]{ "id": _id, date, "startTime": duration.start} +// Query: *[_type=='timeslot' && !(_id in *[_type=='appointment'].timeslot._ref) && date == '' && dateTime(timeslot) >= dateTime(now())]{ "id": _id, date, timeslot,} export type AVAILABLE_TIMESLOT_QUERYResult = Array<{ id: string; date: string; - time: string; + timeslot: string; }>; // Variable: IS_TIMESLOT_RESERVED_QUERY // Query: count(*[_type=='appointment' && references('') ]) > 0 export type IS_TIMESLOT_RESERVED_QUERYResult = unknown; // Variable: APPOINTMENT_QUERY -// Query: *[_type=='appointment' && customer->_id == '']{ "id":_id, "timeslotId":timeslot->_id, "date":timeslot->date, "time":timeslot->duration.start, address1, address2, city, state, zipCode, comment, customer->{"id": _id, firstName, lastName}, stylist->{"id": _id, firstName, lastName}} +// Query: *[_type=='appointment' && customer->_id == '' && dateTime(timeslot->timeslot) >= dateTime(now())]{ "id":_id, "timeslotId":timeslot->_id, "date":timeslot->date, "time":timeslot->timeslot, address1, address2, city, state, zipCode, comment, customer->{"id": _id, firstName, lastName}, stylist->{"id": _id, firstName, lastName}} export type APPOINTMENT_QUERYResult = Array<{ id: string; timeslotId: string; diff --git a/lib/utils.ts b/lib/utils.ts index 289d9e6..ce2f99c 100644 --- a/lib/utils.ts +++ b/lib/utils.ts @@ -1,16 +1,24 @@ import { type ClassValue, clsx } from "clsx"; import { twMerge } from "tailwind-merge"; import { get, set, ref, child } from "firebase/database"; +import { format } from "date-fns"; export function cn(...inputs: ClassValue[]) { return twMerge(clsx(inputs)); } -export function dateTimeToUnixTimeStamp(date: string, time: string) { - const combinedDateTimeString = date + "T" + time; - const dateTime = new Date(combinedDateTimeString); - const unixTimeStamp = dateTime.getTime() / 1000; - return unixTimeStamp; +/** Convert incoming date string to UTC format. This only returns date portion. */ +export function formatToDisplayDate(date: string): string { + const res = new Date(date).toUTCString().slice(0, 16); + + return res; +} + +/** Convert incoming date string to HH:mm AM/PM format*/ +export function formatToDisplayTime(time: string): string { + const res = format(new Date(`${time}`), "p"); + + return res; } export function unixToDateTimeStrings(unixTimeStamp: number) {