Textarea with inline Combobox
Rendering Combobox as a textarea element to create an accessible multiline textbox in React. Inserting specific characters triggers a popup with dynamic suggestions.
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
import * as Ariakit from "@ariakit/react";import { matchSorter } from "match-sorter";import * as React from "react";import { getList, getValue } from "./list.ts";import "./style.css";import {getAnchorRect,getSearchValue,getTrigger,getTriggerOffset,replaceValue,} from "./utils.ts";export default function Example() {const ref = React.useRef<HTMLTextAreaElement>(null);const [value, setValue] = React.useState("");const [trigger, setTrigger] = React.useState<string | null>(null);const [caretOffset, setCaretOffset] = React.useState<number | null>(null);const deferredSearchValue = React.useDeferredValue(searchValue);const matches = React.useMemo(() => {return matchSorter(getList(trigger), deferredSearchValue, {baseSort: (a, b) => (a.index < b.index ? -1 : 1),}).slice(0, 10);}, [trigger, deferredSearchValue]);const hasMatches = !!matches.length;React.useLayoutEffect(() => {combobox.setOpen(hasMatches);}, [combobox, hasMatches]);React.useLayoutEffect(() => {if (caretOffset != null) {ref.current?.setSelectionRange(caretOffset, caretOffset);}}, [caretOffset]);// Re-calculates the position of the combobox popover in case the changes on// the textarea value have shifted the trigger character.React.useEffect(combobox.render, [combobox, value]);const onKeyDown = (event: React.KeyboardEvent<HTMLTextAreaElement>) => {if (event.key === "ArrowLeft" || event.key === "ArrowRight") {combobox.hide();}};const onChange = (event: React.ChangeEvent<HTMLTextAreaElement>) => {const trigger = getTrigger(event.target);const searchValue = getSearchValue(event.target);// If there's a trigger character, we'll show the combobox popover. This can// be true both when the trigger character has just been typed and when// content has been deleted (e.g., with backspace) and the character right// before the caret is the trigger.if (trigger) {setTrigger(trigger);combobox.show();}// There will be no trigger and no search value if the trigger character has// just been deleted.else if (!searchValue) {setTrigger(null);combobox.hide();}// Sets our textarea value.setValue(event.target.value);// Sets the combobox value that will be used to search in the list.combobox.setValue(searchValue);};const onItemClick = (value: string) => () => {const textarea = ref.current;if (!textarea) return;const offset = getTriggerOffset(textarea);const displayValue = getValue(value, trigger);if (!displayValue) return;setTrigger(null);setValue(replaceValue(offset, searchValue, displayValue));const nextCaretOffset = offset + displayValue.length + 1;setCaretOffset(nextCaretOffset);};return (<div className="wrapper"><label className="label">Commentvalue={value}// We'll overwrite how the combobox popover is shown, so we disable// the default behaviors.// To the combobox state, we'll only set the value after the trigger// character (the search value), so we disable the default behavior.className="combobox"<textarearef={ref}rows={5}placeholder="Type @, # or :"// We need to re-calculate the position of the combobox popover// when the textarea contents are scrolled.onScroll={combobox.render}// Hide the combobox popover whenever the selection changes.onPointerDown={combobox.hide}onChange={onChange}onKeyDown={onKeyDown}/>}/></label>hidden={!hasMatches}const textarea = ref.current;if (!textarea) return null;return getAnchorRect(textarea);}}className="popover">{matches.map((value) => (key={value}onClick={onItemClick(value)}className="combobox-item"><span>{value}</span>))}</div>);}import * as Ariakit from "@ariakit/react";import { matchSorter } from "match-sorter";import * as React from "react";import { getList, getValue } from "./list.ts";import "./style.css";import {getAnchorRect,getSearchValue,getTrigger,getTriggerOffset,replaceValue,} from "./utils.ts";export default function Example() {const ref = React.useRef<HTMLTextAreaElement>(null);const [value, setValue] = React.useState("");const [trigger, setTrigger] = React.useState<string | null>(null);const [caretOffset, setCaretOffset] = React.useState<number | null>(null);const deferredSearchValue = React.useDeferredValue(searchValue);const matches = React.useMemo(() => {return matchSorter(getList(trigger), deferredSearchValue, {baseSort: (a, b) => (a.index < b.index ? -1 : 1),}).slice(0, 10);}, [trigger, deferredSearchValue]);const hasMatches = !!matches.length;React.useLayoutEffect(() => {combobox.setOpen(hasMatches);}, [combobox, hasMatches]);React.useLayoutEffect(() => {if (caretOffset != null) {ref.current?.setSelectionRange(caretOffset, caretOffset);}}, [caretOffset]);// Re-calculates the position of the combobox popover in case the changes on// the textarea value have shifted the trigger character.React.useEffect(combobox.render, [combobox, value]);const onKeyDown = (event: React.KeyboardEvent<HTMLTextAreaElement>) => {if (event.key === "ArrowLeft" || event.key === "ArrowRight") {combobox.hide();}};const onChange = (event: React.ChangeEvent<HTMLTextAreaElement>) => {const trigger = getTrigger(event.target);const searchValue = getSearchValue(event.target);// If there's a trigger character, we'll show the combobox popover. This can// be true both when the trigger character has just been typed and when// content has been deleted (e.g., with backspace) and the character right// before the caret is the trigger.if (trigger) {setTrigger(trigger);combobox.show();}// There will be no trigger and no search value if the trigger character has// just been deleted.else if (!searchValue) {setTrigger(null);combobox.hide();}// Sets our textarea value.setValue(event.target.value);// Sets the combobox value that will be used to search in the list.combobox.setValue(searchValue);};const onItemClick = (value: string) => () => {const textarea = ref.current;if (!textarea) return;const offset = getTriggerOffset(textarea);const displayValue = getValue(value, trigger);if (!displayValue) return;setTrigger(null);setValue(replaceValue(offset, searchValue, displayValue));const nextCaretOffset = offset + displayValue.length + 1;setCaretOffset(nextCaretOffset);};return (<div className="wrapper"><label className="label">Commentvalue={value}// We'll overwrite how the combobox popover is shown, so we disable// the default behaviors.// To the combobox state, we'll only set the value after the trigger// character (the search value), so we disable the default behavior.className="combobox"<textarearef={ref}rows={5}placeholder="Type @, # or :"// We need to re-calculate the position of the combobox popover// when the textarea contents are scrolled.onScroll={combobox.render}// Hide the combobox popover whenever the selection changes.onPointerDown={combobox.hide}onChange={onChange}onKeyDown={onKeyDown}/>}/></label>hidden={!hasMatches}const textarea = ref.current;if (!textarea) return null;return getAnchorRect(textarea);}}className="popover">{matches.map((value) => (key={value}onClick={onItemClick(value)}className="combobox-item"><span>{value}</span>))}</div>);}
Components
Related examples
data:image/s3,"s3://crabby-images/bcd03/bcd03d5cc6595aa43606f4241dd2de2c059552fa" alt=""
data:image/s3,"s3://crabby-images/474e5/474e58dba759fc5fb3cacc99e5a9d8d00aaa92a9" alt=""
Animated ComboboxAnimating a Combobox using CSS transitions in React. The component waits for the transition to finish before completely hiding the popover.
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/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.
data:image/s3,"s3://crabby-images/b2dc2/b2dc233ad039a27e6359887c52dcbd4243ee411e" alt=""
data:image/s3,"s3://crabby-images/e5774/e57742aa2f4519b71fbb51660ff32973b8399390" alt=""
Multi-selectable ComboboxAllowing Combobox to select multiple options by passing an array value to the selectedValue prop.
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/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.