apps/mantine.dev/src/pages/core/tabs.mdx
import { FloatingIndicatorDemos, TabsDemos } from '@docs/demos'; import { Layout } from '@/layout'; import { MDX_DATA } from '@/mdx';
export default Layout(MDX_DATA.Tabs);
To control the Tabs state, use value and onChange props:
import { useState } from 'react';
import { Tabs } from '@mantine/core';
function Demo() {
const [activeTab, setActiveTab] = useState<string | null>('first');
return (
<Tabs value={activeTab} onChange={setActiveTab}>
<Tabs.List>
<Tabs.Tab value="first">First tab</Tabs.Tab>
<Tabs.Tab value="second">Second tab</Tabs.Tab>
</Tabs.List>
<Tabs.Panel value="first">First panel</Tabs.Panel>
<Tabs.Panel value="second">Second panel</Tabs.Panel>
</Tabs>
);
}
If you do not need to subscribe to Tabs state changes, use defaultValue:
import { Tabs } from '@mantine/core';
function Demo() {
return (
<Tabs defaultValue="first">
<Tabs.List>
<Tabs.Tab value="first">First tab</Tabs.Tab>
<Tabs.Tab value="second">Second tab</Tabs.Tab>
</Tabs.List>
<Tabs.Panel value="first">First panel</Tabs.Panel>
<Tabs.Panel value="second">Second panel</Tabs.Panel>
</Tabs>
);
}
To change the colors of all tabs, set color on the Tabs component; to change the color of an individual tab,
set color on Tabs.Tab.
To display a tab on the opposite side, set margin-left: auto with the ml="auto" prop or with className:
To make tabs inverted, place Tabs.Panel components before Tabs.List and add the inverted prop to the Tabs component:
To change the placement of Tabs.List in vertical orientation, set the placement prop:
Example of custom variant with FloatingIndicator:
<Demo data={FloatingIndicatorDemos.tabs} />Set the disabled prop on the Tabs.Tab component to disable a tab.
Disabled tabs cannot be activated with the mouse or keyboard, and they will be skipped when the user navigates with arrow keys:
By default, tabs are activated when the user presses arrow keys or Home/End keys.
To disable that, set activateTabWithKeyboard={false} on the Tabs component:
By default, the active tab cannot be deactivated. To allow that, set allowTabDeactivation on the Tabs component:
By default, inactive Tabs.Panel will stay mounted; to unmount inactive tabs, set keepMounted={false} on Tabs.
This is useful when you want to render components that impact performance inside Tabs.Panel. Note that
components that are rendered inside Tabs.Panel will reset their state on each mount (tab change).
import { Tabs } from '@mantine/core';
// Second tab panel will be mounted only when the user activates the second tab
function Demo() {
return (
<Tabs keepMounted={false} defaultValue="first">
<Tabs.List>
<Tabs.Tab value="first">First tab</Tabs.Tab>
<Tabs.Tab value="second">Second tab</Tabs.Tab>
</Tabs.List>
<Tabs.Panel value="first">First panel</Tabs.Panel>
<Tabs.Panel value="second">Second panel</Tabs.Panel>
</Tabs>
);
}
import { useRef } from 'react';
import { Tabs } from '@mantine/core';
function Demo() {
const secondTabRef = useRef<HTMLButtonElement>(null);
return (
<Tabs defaultValue="first">
<Tabs.List>
<Tabs.Tab value="first">First tab</Tabs.Tab>
<Tabs.Tab value="Second" ref={secondTabRef}>
Second tab
</Tabs.Tab>
<Tabs.Tab value="third">Third tab</Tabs.Tab>
</Tabs.List>
</Tabs>
);
}
<Route path="/tabs/:tabValue" element={<Demo />} />
import { useNavigate, useParams } from 'react-router-dom';
import { Tabs } from '@mantine/core';
function Demo() {
const navigate = useNavigate();
const { tabValue } = useParams();
return (
<Tabs
value={tabValue}
onChange={(value) => navigate(`/tabs/${value}`)}
>
<Tabs.List>
<Tabs.Tab value="first">First tab</Tabs.Tab>
<Tabs.Tab value="second">Second tab</Tabs.Tab>
</Tabs.List>
</Tabs>
);
}
// For file /tabs/[activeTab].tsx
import { useRouter } from 'next/router';
import { Tabs } from '@mantine/core';
function Demo() {
const router = useRouter();
return (
<Tabs
value={router.query.activeTab as string}
onChange={(value) => router.push(`/tabs/${value}`)}
>
<Tabs.List>
<Tabs.Tab value="first">First tab</Tabs.Tab>
<Tabs.Tab value="second">Second tab</Tabs.Tab>
</Tabs.List>
</Tabs>
);
}
Example of Styles API usage to customize tab styles:
<Demo data={TabsDemos.customize} />Use Scroller component to make the tabs list scrollable when there are too many tabs to fit in the available space:
<Demo data={TabsDemos.scroller} />The Tabs component follows WAI-ARIA recommendations on accessibility.
If you use Tabs.Tab without text content, for example, only with an icon, then set aria-label
or use the VisuallyHidden component:
import { CoinIcon } from '@phosphor-icons/react';
import { Tabs, VisuallyHidden } from '@mantine/core';
function Demo() {
return (
<Tabs defaultValue="chat">
<Tabs.List>
<Tabs.Tab value="chat">Chat</Tabs.Tab>
<Tabs.Tab
value="money"
aria-label="Get money"
leftSection={<CoinIcon size={14} />}
/>
<Tabs.Tab value="money" leftSection={<CoinIcon size={14} />}>
<VisuallyHidden>Get money</VisuallyHidden>
</Tabs.Tab>
</Tabs.List>
</Tabs>
);
}
To set the tabs list label, set aria-label on the Tabs.List component; it will be announced by screen reader:
import { Tabs } from '@mantine/core';
function Demo() {
return (
<Tabs defaultValue="recent">
<Tabs.List aria-label="Chats">
<Tabs.Tab value="recent">Most recent</Tabs.Tab>
<Tabs.Tab value="recent">Unanswered</Tabs.Tab>
<Tabs.Tab value="archived">Archived</Tabs.Tab>
</Tabs.List>
</Tabs>
);
}
<KeyboardEventsTable data={[ { key: 'ArrowRight', description: 'Focuses and activates next tab that is not disabled', condition: 'orientation="horizontal"', }, { key: 'ArrowLeft', description: 'Focuses and activates previous tab that is not disabled', condition: 'orientation="horizontal"', }, { key: 'ArrowDown', description: 'Focuses and activates next tab that is not disabled', condition: 'orientation="vertical"', }, { key: 'ArrowUp', description: 'Focuses and activates previous tab that is not disabled', condition: 'orientation="vertical"', }, { key: 'Home', description: 'Focuses and activates first tab' }, { key: 'End', description: 'Focuses and activates last tab' }, ]} />