Files
supabase/apps/studio/components/layouts/Navigation/NavigationBar/NavigationIconLink.tsx
Ivan Vasilov 56de26fe22 chore: Migrate the monorepo to use Tailwind v4 (#45318)
This PR migrates the whole monorepo to use Tailwind v4:
- Removed `@tailwindcss/container-queries` plugin since it's included by
default in v4,
- Bump all instances of Tailwind to v4. Made minimal changes to the
shared config to remove non-supported features (`alpha` mentions),
- Migrate all apps to be compatible with v4 configs,
- Fix the `typography.css` import in 3 apps,
- Add missing rules which were included by default in v3,
- Run `pnpm dlx @tailwindcss/upgrade` on all apps, which renames a lot
of classes
- Rename all misnamed classes according to
https://tailwindcss.com/docs/upgrade-guide#renamed-utilities in all
apps.

---------

Co-authored-by: Jordi Enric <jordi.err@gmail.com>
2026-04-30 10:53:24 +00:00

110 lines
3.4 KiB
TypeScript

import { LOCAL_STORAGE_KEYS } from 'common'
import { noop } from 'lodash'
import Link from 'next/link'
import {
AnchorHTMLAttributes,
cloneElement,
ComponentPropsWithoutRef,
forwardRef,
isValidElement,
} from 'react'
import { cn, Tooltip, TooltipContent, TooltipTrigger } from 'ui'
import type { Route } from '@/components/ui/ui.types'
import { useLocalStorageQuery } from '@/hooks/misc/useLocalStorage'
interface NavigationIconButtonProps extends AnchorHTMLAttributes<HTMLAnchorElement> {
route: Route
isActive?: boolean
}
const NavigationIconLink = forwardRef<HTMLAnchorElement, NavigationIconButtonProps>(
({ route, isActive = false, onClick = noop, ...props }, ref) => {
const [storedAllowNavPanel] = useLocalStorageQuery(
LOCAL_STORAGE_KEYS.EXPAND_NAVIGATION_PANEL,
true
)
// Don't allow the nav panel to expand in playwright tests
const allowNavPanelToExpand = process.env.NEXT_PUBLIC_NODE_ENV !== 'test' && storedAllowNavPanel
const iconClasses = [
'absolute left-0 top-0 flex rounded-sm h-10 w-10 items-center justify-center text-foreground-lighter', // Layout
'group-hover/item:text-foreground-light',
isActive ? 'text-foreground! [&_svg]:stroke-[1.5]' : '[&_svg]:stroke-1',
'transition-all',
]
const classes = [
'relative',
'h-10 w-full md:w-10 md:group-data-[state=expanded]:w-full',
'transition-all duration-200',
'flex items-center rounded-sm',
'group-data-[state=collapsed]:justify-center',
'group-data-[state=expanded]:-space-x-2',
'hover:bg-surface-200',
'group/item',
`${isActive && 'bg-selection! shadow-xs'}`,
]
const LinkComponent = forwardRef<HTMLAnchorElement, ComponentPropsWithoutRef<typeof Link>>(
function LinkComponent(props, ref) {
if (route.linkElement && isValidElement(route.linkElement)) {
return cloneElement<any>(route.linkElement, { ...props, ref })
}
return <Link ref={ref} {...props} />
}
)
const linkContent = (
<LinkComponent
role="button"
aria-current={isActive}
ref={ref}
href={route.link || '#'} // Provide a fallback href
{...props}
onClick={(e) => {
if (!route.link) {
e.preventDefault() // Prevent navigation if there's no link
}
onClick(e)
}}
className={cn(classes, props.className)}
>
<span id="icon-link" className={cn(...iconClasses)} {...props}>
{route.icon}
</span>
<span
className={cn(
'min-w-[128px] text-sm text-foreground-light',
'group-hover/item:text-foreground',
'group-aria-current/item:text-foreground',
'absolute left-10 md:left-7 md:group-data-[state=expanded]:left-12',
'opacity-100 md:opacity-0 md:group-data-[state=expanded]:opacity-100',
`${isActive && 'text-foreground hover:text-foreground'}`,
'transition-all'
)}
>
{route.label}
</span>
</LinkComponent>
)
if (!allowNavPanelToExpand) {
return (
<Tooltip>
<TooltipTrigger asChild>{linkContent}</TooltipTrigger>
<TooltipContent side="right">
<span>{route.label}</span>
</TooltipContent>
</Tooltip>
)
}
return linkContent
}
)
NavigationIconLink.displayName = 'NavigationIconLink'
export default NavigationIconLink