Command Menu with Tabs
Combining Dialog, Tab, and Combobox from Ariakit React to build a command palette component.
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
import groupBy from "lodash-es/groupBy.js";import { matchSorter } from "match-sorter";import { useId, useMemo, useState } from "react";import {CommandMenu,CommandMenuFooter,CommandMenuGrid,CommandMenuGroup,CommandMenuInput,CommandMenuItem,CommandMenuList,CommandMenuTab,CommandMenuTabList,CommandMenuTabPanel,} from "./command-menu.tsx";import { flatPages, pages } from "./pages.ts";import "./theme.css";export default function Example() {return (<div className="flex gap-2 justify-center flex-wrap"><Simple /><WithTabs cols={1} /><WithTabs cols={2} /><WithTabs cols={3} /></div>);}const categories = ["All", ...Object.keys(pages)];function getCategoryId(category: string, prefix: string) {return `${prefix}/${category}`;}function getCategoryLabel(tabId: string, prefix: string) {return tabId.replace(`${prefix}/`, "");}function Simple() {const [open, setOpen] = useState(false);const [searchValue, setSearchValue] = useState("");const matches = useMemo(() => {if (!searchValue) return flatPages;return matchSorter(flatPages, searchValue, { keys: ["label", "path"] });}, [searchValue]);const currentPages = matches;return (<>className="ak-button ak-button-default"onClick={() => setOpen(true)}>Simple<CommandMenuaria-label="Command Menu"open={open}setOpen={setOpen}onSearch={setSearchValue}><CommandMenuInput placeholder="Search pages..." /><CommandMenuList>{currentPages.map((page) => (<CommandMenuItem key={page.label} render={<a href={page.path} />}><span className="truncate">{page.label}</span></CommandMenuItem>))}</CommandMenuList><CommandMenuFooter /></CommandMenu></>);}function WithTabs({ cols = 1 }: { cols?: number }) {const prefix = useId();const allLabel = "All";const allId = getCategoryId(allLabel, prefix);const [open, setOpen] = useState(false);const [searchValue, setSearchValue] = useState("");const [tabId, setTabId] = useState(allId);const isAllTab = tabId === allId;const [allMatches, groupMatches] = useMemo(() => {const allMatches = searchValue? matchSorter(flatPages, searchValue, { keys: ["label", "path"] }): flatPages;const groupMatches = groupBy(allMatches, "category");return [allMatches, groupMatches];}, [searchValue]);const matchedPages = isAllTab? groupMatches: groupMatches[getCategoryLabel(tabId, prefix)] || [];const renderGrid = (pages: typeof matchedPages) => {if (!Array.isArray(pages)) {return Object.entries(pages).map(([category, pages], index) => (<CommandMenuGroup key={index} label={category}>{renderGrid(pages)}</CommandMenuGroup>));}return (<CommandMenuGrid cols={cols}>{pages.map((page, index) => (<CommandMenuItemkey={page.label}index={index}render={<a href={page.path} />}><span className="truncate">{page.label}</span></CommandMenuItem>))}</CommandMenuGrid>);};return (<>className="ak-button ak-button-default"onClick={() => setOpen(true)}>With Tabs{cols > 1 ? ` (${cols} columns)` : ""}<CommandMenuaria-label="Command Menu"open={open}setOpen={setOpen}onSearch={setSearchValue}defaultTab={tabId}onTabChange={setTabId}><CommandMenuInput placeholder="Search pages..." /><CommandMenuTabList aria-label="Categories">{categories.map((label) => {const pages = label === allLabel ? allMatches : groupMatches[label];return (<CommandMenuTabkey={label}id={getCategoryId(label, prefix)}disabled={!pages?.length}>{label} ({pages?.length || 0})</CommandMenuTab>);})}</CommandMenuTabList><CommandMenuTabPanel><CommandMenuList className="flex flex-col gap-4">{renderGrid(matchedPages)}</CommandMenuList></CommandMenuTabPanel><CommandMenuFooter /></CommandMenu></>);}import groupBy from "lodash-es/groupBy.js";import { matchSorter } from "match-sorter";import { useId, useMemo, useState } from "react";import {CommandMenu,CommandMenuFooter,CommandMenuGrid,CommandMenuGroup,CommandMenuInput,CommandMenuItem,CommandMenuList,CommandMenuTab,CommandMenuTabList,CommandMenuTabPanel,} from "./command-menu.tsx";import { flatPages, pages } from "./pages.ts";import "./theme.css";export default function Example() {return (<div className="flex gap-2 justify-center flex-wrap"><Simple /><WithTabs cols={1} /><WithTabs cols={2} /><WithTabs cols={3} /></div>);}const categories = ["All", ...Object.keys(pages)];function getCategoryId(category: string, prefix: string) {return `${prefix}/${category}`;}function getCategoryLabel(tabId: string, prefix: string) {return tabId.replace(`${prefix}/`, "");}function Simple() {const [open, setOpen] = useState(false);const [searchValue, setSearchValue] = useState("");const matches = useMemo(() => {if (!searchValue) return flatPages;return matchSorter(flatPages, searchValue, { keys: ["label", "path"] });}, [searchValue]);const currentPages = matches;return (<>className="ak-button ak-button-default"onClick={() => setOpen(true)}>Simple<CommandMenuaria-label="Command Menu"open={open}setOpen={setOpen}onSearch={setSearchValue}><CommandMenuInput placeholder="Search pages..." /><CommandMenuList>{currentPages.map((page) => (<CommandMenuItem key={page.label} render={<a href={page.path} />}><span className="truncate">{page.label}</span></CommandMenuItem>))}</CommandMenuList><CommandMenuFooter /></CommandMenu></>);}function WithTabs({ cols = 1 }: { cols?: number }) {const prefix = useId();const allLabel = "All";const allId = getCategoryId(allLabel, prefix);const [open, setOpen] = useState(false);const [searchValue, setSearchValue] = useState("");const [tabId, setTabId] = useState(allId);const isAllTab = tabId === allId;const [allMatches, groupMatches] = useMemo(() => {const allMatches = searchValue? matchSorter(flatPages, searchValue, { keys: ["label", "path"] }): flatPages;const groupMatches = groupBy(allMatches, "category");return [allMatches, groupMatches];}, [searchValue]);const matchedPages = isAllTab? groupMatches: groupMatches[getCategoryLabel(tabId, prefix)] || [];const renderGrid = (pages: typeof matchedPages) => {if (!Array.isArray(pages)) {return Object.entries(pages).map(([category, pages], index) => (<CommandMenuGroup key={index} label={category}>{renderGrid(pages)}</CommandMenuGroup>));}return (<CommandMenuGrid cols={cols}>{pages.map((page, index) => (<CommandMenuItemkey={page.label}index={index}render={<a href={page.path} />}><span className="truncate">{page.label}</span></CommandMenuItem>))}</CommandMenuGrid>);};return (<>className="ak-button ak-button-default"onClick={() => setOpen(true)}>With Tabs{cols > 1 ? ` (${cols} columns)` : ""}<CommandMenuaria-label="Command Menu"open={open}setOpen={setOpen}onSearch={setSearchValue}defaultTab={tabId}onTabChange={setTabId}><CommandMenuInput placeholder="Search pages..." /><CommandMenuTabList aria-label="Categories">{categories.map((label) => {const pages = label === allLabel ? allMatches : groupMatches[label];return (<CommandMenuTabkey={label}id={getCategoryId(label, prefix)}disabled={!pages?.length}>{label} ({pages?.length || 0})</CommandMenuTab>);})}</CommandMenuTabList><CommandMenuTabPanel><CommandMenuList className="flex flex-col gap-4">{renderGrid(matchedPages)}</CommandMenuList></CommandMenuTabPanel><CommandMenuFooter /></CommandMenu></>);}
Related examples
data:image/s3,"s3://crabby-images/0b20d/0b20dbb6e86c720869b5f345e3d0e3f8bec69edb" alt=""
data:image/s3,"s3://crabby-images/f3295/f3295e458bcd1b2a67e4cc41a1c605612a769542" alt=""
Command MenuCombining Dialog and Combobox to enable users to search a command list in a Raycast-style modal.
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/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/e1854/e1854b7f044cd165b4d51101016e3865d4bad070" alt=""
data:image/s3,"s3://crabby-images/3b8fc/3b8fcd3f5f9d00cbd7756d8cfb748e1c5b46a533" alt=""
Combobox with TabsOrganizing Combobox with Tab components that support mouse, keyboard, and screen reader interactions. The UI remains responsive by using React.startTransition.
data:image/s3,"s3://crabby-images/26546/26546092f2517026ec69c4c85867d6289aa085b6" alt=""
data:image/s3,"s3://crabby-images/1ad7d/1ad7d9fb9294772f9f9a36e75751bbaec6ece96f" alt=""
Dialog with MenuShowing a nested dropdown Menu component inside a modal Dialog using React.
data:image/s3,"s3://crabby-images/b5f2a/b5f2ab1185d08f79ffe4ee5c0b832f71959207cf" alt=""
data:image/s3,"s3://crabby-images/2f5c5/2f5c57c7d7f586e2aeef915a28e48044bea289b2" alt=""
Nested DialogRendering a modal Dialog to confirm an action inside another modal dialog using React.
Menu with ComboboxCombining Menu and Combobox to create a dropdown menu with a search field that can be used to filter menu items.
Select with ComboboxCombining Select and Combobox to create a dropdown with a search field that can be used to filter items.