@leafygreen-ui/tabs
v13.1.1
Published
leafyGreen UI Kit Tabs
Downloads
294,543
Keywords
Readme
Tabs
View on MongoDB.design
Installation
Yarn
yarn add @leafygreen-ui/tabs
NPM
npm install @leafygreen-ui/tabs
Example
import { Tabs, Tab } from '@leafygreen-ui/tabs';
const [selected, setSelected] = useState(0)
<Tabs setSelected={setSelected} selected={selected}>
<Tab name="Tab One">Tab Content One</Tab>
<Tab name="Tab Two">Tab Content Two</Tab>
<Tab name="Tab Three">Tab Content Three</Tab>
</Tabs>
Properties
| Prop | Type | Description | Default |
| -------------- | ------------------------ | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ----------- |
| selected
| 'number'
| 'string'
| Index or name of the Tab that should appear active. If using the name, pass the text content from the Tab
name
prop. If selected is undefined, the <Tabs />
component will behave as an uncontrolled component. If selected is passed a string or number that cannot be found, nothing will be selected. | |
| setSelected
| function
| A callback that receives the index or name of the tab a user is switching to when clicking, or via keyboard navigation. Usually this is used to set the selected prop to the correct index or name. The function is only invoked if the selected prop is set. | |
| as
| React.ElementType
| Sets the root element of all <Tab />
components in <Tabs />
. For example, setting as to Link
will render each tab as a <Link />
component rather than as a button. | button
|
| className
| string
| Adds a className to the root element. | |
| children
| node
| <Tab />
components that will be supplied to <Tabs />
component. | |
| darkMode
| boolean
| Determines whether or not the component will appear in DarkMode | false
|
| baseFontSize
| 13
| 16
| Determines font-size
for Tabs Component | 13
|
| size
| 'small'
| 'default'
| Determines size
for Tabs Component | 'default'
|
Any other properties supplied will be spread on the root element.
Tab
Properties
| Prop | Type | Description | Default |
| ----------------- | -------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------- | ------- |
| name
(Required) | string
, ReactNode
| String that will appear in the list of tabs. | |
| disabled
| boolean
| Indicates whether or not the <Tab />
can be clicked by a user. | false
|
| default
| boolean
| Should be supplied when using the uncontrolled <Tabs />
component. This determines which tab will be active by default. | |
| className
| string
| Adds a className to the root element. | |
| href
| string
| Destination when Tab's name
in the list should be rendered as an a
tag. | |
| to
| string
| Destination when Tab's name
in the list should be rendered as a Link
tag. | |
| children
| node
| Content that appears inside the <Tab />
component | |
| ... | native attributes of component passed to as
prop | Any other props will be spread on the root element | |
Test Harnesses
getTestUtils()
getTestUtils()
is a util that allows consumers to reliably interact with LG Tabs
in a product test suite. If the Tabs
component cannot be found, an error will be thrown.
Usage
import { getTestUtils } from '@leafygreen-ui/tabs';
const utils = getTestUtils(lgId?: string); // lgId refers to the custom `data-lgid` attribute passed to `Tabs`. It defaults to 'lg-tabs' if left empty.
Single Tabs
component
import { render } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
import { Tabs, Tab, getTestUtils } from '@leafygreen-ui/tabs';
...
test('tabs', () => {
render(
<Tabs aria-label="Label">
<Tab name="First" default>
Content 1
</Tab>
<Tab name="Second">
Content 2
</Tab>
<Tab name="Third">
Content 3
</Tab>
</Tabs>
);
const { getAllTabPanelsInDOM, getAllTabsInTabList, getSelectedPanel, getTabUtilsByName } = getTestUtils();
expect(getAllTabsInTabList()).toHaveLength(3);
expect(getAllTabPanelsInDOM()).toHaveLength(1);
const firstTabUtils = getTabUtilsByName('First');
expect(firstTabUtils.isSelected()).toBeTruthy();
expect(getSelectedPanel()).toHaveTextContent('Content 1');
const secondTabUtils = getTabUtilsByName('Second');
// click to second tab
if (secondTabUtils) {
userEvent.click(secondTabUtils.getTab());
}
// selected panel text content is updated
expect(getSelectedPanel()).toHaveTextContent('Content 2');
});
Multiple Tabs
components
When testing multiple Tabs
components, it is recommended to add the custom data-lgid
attribute to each Tabs
.
import { render } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
import { Tabs, Tab, getTestUtils } from '@leafygreen-ui/tabs';
...
test('tabs', () => {
render(
<>
<Tabs aria-label="Label ABC" data-lgid="tabs-abc">
<Tab name="A" default>
Content A
</Tab>
<Tab name="B">
Content B
</Tab>
<Tab name="C">
Content C
</Tab>
</Tabs>
<Tabs aria-label="Label XY" data-lgid="tabs-xy" forceRenderAllTabPanels={true}>
<Tab name="X">
Content X
</Tab>
<Tab name="Y" default>
Content Y
</Tab>
</Tabs>
</>
);
const testUtils1 = getTestUtils('tabs-abc'); // data-lgid
const testUtils2 = getTestUtils('tabs-xy'); // data-lgid
// First tabs
expect(testUtils1.getAllTabsInTabList()).toHaveLength(3);
expect(testUtils1.getAllTabPanelsInDOM()).toHaveLength(1);
expect(testUtils1.getSelectedPanel()).toHaveTextContent('Content A');
// Second tabs
expect(testUtils2.getAllTabsInTabList()).toHaveLength(2);
expect(testUtils2.getAllTabPanelsInDOM()).toHaveLength(2);
expect(testUtils2.getSelectedPanel()).toHaveTextContent('Content Y');
});
Test Utils
const {
getAllTabPanelsInDOM,
getAllTabsInTabList,
getTabUtilsByName: { getTab, isSelected, isDisabled },
getSelectedPanel,
} = getTestUtils();
| Util | Description | Returns |
| ------------------------ | ---------------------------------------------------- | ----------------------- |
| getAllTabPanelsInDOM()
| Returns an array of tab panels | Array<HTMLElement>
|
| getAllTabsInTabList()
| Returns an array of tabs | Array<HTMLElement>
|
| getSelectedPanel()
| Returns the selected tab panel | HTMLElement
| null
|
| getTabUtilsByName()
| Returns tab utils if tab with matching name is found | TabUtils
| null
|
| TabUtils | | |
| getTab()
| Returns the tab | HTMLElement
|
| isSelected()
| Returns whether the tab is selected | boolean
|
| isDisabled()
| Returns whether the tab is disabled | boolean
|
Reference
Usage with NextJS Link components
Tabs may not render with the correct tags or styles if the NextJS Link component is passed to the as
prop directly, given how NextJS handles default rendering of the component based on the href
prop. To work around this, pass the NextJS Link as shown below.
import NextLink from 'next/link';
function Linker({ href, children, ...props }) {
return (
<NextLink href={href}>
<a {...props}>
{children}
</a>
</NextLink>
);
}
<Tabs aria-label="Profile Sections" as={Linker}>
...
</Tabs>