Combobox with Tabs
Organizing Combobox with Tab components that support mouse, keyboard, and screen reader interactions. The UI remains responsive by using React.startTransition
.
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
import groupBy from "lodash-es/groupBy.js";import { matchSorter } from "match-sorter";import { useId, useMemo, useState } from "react";import {ComboboxPanel,ComboboxTab,ComboboxTabList,} from "./combobox.tsx";import { flatPages, pages } from "./pages.ts";import "./style.css";const categories = ["All", ...Object.keys(pages)];function getTabId(category: string, prefix: string) {return `${prefix}/${category}`;}function getCategory(tabId: string, prefix: string) {return tabId.replace(`${prefix}/`, "");}export default function Example() {const prefix = useId();const [searchValue, setSearchValue] = useState("");const [tabId, setTabId] = useState(getTabId("Components", prefix));const matches = useMemo(() => {const keys = ["label", "path"];const allMatches = matchSorter(flatPages, searchValue, { keys });const groups = groupBy(allMatches, "category");groups.All = allMatches;return groups;}, [searchValue]);const currentPages = matches[getCategory(tabId, prefix)] || [];return (defaultTabId={tabId}onTabChange={setTabId}onSearch={setSearchValue}><ComboboxTabList aria-label="Categories">{categories.map((category) => {const currentPages = matches[category];return (<ComboboxTabkey={category}id={getTabId(category, prefix)}>{category}<span className="count">{currentPages?.length || 0}</span></ComboboxTab>);})}</ComboboxTabList><ComboboxPanel>{!currentPages.length && (<div className="no-results">No pages found for "<strong>{searchValue}</strong>"</div>)}{currentPages.map((page, i) => (key={page.path + i}>{page.label}))}</ComboboxPanel>);}import groupBy from "lodash-es/groupBy.js";import { matchSorter } from "match-sorter";import { useId, useMemo, useState } from "react";import {ComboboxPanel,ComboboxTab,ComboboxTabList,} from "./combobox.tsx";import { flatPages, pages } from "./pages.ts";import "./style.css";const categories = ["All", ...Object.keys(pages)];function getTabId(category: string, prefix: string) {return `${prefix}/${category}`;}function getCategory(tabId: string, prefix: string) {return tabId.replace(`${prefix}/`, "");}export default function Example() {const prefix = useId();const [searchValue, setSearchValue] = useState("");const [tabId, setTabId] = useState(getTabId("Components", prefix));const matches = useMemo(() => {const keys = ["label", "path"];const allMatches = matchSorter(flatPages, searchValue, { keys });const groups = groupBy(allMatches, "category");groups.All = allMatches;return groups;}, [searchValue]);const currentPages = matches[getCategory(tabId, prefix)] || [];return (defaultTabId={tabId}onTabChange={setTabId}onSearch={setSearchValue}><ComboboxTabList aria-label="Categories">{categories.map((category) => {const currentPages = matches[category];return (<ComboboxTabkey={category}id={getTabId(category, prefix)}>{category}<span className="count">{currentPages?.length || 0}</span></ComboboxTab>);})}</ComboboxTabList><ComboboxPanel>{!currentPages.length && (<div className="no-results">No pages found for "<strong>{searchValue}</strong>"</div>)}{currentPages.map((page, i) => (key={page.path + i}>{page.label}))}</ComboboxPanel>);}
Related examples
data:image/s3,"s3://crabby-images/05d15/05d1508a9882aa6428561b450051ff355b2f55d5" alt=""
data:image/s3,"s3://crabby-images/b477b/b477bc8b414c62b0ca0a0dbb9b51d99718c596ac" alt=""
Command Menu with TabsCombining Dialog, Tab, and Combobox from Ariakit React to build a command palette component.
data:image/s3,"s3://crabby-images/57fe8/57fe89aeaf3677003ed8b03b47e8b608b64d24fe" alt=""
data:image/s3,"s3://crabby-images/f9d1e/f9d1e6cb1c1b81aeb75171d14e226b4006d0ad63" alt=""
Combobox filteringListing suggestions in a Combobox component based on the input value using React.startTransition to ensure the UI remains responsive during typing.
data:image/s3,"s3://crabby-images/eb155/eb155e9265f8735f6352c800f3fd1c62ccf23155" alt=""
data:image/s3,"s3://crabby-images/c8fe1/c8fe1d39153e2ba3e4248fd4b6bc3cd53d044119" alt=""
Combobox with integrated filterFiltering options in a Combobox component through an abstracted implementation using React.useDeferredValue, resulting in a simple higher-level API.
data:image/s3,"s3://crabby-images/e1099/e1099a278fbcb050fa96f9ddd15fdf57c29dc3d1" alt=""
data:image/s3,"s3://crabby-images/1b85e/1b85e90a9477c1750dce7b09405eaff57f85830b" alt=""
ComboboxGroupOrganizing Combobox items into labelled groups using the ComboboxGroup and ComboboxGroupLabel components in React.
data:image/s3,"s3://crabby-images/1dd70/1dd7014c5581cadf225fc40f7f2631fe265a5c1b" alt=""
data:image/s3,"s3://crabby-images/3aef7/3aef7f7abf8b5cf595b69f9152a4b74fdcc2169d" alt=""
Combobox with linksUsing a Combobox with items rendered as links that can be clicked with keyboard and mouse. This is useful for creating an accessible page search input in React.
data:image/s3,"s3://crabby-images/f87c4/f87c41c2fc0cc349b3dc8c308f278bbd050ca2e1" alt=""
data:image/s3,"s3://crabby-images/b2377/b237712e66344db5cbca511e42dc8849ef51cceb" alt=""
ComboboxCancelReseting the value of a Combobox with a button rendered next to it using the ComboboxCancel component.
data:image/s3,"s3://crabby-images/d750b/d750b1707307c59d49a2bf2047dc86b9b6caa0d1" alt=""
data:image/s3,"s3://crabby-images/c6fa0/c6fa02cca401a101607231fe7fb8c13246008858" alt=""
ComboboxDisclosureOpening and closing a Combobox with the help of a button rendered next to it using the ComboboxDisclosure component.
Select with Combobox and TabsAbstracting Select to work alongside Combobox and Tab components, presenting a searchable, tabbed dropdown.