Alpha component
Multiselect component is subject to major changes and is for experimentation purposes only. Not recommended for use in production software.
Multiselect component is subject to major changes and is for experimentation purposes only. Not recommended for use in production software.
Multiselect allows users to make a multi selection from a list of options. It has multiple optional features like a search.
import { Multiselect } from '@contentful/f36-multiselect';
The Multiselect is a fully controlled form element and expects the selectable items as children via the compound component Multiselect.Option.
Properties are:
For the multiselect itself:
Inherited propertys from Popover Component.
It also accepts refs for the toggle button, the searchInput and the list.
For the option:
function MultiselectBasicUsageExample() {const spaces = ['Travel Blog','Finnance Blog','Fitness App','News Website','eCommerce Catalogue','Photo Gallery',];// This `useState` is going to store the selected "space" so we can show it in the UIconst [selectedSpaces, setSelectedSpaces] = React.useState([]);// This function will be called once the user selects an item in the list of optionsconst handleSelectItem = (event) => {const { checked, value } = event.target;if (checked) {setSelectedSpaces((prevState) => [...prevState, value]);} else {const newSelectedSpaces = selectedSpaces.filter((space) => space !== value,);setSelectedSpaces(newSelectedSpaces);}};return (<Stack flexDirection="column" alignItems="start"><MultiselectcurrentSelection={selectedSpaces}popoverProps={{ isFullWidth: true }}>{spaces.map((space) => {const val = space.toLowerCase().replace(/\s/g, '-');return (<Multiselect.Optionkey={`key-${val}}`}itemId={`space-${val}}`}value={space}label={space}onSelectItem={handleSelectItem}isChecked={selectedSpaces.includes(space)}/>);})}</Multiselect></Stack>);}
One optional feature of the Multiselect component is to allow filtering the list via a search field. To make it work, the onSearchValueChange callback function needs to be provided.
To enable searching the following properties have to be set.
function MultiselectSearchExample() {const spaces = ['Travel Blog','Finnance Blog','Fitness App','News Website','eCommerce Catalogue','Photo Gallery',];const [selectedItems, setSelectedItems] = React.useState([]);const [filteredItems, setFilteredItems] = React.useState(spaces);const handleSearchValueChange = (event) => {const value = event.target.value;const newFilteredItems = spaces.filter((item) =>item.toLowerCase().includes(value.toLowerCase()),);setFilteredItems(newFilteredItems);};const handleSelectItem = (event) => {const { checked, value } = event.target;if (checked) {setSelectedItems((prevState) => [...prevState, value]);} else {const newSelectedItems = selectedItems.filter((space) => space !== value);setSelectedItems(newSelectedItems);}};return (<Stack flexDirection="column" alignItems="start"><MultiselectsearchProps={{searchPlaceholder: 'Search spaces',onSearchValueChange: handleSearchValueChange,}}popoverProps={{ isFullWidth: true }}currentSelection={selectedItems}>{filteredItems.map((item, index) => {return (<Multiselect.Optionvalue={item}label={item}onSelectItem={handleSelectItem}key={`${item}-${index}`}itemId={`${item}-${index}`}isChecked={selectedItems.includes(item)}isDisabled={item === 'eCommerce Catalogue'}/>);})}</Multiselect></Stack>);}
To offer a shortcut for selecting and deselecting all options, you can use the compound component SelectAll. This requires a callback function which needs to contain your implementation for selecting all options.
function MultiselectSelectAllExample() {const spaces = React.useMemo(() => ['Travel Blog','Finnance Blog','Fitness App','News Website','eCommerce Catalogue','Photo Gallery',],[],);// This `useState` is going to store the selected "space" so we can show it in the UIconst [selectedSpaces, setSelectedSpaces] = React.useState([]);// This function will be called once the user selects an item in the list of optionsconst handleSelectItem = (event) => {const { checked, value } = event.target;if (checked) {setSelectedSpaces((prevState) => [...prevState, value]);} else {setSelectedSpaces((prevState) =>//its important to use prevState to avoid race conditions when using the state variable as reference.prevState.filter((space) => space !== value),);}};const toggleAll = (event) => {const { checked } = event.target;if (checked) {setSelectedSpaces(spaces);} else {setSelectedSpaces([]);}};const areAllSelected = React.useMemo(() => {// this can affect the app performance with a larger amount of data, consider changing it in your implementationreturn spaces.every((element) => selectedSpaces.includes(element));}, [selectedSpaces, spaces]);return (<Stack flexDirection="column" alignItems="start"><MultiselectcurrentSelection={selectedSpaces}popoverProps={{ isFullWidth: true }}><Multiselect.SelectAllonSelectItem={toggleAll}isChecked={areAllSelected}/>{spaces.map((space) => {const val = space.toLowerCase().replace(/\s/g, '-');return (<Multiselect.Optionkey={`key-${val}`}itemId={`space-${val}`}value={space}label={space}onSelectItem={handleSelectItem}isChecked={selectedSpaces.includes(space)}/>);})}</Multiselect></Stack>);}
In some cases it is desired to toggle the status of the popover drawer as well as to clear the search field from the parent. E.g. after a change has been applied. For this the Component offers two optional reference properties toggleRef
and resetSearchRef
function MultiselectReferenceExample() {const spaces = ['Travel Blog','Finnance Blog','Fitness App','News Website','eCommerce Catalogue','Photo Gallery',];const [selectedItems, setSelectedItems] = React.useState([]);const [filteredItems, setFilteredItems] = React.useState(spaces);const togglePopOverRef = React.useRef(null);const clearSearchFieldRef = React.useRef(null);const handleSearchValueChange = (event) => {const value = event.target.value;const newFilteredItems = spaces.filter((item) =>item.toLowerCase().includes(value.toLowerCase()),);setFilteredItems(newFilteredItems);};const handleSelectItem = (event) => {const { checked, value } = event.target;if (checked) {setSelectedItems((prevState) => [...prevState, value]);} else {const newSelectedItems = selectedItems.filter((space) => space !== value);setSelectedItems(newSelectedItems);}};const handleExtToggle = () => {togglePopOverRef.current.click();};const handleExtClearSearch = () => {clearSearchFieldRef.current.click();};return (<Stack flexDirection="column" alignItems="start"><Button onClick={handleExtToggle}>Toggle Popover</Button><Button onClick={handleExtClearSearch}>Clear Search Field</Button><MultiselectsearchProps={{searchPlaceholder: 'Search spaces',onSearchValueChange: handleSearchValueChange,resetSearchRef: clearSearchFieldRef,}}popoverProps={{ isFullWidth: true, closeOnBlur: false }}currentSelection={selectedItems}toggleRef={togglePopOverRef}>{filteredItems.map((item, index) => {return (<Multiselect.Optionvalue={item}label={item}onSelectItem={handleSelectItem}key={`${item}-${index}`}itemId={`${item}-${index}`}isChecked={selectedItems.includes(item)}isDisabled={item === 'eCommerce Catalogue'}/>);})}</Multiselect></Stack>);}
You can render React components in the select options, but it's important to note that this makes it your own responsibility to highlight the matching part of the search term. To help with this there's the HighlightedItem
component.
function MultiselectSearchExample() {const spaces = ['Travel Blog','Finnance Blog','Fitness App','News Website','eCommerce Catalogue','Photo Gallery',];const [searchValue, setSearchValue] = React.useState('');const [selectedItems, setSelectedItems] = React.useState([]);const handleSearchValueChange = (event) => {const value = event.target.value;setSearchValue(value);};const handleSelectItem = (event) => {const { checked, value } = event.target;if (checked) {setSelectedItems((prevState) => [...prevState, value]);} else {const newSelectedItems = selectedItems.filter((space) => space !== value);setSelectedItems(newSelectedItems);}};return (<Stack flexDirection="column" alignItems="start"><MultiselectsearchProps={{searchPlaceholder: 'Search spaces',onSearchValueChange: handleSearchValueChange,}}popoverProps={{ isFullWidth: true }}currentSelection={selectedItems}>{spaces.filter((item) =>item.toLowerCase().includes(searchValue.toLowerCase()),).map((item, index) => {return (<Multiselect.Optionvalue={item}onSelectItem={handleSelectItem}key={`${item}-${index}`}itemId={`${item}-${index}`}isChecked={selectedItems.includes(item)}isDisabled={item === 'eCommerce Catalogue'}><Multiselect.HighlightedItemitem={item}inputValue={searchValue}/>{' '}<HelpCircleIcon size="tiny" /></Multiselect.Option>);})}</Multiselect></Stack>);}