MCPcopy
hub / github.com/downshift-js/downshift

github.com/downshift-js/downshift @v9.4.0 sqlite

repository ↗ · DeepWiki ↗ · release v9.4.0 ↗
445 symbols 1,142 edges 241 files 16 documented · 4%
README

downshift 🏎 downshift logo

Primitives to build simple, flexible, WAI-ARIA compliant React autocomplete, combobox or select dropdown components.

Read the docs | See the intro blog post | Listen to the Episode 79 of the Full Stack Radio podcast


[![Build Status][build-badge]][build] [![Code Coverage][coverage-badge]][coverage] [![downloads][downloads-badge]][npmcharts] [![version][version-badge]][package] [![MIT License][license-badge]][license]

All Contributors [![PRs Welcome][prs-badge]][prs] [![Chat][chat-badge]][chat] [![Code of Conduct][coc-badge]][coc] [![Join the community on Spectrum][spectrum-badge]][spectrum]

[![Supports React and Preact][react-badge]][react] [![size][size-badge]][unpkg-dist] [![gzip size][gzip-badge]][unpkg-dist] [![module formats: umd, cjs, and es][module-formats-badge]][unpkg-dist]

The problem

You need an autocomplete, a combobox or a select experience in your application and you want it to be accessible. You also want it to be simple and flexible to account for your use cases. Finally, it should follow the [ARIA][aria] design pattern for a [combobox][combobox-aria-example] or a [select][select-aria-example], depending on your use case.

This solution

The library offers a couple of solutions. The first solution, which is the one we recommend you to try first, is a set of React hooks. Each hook provides the stateful logic needed to make the corresponding component functional and accessible. Navigate to the documentation for each by using the links in the list below.

  • [useSelect][useselect-readme] for a custom select component.
  • [useCombobox][combobox-readme] for a combobox or autocomplete input.
  • [useTagGroup][tag-group-readme] for a tag group component. Also useful to build a multiple selection combobox or select component with tags.

The second solution is the Downshift component, which can also be used to create accessible combobox and select components, providing the logic in the form of a render prop. It served as inspiration for developing the hooks and it has been around for a while. It established a successful pattern for making components accessible and functional while giving developers complete freedom when building the UI.

Both useSelect and useCombobox support the latest ARIA combobox patterns for W3C, which Downshift does not. Consequently, we strongly recommend the you use the hooks. The hooks have been migrated to the ARIA 1.2 combobox pattern in the version 7 of downshift. There is a [Migration Guide][migration-guide-v7] that documents the changes introduced in version 7.

The README on this page covers only the component while each hook has its own README page. You can navigate to the [hooks page][hooks-readme] or go directly to the hook you need by using the links in the list above.

For examples on how to use the hooks or the Downshift component, check out our [docsite][docsite]!

🚨 Use the Downshift hooks 🚨

If you are new to the library, consider the useSelect and useCombobox hooks as the first option. As mentioned above, the hooks benefit from the updated ARIA patterns and are actively maintained and improved. If there are use cases that are supported by the Downshift component and not by the hooks, please create an issue in our repo. The Downshift component is going to be removed completely once the hooks become mature.

Downshift

This is a component that controls user interactions and state for you so you can create autocomplete, combobox or select dropdown components. It uses a [render prop][use-a-render-prop] which gives you maximum flexibility with a minimal API because you are responsible for the rendering of everything and you simply apply props to what you're rendering.

This differs from other solutions which render things for their use case and then expose many options to allow for extensibility resulting in a bigger API that is less flexible as well as making the implementation more complicated and harder to contribute to.

NOTE: The original use case of this component is autocomplete, however the API is powerful and flexible enough to build things like dropdowns as well.

Table of Contents

Installation

This module is distributed via [npm][npm] which is bundled with [node][node] and should be installed as one of your project's dependencies:

npm install --save downshift

This package also depends on react. Please make sure you have it installed as well.

Note also this library supports preact out of the box. If you are using preact then use the corresponding module in the preact/dist folder. You can even import Downshift from 'downshift/preact' 👍

Usage

[Try it out in the browser][code-sandbox-try-it-out]

import * as React from 'react'
import {render} from 'react-dom'
import Downshift from 'downshift'

const items = [
  {value: 'apple'},
  {value: 'pear'},
  {value: 'orange'},
  {value: 'grape'},
  {value: 'banana'},
]

render(
  <Downshift
    onChange={selection =>
      alert(selection ? `You selected ${selection.value}` : 'Selection Cleared')
    }
    itemToString={item => (item ? item.value : '')}
  >
    {({
      getInputProps,
      getItemProps,
      getLabelProps,
      getMenuProps,
      isOpen,
      inputValue,
      highlightedIndex,
      selectedItem,
      getRootProps,
    }) => (



        <label {...getLabelProps()}>Enter a fruit</label>



          <input {...getInputProps()} />



        <ul {...getMenuProps()}>
          {isOpen
            ? items
                .filter(item => !inputValue || item.value.includes(inputValue))
                .map((item, index) => (
                  <li
                    {...getItemProps({
                      key: item.value,
                      index,
                      item,
                      style: {
                        backgroundColor:
                          highlightedIndex === index ? 'lightgray' : 'white',
                        fontWeight: selectedItem === item ? 'bold' : 'normal',
                      },
                    })}
                  >
                    {item.value}
                  </li>
                ))
            : null}
        </ul>



    )}
  </Downshift>,
  document.getElementById('root'),
)

There is also an [example without getRootProps][code-sandbox-no-get-root-props].

Warning: The example without getRootProps is not fully accessible with screen readers as it's not possible to achieve the HTML structure suggested by ARIA. We recommend following the example with getRootProps. Examples on how to use Downshift component with and without getRootProps are on the docsite.

Downshift is the only component exposed by this package. It doesn't render anything itself, it just calls the render function and renders that. ["Use a render prop!"][use-a-render-prop]! `{downshift =>

/ your JSX here! /

}`.

Basic Props

This is the list of props that you should probably know about. There are some advanced props below as well.

children

function({}) | required

This is called with an object. Read more about the properties of this object in the section "Children Function".

itemToString

function(item: any) | defaults to: item => (item ? String(item) : '')

If your items are stored as, say, objects instead of strings, downshift still needs a string representation for each one (e.g., to set inputValue).

Note: This callback must include a null check: it is invoked with null whenever the user abandons input via <Esc>.

onChange

function(selectedItem: any, stateAndHelpers: object) | optional, no useful default

Called when the selected item changes, either by the user selecting an item or the user clearing the selection. Called with the item that was selected or null and the new state of downshift. (see onStateChange for more info on stateAndHelpers).

  • selectedItem: The item that was just selected. null if the selection was cleared.
  • stateAndHelpers: This is the same thing your children function is called with (see Children Function)

stateReducer

function(state: object, changes: object) | optional

🚨 This is a really handy power feature 🚨

This function will be called each time downshift sets its internal state (or calls your onStateChange handler for control props). It allows you to modify the state change that will take place which can give you fine grain control over how the component interacts with user updates without having to use Control Props. It gives you the current state and the state that will be set, and you return the state that you want to set.

  • state: The full current state of downshift.
  • changes: These are the properties that are about to change. This also has a type property which you can learn more about in the stateChangeTypes section.
const ui = (
  <Downshift stateReducer={stateReducer}>{/* your callback */}</Downshift>
)

function stateReducer(state, changes) {
  // this prevents the menu from being closed when the user
  // selects an item with a keyboard or mouse
  switch (changes.type) {
    case Downshift.stateChangeTypes.keyDownEnter:
    case Downshift.stateChangeTypes.clickItem:
      return {
        ...changes,
        isOpen: state.isOpen,
        highlightedIndex: state.highlightedIndex,
      }
    default:
      return changes
  }
}

NOTE: This is only called when state actually changes. You should not attempt to use this to handle events. If you wish to handle events, put your event handlers directly on the elements (make sure to use the prop getters though! For example: <input onBlur={handleBlur} /> should be <input {...getInputProps({onBlur: handleBlur})} />). Also, your reducer function should be "pure." This means it should do nothing other than return the state changes you want to have happen.

Advanced Props

initialSelectedItem

any | defaults to null

Pass an item or an array of items that should be selected when downshift is initialized.

initialInputValue

string | defaults to ''

This is the initial input value when downshift is initialized.

initialHighlightedIndex

number/null | defaults to defaultHighlightedIndex

This is the initial value to set the highlighted index to when downshift is initialized.

initialIsOpen

boolean | defaults to defaultIsOpen

This is the initial isOpen value when downshift is initialized.

defaultHighlightedIndex

number/null | defaults to null

This is the value to set the highlightedIndex to anytime downshift is reset, when the selection is cleared, when an item is selected or when the inputValue is changed.

defaultIsOpen

boolean | defaults to false

This is the value to set the isOpen to anytime downshift is reset, when the the selection is cleared, or when an item is selected.

selectedItemChanged

function(prevItem: any, item: any) | defaults to: (prevItem, item) => (prevItem !== item)

Used to determine if the new `sele

Extension points exported contracts — how you extend this code

Props (Interface)
(no doc)
test/basic.test.tsx
Props (Interface)
(no doc)
test/custom.test.tsx
Environment (Interface)
(no doc)
src/downshift.types.ts
MockedEnvironment (Interface)
(no doc)
src/hooks/utils/__tests__/useMouseAndTouchTracker.test.ts
State (Interface)
(no doc)
test/basic.test.tsx
State (Interface)
(no doc)
test/custom.test.tsx
GetPropsCommonOptions (Interface)
(no doc)
src/downshift.types.ts
UseTagGroupState (Interface)
(no doc)
src/hooks/useTagGroup/index.types.ts

Core symbols most depended-on inside this repo

getToggleButton
called by 136
src/hooks/testUtils/interactions.ts
keyDownOnToggleButton
called by 118
src/hooks/testUtils/interactions.ts
getItems
called by 113
src/hooks/testUtils/interactions.ts
clickOnToggleButton
called by 76
src/hooks/testUtils/interactions.ts
getSelectedItemAtIndex
called by 69
src/hooks/useMultipleSelection/testUtils.js
callAllEventHandlers
called by 49
src/utils/callAllEventHandlers.ts
keyDownOnSelectedItemAtIndex
called by 47
src/hooks/useMultipleSelection/testUtils.js
getHighlightedIndex
called by 44
src/utils/getHighlightedIndex.ts

Shape

Function 299
Interface 95
Method 33
Class 13
Enum 5

Languages

TypeScript100%

Modules by API surface

src/downshift.js29 symbols
src/hooks/useCombobox/index.types.ts24 symbols
src/downshift.types.ts24 symbols
src/hooks/useSelect/index.types.ts21 symbols
src/hooks/useSelect/index.ts18 symbols
src/hooks/useCombobox/index.ts17 symbols
src/hooks/useMultipleSelection/index.types.ts16 symbols
src/hooks/useTagGroup/index.types.ts15 symbols
src/hooks/testUtils/interactions.ts13 symbols
src/hooks/useMultipleSelection/index.js9 symbols
src/utils.legacy.js8 symbols
src/hooks/useMultipleSelection/testUtils.js8 symbols

Dependencies from manifests, versioned

@babel/helpers7.28.6 · 1×
@babel/runtime7.28.6 · 1×
@docusaurus/babel3.10.1 · 1×
@docusaurus/core3.10.1 · 1×
@docusaurus/module-type-aliases3.10.1 · 1×
@docusaurus/preset-classic3.10.1 · 1×
@mdx-js/react3.0.1 · 1×
@playwright/test1.60.0 · 1×
@rollup/plugin-babel6.1.0 · 1×
@rollup/plugin-commonjs29.0.0 · 1×
@testing-library/dom10.4.1 · 1×

For agents

$ claude mcp add downshift \
  -- python -m otcore.mcp_server <graph>

⬇ download graph artifact