cal.pub0.org/components/Shell.tsx

385 lines
15 KiB
TypeScript
Raw Normal View History

2021-08-02 17:40:13 +00:00
import { Menu, Transition } from "@headlessui/react";
import { SelectorIcon } from "@heroicons/react/outline";
2021-07-30 23:05:38 +00:00
import {
CalendarIcon,
ClockIcon,
CogIcon,
2021-07-30 23:05:38 +00:00
ExternalLinkIcon,
LinkIcon,
LogoutIcon,
PuzzleIcon,
2021-07-30 23:05:38 +00:00
} from "@heroicons/react/solid";
import { signOut, useSession } from "next-auth/client";
import Link from "next/link";
import { useRouter } from "next/router";
import React, { Fragment, ReactNode, useEffect } from "react";
import { Toaster } from "react-hot-toast";
import LicenseBanner from "@ee/components/LicenseBanner";
import HelpMenuItemDynamic from "@ee/lib/intercom/HelpMenuItemDynamic";
import classNames from "@lib/classNames";
import { collectPageParameters, telemetryEventTypes, useTelemetry } from "@lib/telemetry";
import { trpc } from "@lib/trpc";
import { HeadSeo } from "@components/seo/head-seo";
import Avatar from "@components/ui/Avatar";
import Logo from "./Logo";
2021-03-24 15:03:04 +00:00
function useMeQuery() {
const [session] = useSession();
const meQuery = trpc.useQuery(["viewer.me"], {
// refetch max once per 5s
staleTime: 5000,
});
useEffect(() => {
// refetch if sesion changes
meQuery.refetch();
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [session]);
return meQuery;
}
function useRedirectToLoginIfUnauthenticated() {
const [session, loading] = useSession();
const router = useRouter();
useEffect(() => {
if (!loading && !session) {
router.replace({
pathname: "/auth/login",
query: {
callbackUrl: `${location.pathname}${location.search}`,
},
});
}
}, [loading, session, router]);
}
export default function Shell(props: {
centered?: boolean;
title?: string;
heading: ReactNode;
subtitle?: ReactNode;
children: ReactNode;
CTA?: ReactNode;
}) {
const router = useRouter();
// eslint-disable-next-line @typescript-eslint/no-unused-vars
useRedirectToLoginIfUnauthenticated();
const telemetry = useTelemetry();
const query = useMeQuery();
2021-07-30 23:05:38 +00:00
const navigation = [
{
name: "Event Types",
href: "/event-types",
icon: LinkIcon,
current: router.asPath.startsWith("/event-types"),
2021-07-30 23:05:38 +00:00
},
{
name: "Bookings",
href: "/bookings/upcoming",
2021-07-30 23:05:38 +00:00
icon: ClockIcon,
current: router.asPath.startsWith("/bookings"),
2021-07-30 23:05:38 +00:00
},
{
name: "Availability",
href: "/availability",
icon: CalendarIcon,
current: router.asPath.startsWith("/availability"),
2021-07-30 23:05:38 +00:00
},
{
name: "Integrations",
2021-07-30 23:05:38 +00:00
href: "/integrations",
icon: PuzzleIcon,
current: router.asPath.startsWith("/integrations"),
2021-07-30 23:05:38 +00:00
},
{
name: "Settings",
href: "/settings/profile",
icon: CogIcon,
current: router.asPath.startsWith("/settings"),
},
2021-07-30 23:05:38 +00:00
];
useEffect(() => {
telemetry.withJitsu((jitsu) => {
return jitsu.track(telemetryEventTypes.pageView, collectPageParameters(router.asPath));
});
}, [telemetry]);
2021-03-24 15:03:04 +00:00
if (query.status !== "loading" && !query.data) {
router.replace("/auth/login");
}
const pageTitle = typeof props.heading === "string" ? props.heading : props.title;
return (
2021-08-03 11:13:48 +00:00
<>
<HeadSeo
title={pageTitle ?? "Cal.com"}
description={props.subtitle ? props.subtitle?.toString() : ""}
nextSeoProps={{
nofollow: true,
noindex: true,
}}
/>
<div>
<Toaster position="bottom-right" />
</div>
2021-08-03 11:13:48 +00:00
<div className="h-screen flex overflow-hidden bg-gray-100">
<div className="hidden md:flex md:flex-shrink-0">
<div className="flex flex-col w-56">
2021-08-03 11:13:48 +00:00
<div className="flex flex-col h-0 flex-1 border-r border-gray-200 bg-white">
<div className="flex-1 flex flex-col pt-5 pb-4 overflow-y-auto">
<Link href="/event-types">
<a className="px-4">
<Logo small />
</a>
</Link>
<nav className="mt-5 flex-1 px-2 bg-white space-y-1">
{navigation.map((item) => (
<Link key={item.name} href={item.href}>
<a
2021-07-30 23:05:38 +00:00
className={classNames(
2021-08-03 11:13:48 +00:00
item.current
? "bg-neutral-100 text-neutral-900"
: "text-neutral-500 hover:bg-gray-50 hover:text-neutral-900",
"group flex items-center px-2 py-2 text-sm font-medium rounded-sm"
)}>
<item.icon
className={classNames(
item.current
? "text-neutral-500"
: "text-neutral-400 group-hover:text-neutral-500",
"mr-3 flex-shrink-0 h-5 w-5"
)}
aria-hidden="true"
/>
{item.name}
</a>
</Link>
))}
</nav>
</div>
2021-08-05 10:02:06 +00:00
<div className="flex-shrink-0 flex p-4">
<UserDropdown />
2021-08-03 11:13:48 +00:00
</div>
</div>
</div>
2021-07-30 23:05:38 +00:00
</div>
2021-08-03 11:13:48 +00:00
<div className="flex flex-col w-0 flex-1 overflow-hidden">
<main className="flex-1 relative z-0 overflow-y-auto focus:outline-none max-w-[1700px]">
2021-08-03 11:13:48 +00:00
{/* show top navigation for md and smaller (tablet and phones) */}
<nav className="md:hidden bg-white shadow p-4 flex justify-between items-center">
<Link href="/event-types">
<a>
2021-08-03 11:13:48 +00:00
<Logo />
</a>
</Link>
2021-08-03 11:13:48 +00:00
<div className="flex gap-3 items-center self-center">
<button className="bg-white p-2 rounded-full text-gray-400 hover:text-gray-500 hover:bg-gray-50 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-black">
<span className="sr-only">View notifications</span>
<Link href="/settings/profile">
<a>
<CogIcon className="h-6 w-6" aria-hidden="true" />
</a>
</Link>
</button>
<div className="mt-1">
<UserDropdown small bottom />
2021-08-03 11:13:48 +00:00
</div>
</div>
2021-08-03 11:13:48 +00:00
</nav>
<div className={classNames(props.centered && "md:max-w-5xl mx-auto", "py-8")}>
<div className="block sm:flex justify-between px-4 sm:px-6 md:px-8 min-h-[80px]">
2021-10-10 09:46:20 +00:00
<div className="mb-8 w-full">
<h1 className="font-cal text-xl font-bold text-gray-900 tracking-wide mb-1">
{props.heading}
</h1>
2021-08-03 11:13:48 +00:00
<p className="text-sm text-neutral-500 mr-4">{props.subtitle}</p>
</div>
<div className="mb-4 flex-shrink-0">{props.CTA}</div>
</div>
<div className="px-4 sm:px-6 md:px-8">{props.children}</div>
2021-08-03 11:13:48 +00:00
{/* show bottom navigation for md and smaller (tablet and phones) */}
<nav className="bottom-nav md:hidden flex fixed bottom-0 bg-white w-full shadow">
2021-08-03 11:13:48 +00:00
{/* note(PeerRich): using flatMap instead of map to remove settings from bottom nav */}
{navigation.flatMap((item, itemIdx) =>
item.name === "Settings" ? (
[]
) : (
<Link key={item.name} href={item.href}>
<a
className={classNames(
2021-08-03 11:13:48 +00:00
item.current ? "text-gray-900" : "text-neutral-400 hover:text-gray-700",
itemIdx === 0 ? "rounded-l-lg" : "",
itemIdx === navigation.length - 1 ? "rounded-r-lg" : "",
"group relative min-w-0 flex-1 overflow-hidden bg-white py-2 px-2 text-xs sm:text-sm font-medium text-center hover:bg-gray-50 focus:z-10"
)}
2021-08-03 11:13:48 +00:00
aria-current={item.current ? "page" : undefined}>
<item.icon
className={classNames(
item.current ? "text-gray-900" : "text-gray-400 group-hover:text-gray-500",
"block mx-auto flex-shrink-0 h-5 w-5 mb-1 text-center"
)}
aria-hidden="true"
/>
<span>{item.name}</span>
</a>
</Link>
)
)}
</nav>
2021-08-03 11:13:48 +00:00
{/* add padding to content for mobile navigation*/}
<div className="block md:hidden pt-12" />
</div>
<LicenseBanner />
2021-08-03 11:13:48 +00:00
</main>
</div>
</div>
2021-08-03 11:13:48 +00:00
</>
);
}
function UserDropdown({ small, bottom }: { small?: boolean; bottom?: boolean }) {
const query = useMeQuery();
const user = query.data;
return (
<Menu as="div" className="w-full relative inline-block text-left">
{({ open }) => (
<>
<div>
{user && (
<Menu.Button className="group w-full rounded-md text-sm text-left font-medium text-gray-700 focus:outline-none">
<span className="flex w-full justify-between items-center">
<span className="flex min-w-0 items-center justify-between space-x-3">
<Avatar
imageSrc={user.avatar}
alt={user.username}
className={classNames(
small ? "w-8 h-8" : "w-10 h-10",
"bg-gray-300 rounded-full flex-shrink-0"
)}
/>
{!small && (
<span className="flex-1 flex flex-col min-w-0">
<span className="text-gray-900 text-sm font-medium truncate">{user.name}</span>
<span className="text-neutral-500 font-normal text-sm truncate">
/{user.username}
</span>
</span>
)}
</span>
{!small && (
<SelectorIcon
className="flex-shrink-0 h-5 w-5 text-gray-400 group-hover:text-gray-500"
aria-hidden="true"
/>
)}
</span>
</Menu.Button>
)}
</div>
<Transition
show={open}
as={Fragment}
enter="transition ease-out duration-100"
enterFrom="transform opacity-0 scale-95"
enterTo="transform opacity-100 scale-100"
leave="transition ease-in duration-75"
leaveFrom="transform opacity-100 scale-100"
leaveTo="transform opacity-0 scale-95">
<Menu.Items
static
className={classNames(
bottom ? "origin-top top-1 right-0" : "origin-bottom bottom-14 left-0",
"w-64 z-10 absolute mt-1 rounded-md shadow-lg bg-white ring-1 ring-black ring-opacity-5 divide-y divide-gray-200 focus:outline-none"
)}>
<div className="py-1">
<a
target="_blank"
rel="noopener noreferrer"
href={`${process.env.NEXT_PUBLIC_APP_URL}/${user?.username || ""}`}
className="flex px-4 py-2 text-sm text-neutral-500">
2021-08-02 15:36:28 +00:00
View public page <ExternalLinkIcon className="ml-1 mt-1 w-3 h-3 text-neutral-400" />
</a>
</div>
<div className="py-1">
<Menu.Item>
{({ active }) => (
<a
href="https://cal.com/slack"
2021-08-02 17:06:24 +00:00
target="_blank"
2021-08-02 17:40:13 +00:00
rel="noreferrer"
className={classNames(
active ? "bg-gray-100 text-gray-900" : "text-neutral-700",
"flex px-4 py-2 text-sm font-medium"
)}>
2021-08-02 17:06:24 +00:00
<svg
viewBox="0 0 2447.6 2452.5"
className={classNames(
"text-neutral-400 group-hover:text-neutral-500",
2021-08-02 17:06:24 +00:00
"mt-0.5 mr-3 flex-shrink-0 h-4 w-4"
)}
2021-08-02 17:06:24 +00:00
xmlns="http://www.w3.org/2000/svg">
<g clipRule="evenodd" fillRule="evenodd">
<path
d="m897.4 0c-135.3.1-244.8 109.9-244.7 245.2-.1 135.3 109.5 245.1 244.8 245.2h244.8v-245.1c.1-135.3-109.5-245.1-244.9-245.3.1 0 .1 0 0 0m0 654h-652.6c-135.3.1-244.9 109.9-244.8 245.2-.2 135.3 109.4 245.1 244.7 245.3h652.7c135.3-.1 244.9-109.9 244.8-245.2.1-135.4-109.5-245.2-244.8-245.3z"
fill="#9BA6B6"></path>
<path
d="m2447.6 899.2c.1-135.3-109.5-245.1-244.8-245.2-135.3.1-244.9 109.9-244.8 245.2v245.3h244.8c135.3-.1 244.9-109.9 244.8-245.3zm-652.7 0v-654c.1-135.2-109.4-245-244.7-245.2-135.3.1-244.9 109.9-244.8 245.2v654c-.2 135.3 109.4 245.1 244.7 245.3 135.3-.1 244.9-109.9 244.8-245.3z"
fill="#9BA6B6"></path>
<path
d="m1550.1 2452.5c135.3-.1 244.9-109.9 244.8-245.2.1-135.3-109.5-245.1-244.8-245.2h-244.8v245.2c-.1 135.2 109.5 245 244.8 245.2zm0-654.1h652.7c135.3-.1 244.9-109.9 244.8-245.2.2-135.3-109.4-245.1-244.7-245.3h-652.7c-135.3.1-244.9 109.9-244.8 245.2-.1 135.4 109.4 245.2 244.7 245.3z"
fill="#9BA6B6"></path>
<path
d="m0 1553.2c-.1 135.3 109.5 245.1 244.8 245.2 135.3-.1 244.9-109.9 244.8-245.2v-245.2h-244.8c-135.3.1-244.9 109.9-244.8 245.2zm652.7 0v654c-.2 135.3 109.4 245.1 244.7 245.3 135.3-.1 244.9-109.9 244.8-245.2v-653.9c.2-135.3-109.4-245.1-244.7-245.3-135.4 0-244.9 109.8-244.8 245.1 0 0 0 .1 0 0"
fill="#9BA6B6"></path>
</g>
</svg>
Join our Slack
</a>
)}
</Menu.Item>
<HelpMenuItemDynamic />
</div>
<div className="py-1">
<Menu.Item>
{({ active }) => (
<a
onClick={() => signOut({ callbackUrl: "/auth/logout" })}
className={classNames(
active ? "bg-gray-100 text-gray-900" : "text-gray-700",
"flex px-4 py-2 text-sm font-medium"
)}>
<LogoutIcon
className={classNames(
"text-neutral-400 group-hover:text-neutral-500",
"mr-2 flex-shrink-0 h-5 w-5"
)}
aria-hidden="true"
/>
Sign out
</a>
)}
</Menu.Item>
</div>
</Menu.Items>
</Transition>
</>
)}
</Menu>
);
2021-08-02 17:40:13 +00:00
}