1
1
import { createContext , use , useCallback , useId , useMemo , useState } from "react"
2
2
3
+ import { useMediaQuery } from "@/utils/use-media-query"
3
4
import { IconHamburger } from "justd-icons"
4
5
import { LayoutGroup , motion } from "motion/react"
5
6
import type { LinkProps } from "react-aria-components"
6
- import { Link , composeRenderProps } from "react-aria-components"
7
- import { type VariantProps , tv } from "tailwind-variants"
8
-
9
- import { cn } from "@/utils/classes"
10
- import { useMediaQuery } from "@/utils/use-media-query"
7
+ import { Link } from "react-aria-components"
8
+ import { twJoin , twMerge } from "tailwind-merge"
9
+ import { tv } from "tailwind-variants"
11
10
import { Button , type ButtonProps } from "./button"
12
11
import { composeTailwindRenderProps } from "./primitive"
13
12
import { Sheet } from "./sheet"
@@ -42,17 +41,6 @@ interface NavbarProps extends React.ComponentProps<"header">, NavbarOptions {
42
41
onOpenChange ?: ( open : boolean ) => void
43
42
}
44
43
45
- const navbarStyles = tv ( {
46
- base : "relative isolate flex w-full flex-col" ,
47
- variants : {
48
- intent : {
49
- floating : "px-2.5 pt-2" ,
50
- navbar : "" ,
51
- inset : "min-h-svh bg-navbar dark:bg-bg" ,
52
- } ,
53
- } ,
54
- } )
55
-
56
44
const Navbar = ( {
57
45
children,
58
46
isOpen : openProp ,
@@ -99,7 +87,13 @@ const Navbar = ({
99
87
< NavbarContext value = { contextValue } >
100
88
< header
101
89
data-navbar-intent = { intent }
102
- className = { navbarStyles ( { intent, className } ) }
90
+ className = { twMerge (
91
+ "relative isolate flex w-full flex-col" ,
92
+ intent === "navbar" && "" ,
93
+ intent === "floating" && "px-2.5 pt-2" ,
94
+ intent === "inset" && "min-h-svh bg-navbar dark:bg-bg" ,
95
+ className ,
96
+ ) }
103
97
{ ...props }
104
98
>
105
99
{ children }
@@ -201,7 +195,7 @@ const NavbarSection = ({ className, ...props }: React.ComponentProps<"div">) =>
201
195
< LayoutGroup id = { id } >
202
196
< div
203
197
data-navbar-section = "true"
204
- className = { cn (
198
+ className = { twMerge (
205
199
"flex" ,
206
200
isCompact ? "flex-col gap-y-4" : "flex-row items-center gap-x-3" ,
207
201
className ,
@@ -214,21 +208,6 @@ const NavbarSection = ({ className, ...props }: React.ComponentProps<"div">) =>
214
208
)
215
209
}
216
210
217
- const navItemStyles = tv ( {
218
- base : [
219
- "*:data-[slot=icon]:-mx-0.5 relative flex cursor-pointer items-center gap-x-2 px-2 text-muted-fg no-underline outline-hidden transition-colors md:text-sm forced-colors:transform-none forced-colors:outline-0 forced-colors:disabled:text-[GrayText]" ,
220
- "pressed:text-fg hover:text-fg data-focused:text-fg data-focus-visible:outline-1 data-focus-visible:outline-primary" ,
221
- "**:data-[slot=chevron]:size-4 **:data-[slot=chevron]:transition-transform" ,
222
- "*:data-[slot=icon]:size-4 *:data-[slot=icon]:shrink-0 pressed:**:data-[slot=chevron]:rotate-180" ,
223
- "disabled:cursor-default disabled:opacity-50 disabled:forced-colors:text-[GrayText]" ,
224
- ] ,
225
- variants : {
226
- isCurrent : {
227
- true : "cursor-default text-navbar-fg" ,
228
- } ,
229
- } ,
230
- } )
231
-
232
211
interface NavbarItemProps extends LinkProps {
233
212
isCurrent ?: boolean
234
213
}
@@ -239,8 +218,16 @@ const NavbarItem = ({ className, isCurrent, ...props }: NavbarItemProps) => {
239
218
< Link
240
219
data-navbar-item = "true"
241
220
aria-current = { isCurrent ? "page" : undefined }
242
- className = { composeRenderProps ( className , ( className , ...renderProps ) =>
243
- navItemStyles ( { ...renderProps , isCurrent, className } ) ,
221
+ className = { composeTailwindRenderProps (
222
+ className ,
223
+ twJoin (
224
+ "*:data-[slot=icon]:-mx-0.5 relative flex cursor-pointer items-center gap-x-2 px-2 text-muted-fg no-underline outline-hidden transition-colors md:text-sm forced-colors:transform-none forced-colors:outline-0 forced-colors:disabled:text-[GrayText]" ,
225
+ "pressed:text-fg hover:text-fg focus:text-fg data-focus-visible:outline-1 data-focus-visible:outline-primary" ,
226
+ "**:data-[slot=chevron]:size-4 **:data-[slot=chevron]:transition-transform" ,
227
+ "*:data-[slot=icon]:size-4 *:data-[slot=icon]:shrink-0 pressed:**:data-[slot=chevron]:rotate-180" ,
228
+ "disabled:cursor-default disabled:opacity-50 disabled:forced-colors:text-[GrayText]" ,
229
+ isCurrent && "cursor-default text-navbar-fg" ,
230
+ ) ,
244
231
) }
245
232
{ ...props }
246
233
>
@@ -266,36 +253,37 @@ const NavbarLogo = ({ className, ...props }: LinkProps) => {
266
253
< Link
267
254
className = { composeTailwindRenderProps (
268
255
className ,
269
- "relative flex items-center gap-x-2 px-2 py-4 text-fg data- focus-visible :outline-1 data-focus-visible:outline-primary data-focused :outline-hidden md:mr-4 md:px-0 md:py-0" ,
256
+ "relative flex items-center gap-x-2 px-2 py-4 text-fg focus:outline-hidden data-focus-visible:outline-1 data-focus-visible :outline-primary md:mr-4 md:px-0 md:py-0" ,
270
257
) }
271
258
{ ...props }
272
259
/>
273
260
)
274
261
}
275
262
276
263
const NavbarFlex = ( { className, ref, ...props } : React . ComponentProps < "div" > ) => {
277
- return < div ref = { ref } className = { cn ( "flex items-center gap-2 md:gap-3" , className ) } { ...props } />
264
+ return (
265
+ < div ref = { ref } className = { twMerge ( "flex items-center gap-2 md:gap-3" , className ) } { ...props } />
266
+ )
278
267
}
279
268
280
- const compactStyles = tv ( {
281
- base : "flex justify-between bg-navbar text-navbar-fg peer-has-[[data-navbar-intent=floating]]:border md:hidden" ,
282
- variants : {
283
- intent : {
284
- floating : "h-12 rounded-lg border px-3.5" ,
285
- inset : "h-14 border-b px-4" ,
286
- navbar : "h-14 border-b px-4" ,
287
- } ,
288
- } ,
289
- } )
290
-
291
- interface NavbarCompactProps
292
- extends React . ComponentProps < "div" > ,
293
- VariantProps < typeof compactStyles > {
269
+ interface NavbarCompactProps extends React . ComponentProps < "div" > , Pick < NavbarOptions , "intent" > {
294
270
ref ?: React . RefObject < HTMLDivElement >
295
271
}
296
272
const NavbarCompact = ( { className, ref, ...props } : NavbarCompactProps ) => {
297
273
const { intent } = useNavbar ( )
298
- return < div ref = { ref } className = { compactStyles ( { intent, className } ) } { ...props } />
274
+ return (
275
+ < div
276
+ ref = { ref }
277
+ className = { twMerge (
278
+ "flex justify-between bg-navbar text-navbar-fg peer-has-[[data-navbar-intent=floating]]:border md:hidden" ,
279
+ intent === "floating" && "h-12 rounded-lg border px-3.5" ,
280
+ intent === "inset" && "h-14 border-b px-4" ,
281
+ intent === "navbar" && "h-14 border-b px-4" ,
282
+ className ,
283
+ ) }
284
+ { ...props }
285
+ />
286
+ )
299
287
}
300
288
301
289
const insetStyles = tv ( {
@@ -316,7 +304,7 @@ const NavbarInset = ({ className, ref, ...props }: React.ComponentProps<"div">)
316
304
< main
317
305
ref = { ref }
318
306
data-navbar-intent = { intent }
319
- className = { cn (
307
+ className = { twMerge (
320
308
"flex flex-1 flex-col" ,
321
309
intent === "inset" && "bg-navbar pb-2 md:px-2 dark:bg-bg" ,
322
310
className ,
0 commit comments