@patternfly/react-core
Version:
This library provides a set of common React components for use with the PatternFly reference implementation.
129 lines • 8.4 kB
JavaScript
import React, { useState, useEffect } from 'react';
import { css } from '@patternfly/react-styles';
import styles from '@patternfly/react-styles/css/components/TreeView/tree-view';
import AngleRightIcon from '@patternfly/react-icons/dist/esm/icons/angle-right-icon';
import { Badge } from '../Badge';
import { GenerateId } from '../../helpers/GenerateId/GenerateId';
const TreeViewListItemBase = ({ name, title, id, isExpanded, defaultExpanded = false, children = null, onSelect, onExpand, onCollapse, onCheck, hasCheckbox = false, checkProps = {
checked: false
}, hasBadge = false, customBadgeContent, badgeProps = { isRead: true }, isSelectable = false, isCompact, activeItems = [], itemData, parentItem, icon, expandedIcon, action, compareItems,
// eslint-disable-next-line @typescript-eslint/no-unused-vars
useMemo }) => {
const [internalIsExpanded, setIsExpanded] = useState(defaultExpanded);
useEffect(() => {
if (isExpanded !== undefined && isExpanded !== null) {
setIsExpanded(isExpanded);
}
else if (defaultExpanded !== undefined && defaultExpanded !== null) {
setIsExpanded(internalIsExpanded || defaultExpanded);
}
}, [isExpanded, defaultExpanded]);
let Component = 'button';
if (hasCheckbox) {
Component = 'label';
}
else if (isSelectable) {
Component = 'div';
}
const ToggleComponent = hasCheckbox || isSelectable ? 'button' : 'span';
const renderToggle = (randomId) => (React.createElement(ToggleComponent, Object.assign({ className: css(styles.treeViewNodeToggle), onClick: (evt) => {
if (isSelectable || hasCheckbox) {
if (internalIsExpanded) {
onCollapse && onCollapse(evt, itemData, parentItem);
}
else {
onExpand && onExpand(evt, itemData, parentItem);
}
setIsExpanded(!internalIsExpanded);
}
if (isSelectable) {
evt.stopPropagation();
}
} }, ((hasCheckbox || isSelectable) && { 'aria-labelledby': `label-${randomId}` }), (ToggleComponent === 'button' && { type: 'button' }), { tabIndex: -1 }),
React.createElement("span", { className: css(styles.treeViewNodeToggleIcon) },
React.createElement(AngleRightIcon, { "aria-hidden": "true" }))));
const isCheckboxChecked = checkProps.checked === null ? false : checkProps.checked;
const renderCheck = (randomId) => (React.createElement("span", { className: css(styles.treeViewNodeCheck) },
React.createElement("input", Object.assign({ type: "checkbox", onChange: (evt) => onCheck && onCheck(evt, itemData, parentItem), onClick: (evt) => evt.stopPropagation(), ref: (elem) => elem && (elem.indeterminate = checkProps.checked === null) }, checkProps, { checked: isCheckboxChecked, id: randomId, tabIndex: -1 }))));
const iconRendered = (React.createElement("span", { className: css(styles.treeViewNodeIcon) },
!internalIsExpanded && icon,
internalIsExpanded && (expandedIcon || icon)));
const renderNodeContent = () => {
const content = (React.createElement(React.Fragment, null,
isCompact && title && React.createElement("span", { className: css(styles.treeViewNodeTitle) }, title),
isSelectable ? (React.createElement("button", { tabIndex: -1, className: css(styles.treeViewNodeText), type: "button" }, name)) : (React.createElement("span", { className: css(styles.treeViewNodeText) }, name))));
return isCompact ? React.createElement("span", { className: css(styles.treeViewNodeContent) }, content) : content;
};
const badgeRendered = (React.createElement(React.Fragment, null,
hasBadge && children && (React.createElement("span", { className: css(styles.treeViewNodeCount) },
React.createElement(Badge, Object.assign({}, badgeProps), customBadgeContent ? customBadgeContent : children.props.data.length))),
hasBadge && !children && customBadgeContent !== undefined && (React.createElement("span", { className: css(styles.treeViewNodeCount) },
React.createElement(Badge, Object.assign({}, badgeProps), customBadgeContent)))));
const isSelected = (!children || isSelectable) &&
activeItems &&
activeItems.length > 0 &&
activeItems.some((item) => compareItems && item && compareItems(item, itemData));
return (React.createElement("li", Object.assign({ id: id, className: css(styles.treeViewListItem, internalIsExpanded && styles.modifiers.expanded), "aria-expanded": internalIsExpanded, role: "treeitem", tabIndex: -1 }, (hasCheckbox && { 'aria-checked': isCheckboxChecked }), (!hasCheckbox && { 'aria-selected': isSelected })),
React.createElement("div", { className: css(styles.treeViewContent) },
React.createElement(GenerateId, { prefix: isSelectable ? 'selectable-id' : 'checkbox-id' }, (randomId) => (React.createElement(Component, Object.assign({ className: css(styles.treeViewNode, children && (isSelectable || hasCheckbox) && styles.modifiers.selectable, isSelected && styles.modifiers.current), onClick: (evt) => {
if (!hasCheckbox) {
onSelect && onSelect(evt, itemData, parentItem);
if (!isSelectable && children && evt.isDefaultPrevented() !== true) {
if (internalIsExpanded) {
onCollapse && onCollapse(evt, itemData, parentItem);
}
else {
onExpand && onExpand(evt, itemData, parentItem);
}
setIsExpanded(!internalIsExpanded);
}
}
} }, (hasCheckbox && { htmlFor: randomId }), ((hasCheckbox || (isSelectable && children)) && { id: `label-${randomId}` }), (Component === 'button' && { type: 'button' })),
React.createElement("span", { className: css(styles.treeViewNodeContainer) },
children && renderToggle(randomId),
hasCheckbox && renderCheck(randomId),
icon && iconRendered,
renderNodeContent(),
badgeRendered)))),
action && React.createElement("div", { className: css(styles.treeViewAction) }, action)),
internalIsExpanded && children));
};
export const TreeViewListItem = React.memo(TreeViewListItemBase, (prevProps, nextProps) => {
if (!nextProps.useMemo) {
return false;
}
const prevIncludes = prevProps.activeItems &&
prevProps.activeItems.length > 0 &&
prevProps.activeItems.some((item) => prevProps.compareItems && item && prevProps.compareItems(item, prevProps.itemData));
const nextIncludes = nextProps.activeItems &&
nextProps.activeItems.length > 0 &&
nextProps.activeItems.some((item) => nextProps.compareItems && item && nextProps.compareItems(item, nextProps.itemData));
if (prevIncludes || nextIncludes) {
return false;
}
if (prevProps.name !== nextProps.name ||
prevProps.title !== nextProps.title ||
prevProps.id !== nextProps.id ||
prevProps.isExpanded !== nextProps.isExpanded ||
prevProps.defaultExpanded !== nextProps.defaultExpanded ||
prevProps.onSelect !== nextProps.onSelect ||
prevProps.onCheck !== nextProps.onCheck ||
prevProps.onExpand !== nextProps.onExpand ||
prevProps.onCollapse !== nextProps.onCollapse ||
prevProps.hasCheckbox !== nextProps.hasCheckbox ||
prevProps.checkProps !== nextProps.checkProps ||
prevProps.hasBadge !== nextProps.hasBadge ||
prevProps.customBadgeContent !== nextProps.customBadgeContent ||
prevProps.badgeProps !== nextProps.badgeProps ||
prevProps.isCompact !== nextProps.isCompact ||
prevProps.icon !== nextProps.icon ||
prevProps.expandedIcon !== nextProps.expandedIcon ||
prevProps.action !== nextProps.action ||
prevProps.parentItem !== nextProps.parentItem ||
prevProps.itemData !== nextProps.itemData) {
return false;
}
return true;
});
TreeViewListItem.displayName = 'TreeViewListItem';
//# sourceMappingURL=TreeViewListItem.js.map