165 lines
5 KiB
TypeScript
165 lines
5 KiB
TypeScript
import {useEffect, useState, useMemo} from 'react';
|
|
import Head from 'next/head';
|
|
import Link from 'next/link';
|
|
import prisma from '../../lib/prisma';
|
|
import dayjs, { Dayjs } from 'dayjs';
|
|
import { ClockIcon, GlobeIcon, ChevronDownIcon, ChevronLeftIcon, ChevronRightIcon } from '@heroicons/react/solid';
|
|
import isSameOrBefore from 'dayjs/plugin/isSameOrBefore';
|
|
import utc from 'dayjs/plugin/utc';
|
|
import timezone from 'dayjs/plugin/timezone';
|
|
dayjs.extend(isSameOrBefore);
|
|
dayjs.extend(utc);
|
|
dayjs.extend(timezone);
|
|
|
|
import {collectPageParameters, telemetryEventTypes, useTelemetry} from "../../lib/telemetry";
|
|
import AvailableTimes from "../../components/booking/AvailableTimes";
|
|
import TimeOptions from "../../components/booking/TimeOptions"
|
|
import Avatar from '../../components/Avatar';
|
|
import {timeZone} from "../../lib/clock";
|
|
import DatePicker from "../../components/booking/DatePicker";
|
|
import PoweredByCalendso from "../../components/ui/PoweredByCalendso";
|
|
import {useRouter} from "next/router";
|
|
|
|
export default function Type(props) {
|
|
|
|
const router = useRouter();
|
|
const { rescheduleUid } = router.query;
|
|
|
|
const [selectedDate, setSelectedDate] = useState<Dayjs>();
|
|
const [isTimeOptionsOpen, setIsTimeOptionsOpen] = useState(false);
|
|
const [timeFormat, setTimeFormat] = useState('hh:mm');
|
|
const telemetry = useTelemetry();
|
|
|
|
useEffect(() => {
|
|
telemetry.withJitsu((jitsu) => jitsu.track(telemetryEventTypes.pageView, collectPageParameters()))
|
|
}, []);
|
|
|
|
const changeDate = (date: Dayjs) => {
|
|
telemetry.withJitsu((jitsu) => jitsu.track(telemetryEventTypes.dateSelected, collectPageParameters()))
|
|
setSelectedDate(date);
|
|
};
|
|
|
|
const handleSelectTimeZone = (selectedTimeZone: string) => {
|
|
if (selectedDate) {
|
|
setSelectedDate(selectedDate.tz(selectedTimeZone))
|
|
}
|
|
};
|
|
|
|
const handleToggle24hClock = (is24hClock: boolean) => {
|
|
if (selectedDate) {
|
|
setTimeFormat(is24hClock ? 'HH:mm' : 'h:mma');
|
|
}
|
|
}
|
|
|
|
return (
|
|
<div>
|
|
<Head>
|
|
<title>
|
|
{rescheduleUid && "Reschedule"} {props.eventType.title} | {props.user.name || props.user.username} |
|
|
Calendso
|
|
</title>
|
|
<link rel="icon" href="/favicon.ico" />
|
|
</Head>
|
|
<main
|
|
className={
|
|
"mx-auto my-24 transition-max-width ease-in-out duration-500 " +
|
|
(selectedDate ? "max-w-6xl" : "max-w-3xl")
|
|
}
|
|
>
|
|
<div className="bg-white shadow rounded-lg">
|
|
<div className="sm:flex px-4 py-5 sm:p-4">
|
|
<div
|
|
className={
|
|
"pr-8 sm:border-r " + (selectedDate ? "sm:w-1/3" : "sm:w-1/2")
|
|
}
|
|
>
|
|
<Avatar user={props.user} className="w-16 h-16 rounded-full mb-4" />
|
|
<h2 className="font-medium text-gray-500">{props.user.name}</h2>
|
|
<h1 className="text-3xl font-semibold text-gray-800 mb-4">
|
|
{props.eventType.title}
|
|
</h1>
|
|
<p className="text-gray-500 mb-1 px-2 py-1 -ml-2">
|
|
<ClockIcon className="inline-block w-4 h-4 mr-1 -mt-1" />
|
|
{props.eventType.length} minutes
|
|
</p>
|
|
<button
|
|
onClick={() => setIsTimeOptionsOpen(true)}
|
|
className="text-gray-500 mb-1 px-2 py-1 -ml-2"
|
|
>
|
|
<GlobeIcon className="inline-block w-4 h-4 mr-1 -mt-1" />
|
|
{timeZone()}
|
|
<ChevronDownIcon className="inline-block w-4 h-4 ml-1 -mt-1" />
|
|
</button>
|
|
{ isTimeOptionsOpen && <TimeOptions onSelectTimeZone={handleSelectTimeZone}
|
|
onToggle24hClock={handleToggle24hClock} />}
|
|
<p className="text-gray-600 mt-3 mb-8">
|
|
{props.eventType.description}
|
|
</p>
|
|
</div>
|
|
<DatePicker weekStart={props.user.weekStart} onDatePicked={changeDate} />
|
|
{selectedDate && <AvailableTimes timeFormat={timeFormat} user={props.user} eventType={props.eventType} date={selectedDate} />}
|
|
</div>
|
|
</div>
|
|
{/* note(peer):
|
|
you can remove calendso branding here, but we'd also appreciate it, if you don't <3
|
|
*/}
|
|
<PoweredByCalendso />
|
|
</main>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export async function getServerSideProps(context) {
|
|
const user = await prisma.user.findFirst({
|
|
where: {
|
|
username: context.query.user,
|
|
},
|
|
select: {
|
|
id: true,
|
|
username: true,
|
|
name: true,
|
|
email: true,
|
|
bio: true,
|
|
avatar: true,
|
|
eventTypes: true,
|
|
startTime: true,
|
|
timeZone: true,
|
|
endTime: true,
|
|
weekStart: true,
|
|
}
|
|
});
|
|
|
|
if (!user) {
|
|
return {
|
|
notFound: true,
|
|
}
|
|
}
|
|
|
|
const eventType = await prisma.eventType.findFirst({
|
|
where: {
|
|
userId: user.id,
|
|
slug: {
|
|
equals: context.query.type,
|
|
},
|
|
},
|
|
select: {
|
|
id: true,
|
|
title: true,
|
|
description: true,
|
|
length: true
|
|
}
|
|
});
|
|
|
|
if (!eventType) {
|
|
return {
|
|
notFound: true,
|
|
}
|
|
}
|
|
|
|
return {
|
|
props: {
|
|
user,
|
|
eventType,
|
|
},
|
|
}
|
|
}
|