manifold/web/components/nav/nav-bar.tsx

207 lines
6.3 KiB
TypeScript
Raw Permalink Normal View History

import Link from 'next/link'
2022-02-01 01:02:17 +00:00
import {
HomeIcon,
MenuAlt3Icon,
2022-02-01 01:02:17 +00:00
SearchIcon,
XIcon,
2022-02-01 01:02:17 +00:00
} from '@heroicons/react/outline'
import { Transition, Dialog } from '@headlessui/react'
import { useState, Fragment } from 'react'
2022-05-18 19:45:08 +00:00
import Sidebar, { Item } from './sidebar'
2022-07-07 21:52:28 +00:00
import { usePrivateUser, useUser } from 'web/hooks/use-user'
import { formatMoney } from 'common/util/format'
import { Avatar } from '../avatar'
2022-05-18 19:45:08 +00:00
import clsx from 'clsx'
import { useRouter } from 'next/router'
import NotificationsIcon from 'web/components/notifications-icon'
import { useIsIframe } from 'web/hooks/use-is-iframe'
import { trackCallback } from 'web/lib/service/analytics'
2022-07-07 21:52:28 +00:00
import { useUnseenPreferredNotifications } from 'web/hooks/use-notifications'
import { PrivateUser } from 'common/user'
2022-05-18 19:45:08 +00:00
function getNavigation() {
2022-05-18 19:45:08 +00:00
return [
{ name: 'Home', href: '/home', icon: HomeIcon },
{
name: 'Notifications',
href: `/notifications`,
icon: NotificationsIcon,
},
2022-05-18 19:45:08 +00:00
]
}
const signedOutNavigation = [
{ name: 'Home', href: '/', icon: HomeIcon },
{ name: 'Explore', href: '/markets', icon: SearchIcon },
]
2022-03-31 03:51:15 +00:00
// From https://codepen.io/chris__sev/pen/QWGvYbL
export function BottomNavBar() {
const [sidebarOpen, setSidebarOpen] = useState(false)
2022-05-18 19:45:08 +00:00
const router = useRouter()
const currentPage = router.pathname
const user = useUser()
2022-07-07 21:52:28 +00:00
const privateUser = usePrivateUser(user?.id)
const isIframe = useIsIframe()
if (isIframe) {
return null
}
2022-05-18 19:45:08 +00:00
const navigationOptions =
user === null ? signedOutNavigation : getNavigation()
2022-05-18 19:45:08 +00:00
2022-02-01 01:02:17 +00:00
return (
<nav className="fixed inset-x-0 bottom-0 z-20 flex justify-between border-t-2 bg-white text-xs text-gray-700 lg:hidden">
2022-05-18 19:45:08 +00:00
{navigationOptions.map((item) => (
<NavBarItem key={item.name} item={item} currentPage={currentPage} />
))}
{user && (
<NavBarItem
key={'profile'}
currentPage={currentPage}
item={{
name: formatMoney(user.balance),
2022-07-04 03:43:18 +00:00
trackingEventName: 'profile',
href: `/${user.username}?tab=bets`,
icon: () => (
<Avatar
className="mx-auto my-1"
size="xs"
username={user.username}
avatarUrl={user.avatarUrl}
noLink
/>
),
}}
/>
)}
<div
className="w-full select-none py-1 px-3 text-center hover:cursor-pointer hover:bg-indigo-200 hover:text-indigo-700"
onClick={() => setSidebarOpen(true)}
>
2022-07-07 21:52:28 +00:00
<MenuAlt3Icon className=" my-1 mx-auto h-6 w-6" aria-hidden="true" />
{privateUser ? (
<MoreMenuWithGroupNotifications privateUser={privateUser} />
) : (
'More'
)}
</div>
<MobileSidebar
sidebarOpen={sidebarOpen}
setSidebarOpen={setSidebarOpen}
/>
</nav>
)
}
2022-03-31 03:51:15 +00:00
2022-07-07 21:52:28 +00:00
function MoreMenuWithGroupNotifications(props: { privateUser: PrivateUser }) {
const { privateUser } = props
const preferredNotifications = useUnseenPreferredNotifications(privateUser, {
customHref: '/group/',
})
return (
<span
className={
preferredNotifications.length > 0 ? 'font-bold' : 'font-normal'
}
>
More
</span>
)
}
2022-05-18 19:45:08 +00:00
function NavBarItem(props: { item: Item; currentPage: string }) {
const { item, currentPage } = props
2022-07-04 03:43:18 +00:00
const track = trackCallback(`navbar: ${item.trackingEventName ?? item.name}`)
2022-05-18 19:45:08 +00:00
return (
<Link href={item.href}>
<a
className={clsx(
'block w-full py-1 px-3 text-center hover:bg-indigo-200 hover:text-indigo-700',
currentPage === item.href && 'bg-gray-200 text-indigo-700'
2022-05-18 19:45:08 +00:00
)}
2022-07-04 03:43:18 +00:00
onClick={track}
2022-05-18 19:45:08 +00:00
>
{item.icon && <item.icon className="my-1 mx-auto h-6 w-6" />}
2022-05-18 19:45:08 +00:00
{item.name}
</a>
</Link>
)
}
// Sidebar that slides out on mobile
export function MobileSidebar(props: {
sidebarOpen: boolean
setSidebarOpen: (open: boolean) => void
2022-03-31 03:51:15 +00:00
}) {
const { sidebarOpen, setSidebarOpen } = props
2022-03-31 03:51:15 +00:00
return (
<div>
<Transition.Root show={sidebarOpen} as={Fragment}>
<Dialog
as="div"
className="fixed inset-0 z-40 flex"
onClose={setSidebarOpen}
2022-03-31 03:51:15 +00:00
>
<Transition.Child
as={Fragment}
enter="transition-opacity ease-linear duration-300"
enterFrom="opacity-0"
enterTo="opacity-100"
leave="transition-opacity ease-linear duration-300"
leaveFrom="opacity-100"
leaveTo="opacity-0"
>
<Dialog.Overlay className="fixed inset-0 bg-gray-600 bg-opacity-75" />
</Transition.Child>
<Transition.Child
as={Fragment}
enter="transition ease-in-out duration-300 transform"
enterFrom="-translate-x-full"
enterTo="translate-x-0"
leave="transition ease-in-out duration-300 transform"
leaveFrom="translate-x-0"
leaveTo="-translate-x-full"
2022-03-31 03:51:15 +00:00
>
<div className="relative flex w-full max-w-xs flex-1 flex-col bg-white">
<Transition.Child
as={Fragment}
enter="ease-in-out duration-300"
enterFrom="opacity-0"
enterTo="opacity-100"
leave="ease-in-out duration-300"
leaveFrom="opacity-100"
leaveTo="opacity-0"
>
<div className="absolute top-0 right-0 -mr-12 pt-2">
<button
type="button"
className="ml-1 flex h-10 w-10 items-center justify-center rounded-full focus:outline-none focus:ring-2 focus:ring-inset focus:ring-white"
onClick={() => setSidebarOpen(false)}
>
<span className="sr-only">Close sidebar</span>
<XIcon className="h-6 w-6 text-white" aria-hidden="true" />
</button>
</div>
</Transition.Child>
<div className="mx-2 h-0 flex-1 overflow-y-auto">
<Sidebar className="pl-2" />
</div>
</div>
</Transition.Child>
<div className="w-14 flex-shrink-0" aria-hidden="true">
{/* Dummy element to force sidebar to shrink to fit close icon */}
</div>
</Dialog>
</Transition.Root>
</div>
2022-03-31 03:51:15 +00:00
)
}