UNPKG

lucid-ui

Version:

A UI component library from AppNexus.

177 lines (176 loc) 8.47 kB
import _ from 'lodash'; import React from 'react'; import PropTypes from 'react-peek/prop-types'; import { lucidClassNames } from '../../util/style-helpers'; import { findTypes, omitProps, } from '../../util/component-types'; import { buildModernHybridComponent } from '../../util/state-management'; import * as reducers from './VerticalListMenu.reducers'; import ChevronIcon from '../Icon/ChevronIcon/ChevronIcon'; import Collapsible from '../Collapsible/Collapsible'; const cx = lucidClassNames.bind('&-VerticalListMenu'); const { func, arrayOf, bool, string, number, node, object, shape } = PropTypes; const Item = (_props) => null; Item.peek = { description: ` A child item that can contain content or another VerticalListMenu. `, }; Item.displayName = 'VerticalListMenu.Item'; Item.propTypes = { hasExpander: bool ` Show or hide the expand button. Should be \`true\` if you want to nest menus. `, isExpanded: bool ` Determines the visibility of nested menus. `, isSelected: bool ` If \`true\` then a small bar on the left side of the item will be shown indicating this item is selected. `, isActionable: bool ` Determines the visibility of the small bar on the left when the user hovers over the item. This should indicate to the user that an item is clickable. `, onSelect: func ` Called when the user clicks the main body of the item. Signature: \`(index, { event, props}) => {}\` `, onToggle: func ` Called when the user clicks the expand button. Signature: \`(index, { event, props}) => {}\` `, Collapsible: shape(Collapsible.propTypes) ` Props that are passed through to the underlying Collapsible component if the item has children. `, }; const defaultProps = { onSelect: _.noop, onToggle: _.noop, expandedIndices: [], selectedIndices: [], }; class VerticalListMenu extends React.Component { constructor() { super(...arguments); this.handleToggle = (index, itemChildProp, event) => { const { onToggle } = itemChildProp; // Prevent the user from also selecting the current item. event.stopPropagation(); this.props.onToggle(index, { event, props: itemChildProp }); if (onToggle) { onToggle(index, { event, props: itemChildProp }); } }; this.handleClickItem = (index, itemChildProp, event) => { const { onSelect } = itemChildProp; this.props.onSelect(index, { event, props: itemChildProp }); if (onSelect) { onSelect(index, { event, props: itemChildProp }); } }; } render() { const { children, className, style, selectedIndices, expandedIndices, ...passThroughs } = this.props; const itemChildProps = _.map(findTypes(this.props, VerticalListMenu.Item), 'props'); return (React.createElement("ul", Object.assign({}, omitProps(passThroughs, undefined, _.keys(VerticalListMenu.propTypes)), { className: cx('&', className), style: style }), _.map(itemChildProps, (itemChildProp, index) => { const { hasExpander = false, isActionable = true, Collapsible: collapsibleProps = Collapsible.defaultProps, } = itemChildProp; const itemChildrenAsArray = React.Children.toArray(itemChildProp.children); // Was not able to get `child.Type` to work correctly, I suspect this // is due to the way we wrap components with createLucidComponentDefinition const listChildren = _.filter(itemChildrenAsArray, child => _.get(child, 'type.displayName', '') === 'VerticalListMenu'); const otherChildren = _.filter(itemChildrenAsArray, child => _.get(child, 'type.displayName', '') !== 'VerticalListMenu'); // If the prop is found on the child, it should override what was // passed in at the top level for selectedIndices and expandedIndices const actualIsExpanded = _.has(itemChildProp, 'isExpanded') ? _.get(itemChildProp, 'isExpanded', true) : _.includes(expandedIndices, index); const actualIsSelected = _.has(itemChildProp, 'isSelected') ? _.get(itemChildProp, 'isSelected', false) : _.includes(selectedIndices, index); return (React.createElement("li", Object.assign({ key: index }, itemChildProp.passThroughs, { className: cx('&-Item', itemChildProp.className) }), React.createElement("div", { className: cx('&-Item-content', { '&-Item-content-is-selected': actualIsSelected, '&-Item-content-is-not-selected': !actualIsSelected, '&-Item-content-is-expanded': actualIsExpanded, '&-Item-content-is-actionable': isActionable, }), onClick: _.partial(this.handleClickItem, index, itemChildProp) }, React.createElement("div", { className: cx('&-Item-content-body') }, React.createElement("div", { className: cx('&-Item-content-text') }, otherChildren), hasExpander ? (React.createElement("div", { className: cx('&-Item-expander'), onClick: _.partial(this.handleToggle, index, itemChildProp) }, React.createElement(ChevronIcon, { size: 12, direction: actualIsExpanded ? 'up' : 'down' }))) : null)), !_.isEmpty(listChildren) ? (React.createElement(Collapsible, Object.assign({}, collapsibleProps, { className: cx('&-Item-nested-list'), isExpanded: actualIsExpanded }), listChildren)) : null)); }), children)); } } VerticalListMenu.displayName = 'VerticalListMenu'; VerticalListMenu.Item = Item; VerticalListMenu.peek = { description: ` Used primarily for navigation lists. It supports nesting \`VerticalListMenu\`s below \`VerticalListMenu.Item\`s and animating expanding of those sub lists. The default reducer behavior is for only one \`VerticalListMenu.Item\` to be selected at any given time; that is easily overridden by handling \`onSelect\` yourself. `, categories: ['navigation'], madeFrom: ['ChevronIcon'], }; VerticalListMenu.reducers = reducers; // TODO: remove this once we move to only buildModernHybridComponent VerticalListMenu.definition = { statics: { Item, reducers, peek: { description: ` Used primarily for navigation lists. It supports nesting \`VerticalListMenu\`s below \`VerticalListMenu.Item\`s and animating expanding of those sub lists. The default reducer behavior is for only one \`VerticalListMenu.Item\` to be selected at any given time; that is easily overridden by handling \`onSelect\` yourself. `, categories: ['navigation'], madeFrom: ['ChevronIcon'], }, }, }; VerticalListMenu.propTypes = { children: node ` Regular \`children\` aren't really used in this component, but if you do add them they will be placed at the end of the component. You should be using \`VerticalListMenu.Item\`s instead of regular children. `, className: string ` Appended to the component-specific class names set on the root element. `, style: object ` Passed through to the root element. `, selectedIndices: arrayOf(number) ` Indicates which of the \`VerticalListMenu.Item\` children are currently selected. You can also put the \`isSelected\` prop directly on the \`VerticalListMenu.Item\`s if you wish. `, expandedIndices: arrayOf(number) ` Indicates which of the \`VerticalListMenu.Item\` children are currently expanded. You can also put the \`isExpanded\` prop directly on the \`VerticalListMenu.Item\`s if you wish. `, onSelect: func ` Callback fired when the user selects a \`VerticalListMenu.Item\`. Signature: \`(index, { event, props }) => {}\` `, onToggle: func ` Callback fired when the user expands or collapses an expandable \`VerticalListMenu.Item\`. Signature: \`(index, { event, props }) => {}\` `, }; VerticalListMenu.defaultProps = defaultProps; export default buildModernHybridComponent(VerticalListMenu, { reducers }); export { VerticalListMenu as VerticalListMenuDumb };