Skip to content

Commit 45cccd4

Browse files
authoredNov 6, 2024··
allow passing className for <Tabs> and <Tabs.Tab> (#3661)
aa
1 parent 91e9399 commit 45cccd4

File tree

4 files changed

+42
-32
lines changed

4 files changed

+42
-32
lines changed
 

‎.changeset/cuddly-otters-allow.md

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
---
2+
'nextra': patch
3+
'nextra-theme-docs': patch
4+
---
5+
6+
allow passing `className` for `<Tabs>` and `<Tabs.Tab>`

‎README.md

-3
Original file line numberDiff line numberDiff line change
@@ -61,7 +61,4 @@ the watch mode for both nextra and the theme in separated terminals.
6161
<a href="https://speakeasyapi.dev/docs?utm_source=github&utm_campaign=nextra&utm_content=logolink">
6262
<img src="/docs/pages/showcase/speakeasy.png" alt="Speakeasy preview" width="256">
6363
</a>
64-
<a href="https://xyflow.com?utm_source=github&utm_campaign=nextra&utm_content=logolink">
65-
<img src="/docs/pages/showcase/xyflow.jpg" alt="xyflow preview" width="256">
66-
</a>
6764
</div>

‎docs/pages/sponsors.mdx

-6
Original file line numberDiff line numberDiff line change
@@ -29,12 +29,6 @@ import { Card } from './showcase.mdx'
2929
>
3030
<>![Speakeasy preview](./showcase/speakeasy.png)</>
3131
</Card>
32-
<Card
33-
title="xyflow"
34-
href="https://xyflow.com?utm_source=nextra.site&utm_campaign=nextra&utm_content=logolink"
35-
>
36-
<>![xyflow preview](./showcase/xyflow.jpg)</>
37-
</Card>
3832
</Cards>
3933

4034
<style global jsx>{`

‎packages/nextra/src/client/components/tabs/tabs.tsx

+36-23
Original file line numberDiff line numberDiff line change
@@ -7,9 +7,13 @@ import {
77
TabPanel,
88
TabPanels
99
} from '@headlessui/react'
10-
import type { TabPanelProps } from '@headlessui/react'
10+
import type {
11+
TabGroupProps,
12+
TabListProps,
13+
TabPanelProps
14+
} from '@headlessui/react'
1115
import cn from 'clsx'
12-
import type { ReactElement, ReactNode } from 'react'
16+
import type { FC, ReactElement, ReactNode } from 'react'
1317
import { useCallback, useEffect, useState } from 'react'
1418

1519
type TabItem = string | ReactElement
@@ -23,21 +27,22 @@ function isTabObjectItem(item: unknown): item is TabObjectItem {
2327
return !!item && typeof item === 'object' && 'label' in item
2428
}
2529

26-
export function Tabs({
30+
export const Tabs: FC<
31+
{
32+
items: (TabItem | TabObjectItem)[]
33+
children: ReactNode
34+
storageKey?: string
35+
} & Pick<TabGroupProps, 'defaultIndex' | 'selectedIndex' | 'onChange'> &
36+
Pick<TabListProps, 'className'>
37+
> = ({
2738
items,
28-
selectedIndex: _selectedIndex,
39+
children,
40+
storageKey,
2941
defaultIndex = 0,
42+
selectedIndex: _selectedIndex,
3043
onChange,
31-
children,
32-
storageKey
33-
}: {
34-
items: (TabItem | TabObjectItem)[]
35-
selectedIndex?: number
36-
defaultIndex?: number
37-
onChange?: (index: number) => void
38-
children: ReactNode
39-
storageKey?: string
40-
}): ReactElement {
44+
className
45+
}) => {
4146
const [selectedIndex, setSelectedIndex] = useState(defaultIndex)
4247

4348
useEffect(() => {
@@ -91,11 +96,14 @@ export function Tabs({
9196
tabIndex={-1} // disables focus in Firefox
9297
>
9398
<TabList
94-
className={cn(
95-
'nextra-scrollbar _overflow-x-auto _overscroll-x-contain _overflow-y-hidden',
96-
'_mt-4 _flex _w-full _gap-2 _border-b _border-gray-200 _pb-px dark:_border-neutral-800',
97-
'nextra-focus'
98-
)}
99+
className={args =>
100+
cn(
101+
'nextra-scrollbar _overflow-x-auto _overscroll-x-contain _overflow-y-hidden',
102+
'_mt-4 _flex _w-full _gap-2 _border-b _border-gray-200 _pb-px dark:_border-neutral-800',
103+
'nextra-focus',
104+
typeof className === 'function' ? className(args) : className
105+
)
106+
}
99107
>
100108
{items.map((item, index) => (
101109
<HeadlessTab
@@ -132,18 +140,23 @@ export function Tabs({
132140
)
133141
}
134142

135-
export function Tab({
143+
export const Tab: FC<TabPanelProps> = ({
136144
children,
137145
// For SEO display all the Panel in the DOM and set `display: none;` for those that are not selected
138146
unmount = false,
147+
className,
139148
...props
140-
}: TabPanelProps): ReactElement {
149+
}) => {
141150
return (
142151
<TabPanel
143152
{...props}
144153
unmount={unmount}
145-
className={({ focus }) =>
146-
cn('_rounded _mt-6', focus && 'nextra-focusable')
154+
className={args =>
155+
cn(
156+
'_rounded _mt-6',
157+
args.focus && 'nextra-focusable',
158+
typeof className === 'function' ? className(args) : className
159+
)
147160
}
148161
>
149162
{children}

0 commit comments

Comments
 (0)
Please sign in to comment.