Skip to Content

Nested menu

A nested menu provides multiple levels of categories with items for selection. Categories expand to reveal additional sub-items or options.

Overview

Resources

Loading...

Loading...

Loading...

Loading...

Loading...

Install

yarn add @activecampaign/camp-components-nested-menu

Implementation details

Install

items: ItemProps<T>[]

The NestedMenu supports use as both a ‘uncontrolled’ or ‘controlled’ React component, however, consumers will likely want to track their ‘selection’ within state. Similarly to dropdown, NestedMenu supports dynamic ‘trigger’ elements. Unlike dropdown, we do not provide a ‘default’ trigger element, we expect the user to specify one. To specify the trigger, you simply can pass a ReactNode as the single ‘child’ of the NestedMenu:

type ItemProps<T> = { /** * determines if the item is a 'selectable' item, or just a styled 'menuGroup' with 'sub' items */ type: 'selectable' | 'menuGroup'; /** * the generic 'value' of the list item * - if the type is 'menuGroup', this will always be a string. else, this is type is inferred from the generic */ value: T; /** * the next layer of nesting */ items: ItemProps<T>[]; };

The consumer needs to provide a list of these objects.

In the example below, the menuGroup options has a value of type string, and the selectable options values are objects. For menuGroup options which contain nested items, the value we want to display will always be a string. our ‘selectable’ value is ‘generic’, and can be any type.

export const timezone_objects: ItemProps<TimezoneObject>[] = [ { type: 'menuGroup', value: 'America', // because 'type' is set to 'menuGroup', the header 'value' just needs to be a string items: [ { type: 'selectable', // when the type is 'selectable', the 'value' is our generic type, `TimezoneObject` value: { state: 'New York City', city: 'New York City', timezone: 'Eastern Time Zone (ET)', }, }, { type: 'selectable', value: { state: 'California', city: 'Los Angeles', timezone: 'Pacific Time Zone (PT)', }, }, { type: 'selectable', value: { state: 'Illinois', city: 'Chicago', timezone: 'Central Time Zone (CT)', }, }, { type: 'selectable', value: { state: 'Texas', city: 'Houston', timezone: 'Central Time Zone (CT)', }, }, { type: 'selectable', value: { state: 'Arizona', city: 'Phoenix', timezone: 'Mountain Time Zone (MT)', }, }, { type: 'selectable', value: { state: 'Pennsylvania', city: 'Philadelphia', timezone: 'Eastern Time Zone (ET)', }, }, { type: 'selectable', value: { state: 'Texas', city: 'San Antonio', timezone: 'Central Time Zone (CT)', }, }, { type: 'selectable', value: { state: 'California', city: 'San Diego', timezone: 'Pacific Time Zone (PT)', }, }, { type: 'selectable', value: { state: 'Texas', city: 'Dallas', timezone: 'Central Time Zone (CT)', }, }, { type: 'selectable', value: { state: 'California', city: 'San Jose', timezone: 'Pacific Time Zone (PT)', }, }, ], }, { type: 'menuGroup', value: 'Europe', items: [ { type: 'selectable', value: { country: 'France', state: 'Île-de-France Region', city: 'Paris', timezone: 'Central European Time (CET)', }, }, { type: 'selectable', value: { country: 'Sweden', city: 'Stockholm', timezone: 'Central European Time (CET)', }, }, ], }, ];

Item state

selected:T
onSelectItem?: (changes: T) => void
itemToString?: (item: T) => string

When using the component in ‘controlled’ mode with React state, the component will infer the type of the state the user is setting, as well as the items being passed to the component. If we were to use timezone_objects from the previous code snippet:

const [state, setState] = React.useState<TimezoneObject>(); return ( <NestedMenu onSelectItem={(option) => setState(option)} // 'option' should be of type TimezoneObject, this sets our state itemToString={(obj) => obj.city} // this is required if `value` is not a string, as we need to determine what text is displayed in the option items={timezone_objects} // our list of 'items' selected={state} // our controlled state object > <IconButton icon="lightning" size="small" /> </NestedMenu> );
  • onSelectItem: this is what we’re using to actually set our state object, in this case, we’re setting the entire timezone object in our state
  • itemToString: this prop is only required if the value property for your item object is not typeof string, the reason we need this prop is to specify the displayed text within the list option
  • selected: we simply pass our React state object to this prop
menuHeight?: MenuHeight

Optionally, the NestedMenu supports the ability to override the height and maxHeight of the ‘menu window’ itself (the ul element). The MenuHeight object looks like this:

type MenuHeight = Partial<{ height: React.CSSProperties['height']; maxHeight: React.CSSProperties['maxHeight']; }>;

Custom “no results” message

noSearchResultsMessage?: string
noOptionsMessage?: string

Optionally, the NestedMenu supports rendering a custom ‘no results’ and ‘no options’ message to override the default ones.

return ( <NestedMenu onSelectItem={(option) => setState(option)} // items={pallete_strings} - will need to 'not' provide 'items' in order to see 'no options' message show up selected={state} noSearchResultsMessage="nothing find, try a different search term." noOptionsMessage="you forgot to provide any 'items'!" > <IconButton icon="lightning" size="small" /> </NestedMenu> );

Custom content in a list item

renderCallback

Optionally specify a ‘render callback’ that enables rendering custom content within the list option.

(itemToStringText: string, option: T) => React.ReactNode;

This is useful for when value is an object. The callback arguments exposes two values: itemToStringText: string, the required string required to use this prop, and option: T, the value for that particular list item.

<NestedMenu onSelectItem={(option) => setState(option)} itemToString={(obj) => obj.city} renderCallback={(string, option) => ( <Styled> {/* need to render `itemToString` returned value */} <Styled>{string}</Styled> {/* additional content */} <Text.Body dangerouslySetStyles={{ color: 'slate400' }} appearance="xSmall"> {option.timezone} </Text.Body> </Styled> )} items={timezone_objects} selected={state} > <IconButton icon="lightning" size="small" /> </NestedMenu>

Setting up the trigger

A dropdown field or an icon button is most commonly used as the nested menu trigger.

NestedMenuDropdownTrigger

The NestedMenu exports out a special ‘trigger’ that allows it to function / appear similarly to the camp Dropdown component.

To set this up, simply import the NestedMenuDropdownTrigger alongside the NestedMenu, and pass it as a ‘child’ of the NestedMenu like so:

import { NestedMenu, NestedMenuDropdownTrigger } from '@activecampaign/camp-components-nested-menu' const App = () => { const [state, setState] = React.useState<string>(); return ( <NestedMenu selected={state} items={palette_strings} onSelectItem={(option) => setState(option)} {...props} > <NestedMenuDropdownTrigger helperText="helper text for dropdown trigger" label="Select an option" placeholder="Select something" /> </NestedMenu> ) }

When using the NestedMenuDropdownTrigger and working with a list of objects, you’ll need to add the renderObjectCallback prop on the trigger in order to specify which field in the selection object to render within the trigger:

Loading State

The NestedMenuDropdownTrigger has a loading state that can be used by setting the loadingIndicator prop on the trigger component to true.

import { NestedMenu, NestedMenuDropdownTrigger } from '@activecampaign/camp-components-nested-menu' const App = () => { const [state, setState] = React.useState<string>(); const [isLoading, setIsLoading] = React.useState<boolean>(true); return ( <NestedMenu selected={state} items={palette_strings} onSelectItem={(option) => setState(option)} {...props} > <NestedMenuDropdownTrigger // control the loadingIndicator prop here loadingIndicator={isLoading} helperText="helper text for dropdown trigger" label="Select an option" placeholder="Select something" /> </NestedMenu> ) }

Adding a menu action

Similar to a dropdown, a nested menu can contain an action in the footer of the menu.

<NestedMenu onSelectItem={(option) => setState(option)} itemToString={(obj) => obj?.text} items={formatted_items} selected={state} // return some jsx to render in the menu actions section renderMenuActions={(onClose: () => void) => ( <Styled styles={{ display: 'flex' }}> <Button.Fill onClick={() => { console.log('close menu!'); onClose(); }} mr="sp200" styles={{ width: '100%' }} > Trigger Close Callback </Button.Fill> </Styled> )} > <NestedMenuDropdownTrigger renderObjectCallback={(item: Selection) => item.selection} helperText="helper text for dropdown trigger" label="Nested Dropdown Trigger Label" placeholder="Select" /> </NestedMenu>

Usage

Best practices

  • Dropdown menus work best for small data sets where a user has anywhere from 5 to 15 items to choose from. For smaller datasets, checkboxes or radios should be used (depending on single or multiselect), and for datasets larger than 15, it is recommended to use a combobox instead.
  • When grouped, liste items should be in some logical order that makes it easy for the user to read and understand, whether that is numerical or alphabetical or some other contextually-based grouping.

Accessibility

Keyboard support

  • Tab or (shift + tab to move backward) opens and closes the menu and moves focus. If the menu is closed the input is in focus, then tab will navigate to the next element in the tab order.
  • To open the dropdown on focus, press space to open the menu. When the dropdown menu is open, use or to navigate list items and enter to select or deselect an item within the list. When selecting items, the menu closes after selecting in a single select dropdown but remains open in a multiselect dropdown.
  • or can also be used to open a closed dropdown menu.
  • If a multiselect dropdown has a search, upon open of the dropdown, the search input is in focus. Search can either be navigated away from using tab or arrow keys.
  • esc will close the dropdown when focused anywhere inside the component.
  • Entering search terms will either yield results, or show a ‘no results’ message if no results are found

Accessible props

The nested menu component gets its static and dynamic accessible props and attributes from a library it leverages under the hood (downshift, specifically the useSelect hook).

Roles

This component makes use of several roles:

listbox

This is used to identify an element that creates a list from which a user may select one or more static item.

<div id="downshift-0-menu" role="listbox" aria-labelledby="downshift-0-label" class="is-styled css-r6a1zw">
combobox

This role identifies an element as an input that controls another element, such as a listbox. In the case of nested menu, the ‘trigger’ element is what controls our ‘listbox’.

<button aria-activedescendant="" aria-controls="downshift-0-menu" aria-expanded="true" aria-haspopup="listbox" aria-labelledby="downshift-0-label" id="downshift-0-toggle-button" role="combobox" tabindex="0" class="icon-button is-styled css-10rjsc css-yyccrc">
option

Used to identify selections a user can make in a listbox.

<li aria-disabled="false" aria-selected="false" id="downshift-0-item-1" role="option" class="is-styled css-h3vdao" ></li>

aria properties

aria-activedescendant

This attribute identifies the currently ‘active’ element, which is the highlighted, ‘navigated’ option.

aria-controls

This attribute is set on the element that ‘controls’ the contents, or ‘presence’ of the menu.

aria-expanded

This attribute is set on the trigger element, and automatically updated based on the ‘open’ state of the nested menu. When open, this attribute is set to ‘true’, when the menu closes, it gets set to false.

aria-haspopup

The aria-haspopupattribute helps screen readers determine if an element has a popup menu / dialog associated with it.

aria-labelledby

The aria-labelledby attribute associates the trigger element with the label through the labels id.

<button aria-activedescendant="downshift-0-item-1" aria-controls="downshift-0-menu" aria-expanded="true" aria-haspopup="listbox" aria-labelledby="downshift-0-label" id="downshift-0-toggle-button" role="combobox" tabindex="0" class="icon-button is-styled css-10rjsc css-yyccrc">
aria-selected

This attribute is set on the ‘option’, and automatically updates based on the ‘selected’ state of the nested menu. An option that has been ‘selected’ has the aria-selected attribute set to ‘true’.

<li aria-disabled="false" aria-selected="true" id="downshift-0-item-1" role="option" class="is-styled css-h3vdao" ></li>
Last updated on