UNPKG

semantic-ui-react

Version:
150 lines (122 loc) 4.25 kB
import cx from 'clsx' import _ from 'lodash' import PropTypes from 'prop-types' import React from 'react' import { ModernAutoControlledComponent as Component, childrenUtils, createShorthandFactory, customPropTypes, getElementType, getUnhandledProps, } from '../../lib' import AccordionPanel from './AccordionPanel' const warnIfPropsAreInvalid = (props, state) => { const { exclusive } = props const { activeIndex } = state /* eslint-disable no-console */ if (exclusive && typeof activeIndex !== 'number') { console.error('`activeIndex` must be a number if `exclusive` is true') } else if (!exclusive && !_.isArray(activeIndex)) { console.error('`activeIndex` must be an array if `exclusive` is false') } /* eslint-enable no-console */ } /** * An Accordion can contain sub-accordions. */ export default class AccordionAccordion extends Component { getInitialAutoControlledState({ exclusive }) { return { activeIndex: exclusive ? -1 : [] } } componentDidMount() { if (process.env.NODE_ENV !== 'production') { warnIfPropsAreInvalid(this.props, this.state) } } componentDidUpdate() { if (process.env.NODE_ENV !== 'production') { warnIfPropsAreInvalid(this.props, this.state) } } computeNewIndex = (index) => { const { exclusive } = this.props const { activeIndex } = this.state if (exclusive) return index === activeIndex ? -1 : index // check to see if index is in array, and remove it, if not then add it return _.includes(activeIndex, index) ? _.without(activeIndex, index) : [...activeIndex, index] } handleTitleClick = (e, titleProps) => { const { index } = titleProps this.setState({ activeIndex: this.computeNewIndex(index) }) _.invoke(this.props, 'onTitleClick', e, titleProps) } isIndexActive = (index) => { const { exclusive } = this.props const { activeIndex } = this.state return exclusive ? activeIndex === index : _.includes(activeIndex, index) } render() { const { className, children, panels } = this.props const classes = cx('accordion', className) const rest = getUnhandledProps(AccordionAccordion, this.props) const ElementType = getElementType(AccordionAccordion, this.props) return ( <ElementType {...rest} className={classes}> {childrenUtils.isNil(children) ? _.map(panels, (panel, index) => AccordionPanel.create(panel, { defaultProps: { active: this.isIndexActive(index), index, onTitleClick: this.handleTitleClick, }, }), ) : children} </ElementType> ) } } AccordionAccordion.propTypes = { /** An element type to render as (string or function). */ as: PropTypes.elementType, /** Index of the currently active panel. */ activeIndex: customPropTypes.every([ customPropTypes.disallow(['children']), PropTypes.oneOfType([PropTypes.arrayOf(PropTypes.number), PropTypes.number]), ]), /** Primary content. */ children: PropTypes.node, /** Additional classes. */ className: PropTypes.string, /** Initial activeIndex value. */ defaultActiveIndex: customPropTypes.every([ customPropTypes.disallow(['children']), PropTypes.oneOfType([PropTypes.arrayOf(PropTypes.number), PropTypes.number]), ]), /** Only allow one panel open at a time. */ exclusive: PropTypes.bool, /** * Called when a panel title is clicked. * * @param {SyntheticEvent} event - React's original SyntheticEvent. * @param {object} data - All item props. */ onTitleClick: customPropTypes.every([customPropTypes.disallow(['children']), PropTypes.func]), /** Shorthand array of props for Accordion. */ panels: customPropTypes.every([ customPropTypes.disallow(['children']), PropTypes.arrayOf( PropTypes.shape({ content: customPropTypes.itemShorthand, title: customPropTypes.itemShorthand, }), ), ]), } AccordionAccordion.defaultProps = { exclusive: true, } AccordionAccordion.autoControlledProps = ['activeIndex'] AccordionAccordion.create = createShorthandFactory(AccordionAccordion, (content) => ({ content }))