132 lines
3.6 KiB
TypeScript
132 lines
3.6 KiB
TypeScript
import clsx from 'clsx'
|
|
import { useRouter, NextRouter } from 'next/router'
|
|
import { ReactNode, useState } from 'react'
|
|
import { track } from '@amplitude/analytics-browser'
|
|
import { Col } from './col'
|
|
import { Tooltip } from 'web/components/tooltip'
|
|
|
|
type Tab = {
|
|
title: string
|
|
tabIcon?: ReactNode
|
|
content: ReactNode
|
|
// If set, show a badge with this content
|
|
badge?: string
|
|
tooltip?: string
|
|
}
|
|
|
|
type TabProps = {
|
|
tabs: Tab[]
|
|
labelClassName?: string
|
|
onClick?: (tabTitle: string, index: number) => void
|
|
className?: string
|
|
currentPageForAnalytics?: string
|
|
}
|
|
|
|
export function ControlledTabs(props: TabProps & { activeIndex: number }) {
|
|
const {
|
|
tabs,
|
|
activeIndex,
|
|
labelClassName,
|
|
onClick,
|
|
className,
|
|
currentPageForAnalytics,
|
|
} = props
|
|
return (
|
|
<>
|
|
<nav
|
|
className={clsx('space-x-8 border-b border-gray-200', className)}
|
|
aria-label="Tabs"
|
|
>
|
|
{tabs.map((tab, i) => (
|
|
<a
|
|
href="#"
|
|
key={tab.title}
|
|
onClick={(e) => {
|
|
e.preventDefault()
|
|
track('Clicked Tab', {
|
|
title: tab.title,
|
|
currentPage: currentPageForAnalytics,
|
|
})
|
|
onClick?.(tab.title, i)
|
|
}}
|
|
className={clsx(
|
|
activeIndex === i
|
|
? 'border-indigo-500 text-indigo-600'
|
|
: 'border-transparent text-gray-500 hover:border-gray-300 hover:text-gray-700',
|
|
'inline-flex cursor-pointer flex-row gap-1 whitespace-nowrap border-b-2 px-1 py-3 text-sm font-medium',
|
|
labelClassName
|
|
)}
|
|
aria-current={activeIndex === i ? 'page' : undefined}
|
|
>
|
|
{tab.badge ? (
|
|
<span className="px-0.5 font-bold">{tab.badge}</span>
|
|
) : null}
|
|
<Col>
|
|
<Tooltip text={tab.tooltip}>
|
|
{tab.tabIcon && <div className="mx-auto">{tab.tabIcon}</div>}
|
|
{tab.title}
|
|
</Tooltip>
|
|
</Col>
|
|
</a>
|
|
))}
|
|
</nav>
|
|
{tabs.map((tab, i) => (
|
|
<div key={i} className={i === activeIndex ? 'block' : 'hidden'}>
|
|
{tab.content}
|
|
</div>
|
|
))}
|
|
</>
|
|
)
|
|
}
|
|
|
|
export function UncontrolledTabs(props: TabProps & { defaultIndex?: number }) {
|
|
const { defaultIndex, onClick, ...rest } = props
|
|
const [activeIndex, setActiveIndex] = useState(defaultIndex ?? 0)
|
|
return (
|
|
<ControlledTabs
|
|
{...rest}
|
|
activeIndex={activeIndex}
|
|
onClick={(title, i) => {
|
|
setActiveIndex(i)
|
|
onClick?.(title, i)
|
|
}}
|
|
/>
|
|
)
|
|
}
|
|
|
|
const isTabSelected = (router: NextRouter, queryParam: string, tab: Tab) => {
|
|
const selected = router.query[queryParam]
|
|
if (typeof selected === 'string') {
|
|
return tab.title.toLowerCase() === selected
|
|
} else {
|
|
return false
|
|
}
|
|
}
|
|
|
|
export function QueryUncontrolledTabs(
|
|
props: TabProps & { defaultIndex?: number }
|
|
) {
|
|
const { tabs, defaultIndex, onClick, ...rest } = props
|
|
const router = useRouter()
|
|
const selectedIdx = tabs.findIndex((t) => isTabSelected(router, 'tab', t))
|
|
const activeIndex = selectedIdx !== -1 ? selectedIdx : defaultIndex ?? 0
|
|
return (
|
|
<ControlledTabs
|
|
{...rest}
|
|
tabs={tabs}
|
|
activeIndex={activeIndex}
|
|
onClick={(title, i) => {
|
|
router.replace(
|
|
{ query: { ...router.query, tab: title.toLowerCase() } },
|
|
undefined,
|
|
{ shallow: true }
|
|
)
|
|
onClick?.(title, i)
|
|
}}
|
|
/>
|
|
)
|
|
}
|
|
|
|
// legacy code that didn't know about any other kind of tabs imports this
|
|
export const Tabs = UncontrolledTabs
|