react-magma-dom 4.10.0-next.14 → 4.10.0-next.16
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/components/TreeView/TreeViewContext.d.ts +5 -0
- package/dist/components/TreeView/useTreeItem.d.ts +27 -18
- package/dist/components/TreeView/useTreeView.d.ts +55 -39
- package/dist/esm/index.js +64 -26
- package/dist/esm/index.js.map +1 -1
- package/dist/properties.json +124 -3
- package/dist/react-magma-dom.cjs.development.js +64 -26
- package/dist/react-magma-dom.cjs.development.js.map +1 -1
- package/dist/react-magma-dom.cjs.production.min.js +1 -1
- package/dist/react-magma-dom.cjs.production.min.js.map +1 -1
- package/package.json +1 -1
|
@@ -1317,7 +1317,7 @@ function buildPropsWithDefaultButtonStyles(props) {
|
|
|
1317
1317
|
function buildButtonBorderRadius(props) {
|
|
1318
1318
|
switch (props.shape) {
|
|
1319
1319
|
case 'round':
|
|
1320
|
-
return '100%';
|
|
1320
|
+
return props.iconOnly ? '100%' : '2rem';
|
|
1321
1321
|
case 'leftCap':
|
|
1322
1322
|
return props.theme.borderRadius + " 0 0 " + props.theme.borderRadius;
|
|
1323
1323
|
case 'rightCap':
|
|
@@ -16044,7 +16044,11 @@ var TreeViewContext = /*#__PURE__*/React.createContext({
|
|
|
16044
16044
|
return undefined;
|
|
16045
16045
|
},
|
|
16046
16046
|
isTopLevelSelectable: true,
|
|
16047
|
-
expandedSet: /*#__PURE__*/new Set()
|
|
16047
|
+
expandedSet: /*#__PURE__*/new Set(),
|
|
16048
|
+
expandIconStyles: {
|
|
16049
|
+
size: magma.iconSizes.medium,
|
|
16050
|
+
color: undefined
|
|
16051
|
+
}
|
|
16048
16052
|
});
|
|
16049
16053
|
|
|
16050
16054
|
(function (TreeNodeType) {
|
|
@@ -17000,7 +17004,7 @@ function useTreeItem(props, forwardedRef) {
|
|
|
17000
17004
|
};
|
|
17001
17005
|
}
|
|
17002
17006
|
|
|
17003
|
-
var _excluded$1I = ["additionalContent", "children", "icon", "index", "label", "labelStyle", "style", "testId", "topLevel"];
|
|
17007
|
+
var _excluded$1I = ["additionalContent", "children", "hoverColor", "icon", "index", "label", "labelStyle", "style", "testId", "topLevel", "treeItemStyles"];
|
|
17004
17008
|
var StyledTreeItem = /*#__PURE__*/_styled("li", {
|
|
17005
17009
|
target: "e1xiryew6",
|
|
17006
17010
|
label: "StyledTreeItem"
|
|
@@ -17025,10 +17029,25 @@ var StyledTreeItem = /*#__PURE__*/_styled("li", {
|
|
|
17025
17029
|
}, ";padding-right:", function (props) {
|
|
17026
17030
|
return props.theme.spaceScale.spacing02;
|
|
17027
17031
|
}, ";", function (props) {
|
|
17028
|
-
return props.selected && /*#__PURE__*/react.css("&:before{position:absolute;background-color:", props.isInverse ? props.theme.colors.tertiary500 : props.theme.colors.primary500, ";block-size:100%;content:'';inline-size:", props.theme.spaceScale.spacing02, ";inset-block-start:0;inset-inline-start:0;};label:StyledTreeItem;" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAsDQ","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => !props.isDisabled\r\n    ? props.isInverse\r\n        ? transparentize(0.8, props.theme.colors.neutral900)\r\n        : transparentize(0.95, props.theme.colors.neutral900)\r\n    : undefined};\n    }\n  }\n`;\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${props => props.theme.spaceScale.spacing06};\n  height: ${props => props.theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: flex-start;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, icon, index, label, labelStyle, style, testId, topLevel, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { selectable, hasIcons, itemToFocus, handleExpandedChange, isTopLevelSelectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef }),\r\n            React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                        if (!isDisabled) {\r\n                            onExpandedClicked(event);\r\n                        }\r\n                    }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true })))),\r\n                shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                    hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                    treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                    labelText,\r\n                    treeItemAdditionalContent))),\r\n            React.Children.map(children, (child, index) => {\r\n                return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                    React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                        index,\r\n                        key: child.props.itemId,\r\n                        itemDepth,\r\n                        parentDepth,\r\n                        topLevel: false,\r\n                    })))) : (child);\r\n            }))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17032
|
+
return props.selected && /*#__PURE__*/react.css("&:before{position:absolute;background-color:", props.isInverse ? props.theme.colors.tertiary500 : props.theme.colors.primary500, ";block-size:100%;content:'';inline-size:", props.theme.spaceScale.spacing02, ";inset-block-start:0;inset-inline-start:0;};label:StyledTreeItem;" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAsDQ","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => getHoverBackground({\r\n    isDisabled: props.isDisabled,\r\n    hoverColor: props.hoverColor,\r\n    isInverse: props.isInverse,\r\n    theme: props.theme,\r\n})};\n    }\n  }\n`;\r\nfunction getHoverBackground({ isDisabled, hoverColor, isInverse, theme }) {\r\n    if (isDisabled)\r\n        return undefined;\r\n    if (hoverColor)\r\n        return hoverColor;\r\n    const transparency = isInverse ? 0.8 : 0.95;\r\n    return transparentize(transparency, theme.colors.neutral900);\r\n}\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => props.color ||\r\n    getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n  height: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: ${props => (props.hasCustomIconSize ? 'center' : 'flex-start')};\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index, label, labelStyle, style, testId, topLevel, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { expandIconStyles, handleExpandedChange, hasIcons, itemToFocus, isTopLevelSelectable, selectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(\"div\", { style: treeItemStyles },\r\n            React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef, hoverColor: hoverColor }),\r\n                React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, hasCustomIconSize: !!expandIconStyles?.size, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                    hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), size: expandIconStyles?.size, color: expandIconStyles?.color, \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                            if (!isDisabled) {\r\n                                onExpandedClicked(event);\r\n                            }\r\n                        }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })))),\r\n                    shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                        hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                        treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                        labelText,\r\n                        treeItemAdditionalContent))),\r\n                React.Children.map(children, (child, index) => {\r\n                    return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                            index,\r\n                            key: child.props.itemId,\r\n                            itemDepth,\r\n                            parentDepth,\r\n                            topLevel: false,\r\n                        })))) : (child);\r\n                })))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17029
17033
|
}, " &:hover{background:", function (props) {
|
|
17030
|
-
return
|
|
17031
|
-
}, ";}}" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAiBiC","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => !props.isDisabled\r\n    ? props.isInverse\r\n        ? transparentize(0.8, props.theme.colors.neutral900)\r\n        : transparentize(0.95, props.theme.colors.neutral900)\r\n    : undefined};\n    }\n  }\n`;\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${props => props.theme.spaceScale.spacing06};\n  height: ${props => props.theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: flex-start;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, icon, index, label, labelStyle, style, testId, topLevel, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { selectable, hasIcons, itemToFocus, handleExpandedChange, isTopLevelSelectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef }),\r\n            React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                        if (!isDisabled) {\r\n                            onExpandedClicked(event);\r\n                        }\r\n                    }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true })))),\r\n                shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                    hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                    treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                    labelText,\r\n                    treeItemAdditionalContent))),\r\n            React.Children.map(children, (child, index) => {\r\n                return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                    React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                        index,\r\n                        key: child.props.itemId,\r\n                        itemDepth,\r\n                        parentDepth,\r\n                        topLevel: false,\r\n                    })))) : (child);\r\n            }))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17034
|
+
return getHoverBackground({
|
|
17035
|
+
isDisabled: props.isDisabled,
|
|
17036
|
+
hoverColor: props.hoverColor,
|
|
17037
|
+
isInverse: props.isInverse,
|
|
17038
|
+
theme: props.theme
|
|
17039
|
+
});
|
|
17040
|
+
}, ";}}" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAiBiC","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => getHoverBackground({\r\n    isDisabled: props.isDisabled,\r\n    hoverColor: props.hoverColor,\r\n    isInverse: props.isInverse,\r\n    theme: props.theme,\r\n})};\n    }\n  }\n`;\r\nfunction getHoverBackground({ isDisabled, hoverColor, isInverse, theme }) {\r\n    if (isDisabled)\r\n        return undefined;\r\n    if (hoverColor)\r\n        return hoverColor;\r\n    const transparency = isInverse ? 0.8 : 0.95;\r\n    return transparentize(transparency, theme.colors.neutral900);\r\n}\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => props.color ||\r\n    getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n  height: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: ${props => (props.hasCustomIconSize ? 'center' : 'flex-start')};\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index, label, labelStyle, style, testId, topLevel, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { expandIconStyles, handleExpandedChange, hasIcons, itemToFocus, isTopLevelSelectable, selectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(\"div\", { style: treeItemStyles },\r\n            React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef, hoverColor: hoverColor }),\r\n                React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, hasCustomIconSize: !!expandIconStyles?.size, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                    hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), size: expandIconStyles?.size, color: expandIconStyles?.color, \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                            if (!isDisabled) {\r\n                                onExpandedClicked(event);\r\n                            }\r\n                        }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })))),\r\n                    shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                        hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                        treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                        labelText,\r\n                        treeItemAdditionalContent))),\r\n                React.Children.map(children, (child, index) => {\r\n                    return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                            index,\r\n                            key: child.props.itemId,\r\n                            itemDepth,\r\n                            parentDepth,\r\n                            topLevel: false,\r\n                        })))) : (child);\r\n                })))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17041
|
+
function getHoverBackground(_ref) {
|
|
17042
|
+
var isDisabled = _ref.isDisabled,
|
|
17043
|
+
hoverColor = _ref.hoverColor,
|
|
17044
|
+
isInverse = _ref.isInverse,
|
|
17045
|
+
theme = _ref.theme;
|
|
17046
|
+
if (isDisabled) return undefined;
|
|
17047
|
+
if (hoverColor) return hoverColor;
|
|
17048
|
+
var transparency = isInverse ? 0.8 : 0.95;
|
|
17049
|
+
return polished.transparentize(transparency, theme.colors.neutral900);
|
|
17050
|
+
}
|
|
17032
17051
|
var IconWrapper$8 = /*#__PURE__*/_styled("span", {
|
|
17033
17052
|
target: "e1xiryew5",
|
|
17034
17053
|
label: "IconWrapper"
|
|
@@ -17040,25 +17059,29 @@ var IconWrapper$8 = /*#__PURE__*/_styled("span", {
|
|
|
17040
17059
|
return props.theme.iconSizes.medium;
|
|
17041
17060
|
}, "px;width:", function (props) {
|
|
17042
17061
|
return props.theme.iconSizes.medium;
|
|
17043
|
-
}, "px;vertical-align:middle;}" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AA4EgC","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => !props.isDisabled\r\n    ? props.isInverse\r\n        ? transparentize(0.8, props.theme.colors.neutral900)\r\n        : transparentize(0.95, props.theme.colors.neutral900)\r\n    : undefined};\n    }\n  }\n`;\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${props => props.theme.spaceScale.spacing06};\n  height: ${props => props.theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: flex-start;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, icon, index, label, labelStyle, style, testId, topLevel, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { selectable, hasIcons, itemToFocus, handleExpandedChange, isTopLevelSelectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef }),\r\n            React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                        if (!isDisabled) {\r\n                            onExpandedClicked(event);\r\n                        }\r\n                    }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true })))),\r\n                shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                    hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                    treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                    labelText,\r\n                    treeItemAdditionalContent))),\r\n            React.Children.map(children, (child, index) => {\r\n                return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                    React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                        index,\r\n                        key: child.props.itemId,\r\n                        itemDepth,\r\n                        parentDepth,\r\n                        topLevel: false,\r\n                    })))) : (child);\r\n            }))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17062
|
+
}, "px;vertical-align:middle;}" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAqFgC","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => getHoverBackground({\r\n    isDisabled: props.isDisabled,\r\n    hoverColor: props.hoverColor,\r\n    isInverse: props.isInverse,\r\n    theme: props.theme,\r\n})};\n    }\n  }\n`;\r\nfunction getHoverBackground({ isDisabled, hoverColor, isInverse, theme }) {\r\n    if (isDisabled)\r\n        return undefined;\r\n    if (hoverColor)\r\n        return hoverColor;\r\n    const transparency = isInverse ? 0.8 : 0.95;\r\n    return transparentize(transparency, theme.colors.neutral900);\r\n}\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => props.color ||\r\n    getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n  height: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: ${props => (props.hasCustomIconSize ? 'center' : 'flex-start')};\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index, label, labelStyle, style, testId, topLevel, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { expandIconStyles, handleExpandedChange, hasIcons, itemToFocus, isTopLevelSelectable, selectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(\"div\", { style: treeItemStyles },\r\n            React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef, hoverColor: hoverColor }),\r\n                React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, hasCustomIconSize: !!expandIconStyles?.size, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                    hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), size: expandIconStyles?.size, color: expandIconStyles?.color, \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                            if (!isDisabled) {\r\n                                onExpandedClicked(event);\r\n                            }\r\n                        }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })))),\r\n                    shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                        hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                        treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                        labelText,\r\n                        treeItemAdditionalContent))),\r\n                React.Children.map(children, (child, index) => {\r\n                    return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                            index,\r\n                            key: child.props.itemId,\r\n                            itemDepth,\r\n                            parentDepth,\r\n                            topLevel: false,\r\n                        })))) : (child);\r\n                })))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17044
17063
|
var StyledLabelWrapper = /*#__PURE__*/_styled("span", {
|
|
17045
17064
|
target: "e1xiryew4",
|
|
17046
17065
|
label: "StyledLabelWrapper"
|
|
17047
17066
|
})("display:flex;align-items:flex-start;color:", function (props) {
|
|
17048
17067
|
return getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme);
|
|
17049
|
-
}, ";width:100%;" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAuFuC","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => !props.isDisabled\r\n    ? props.isInverse\r\n        ? transparentize(0.8, props.theme.colors.neutral900)\r\n        : transparentize(0.95, props.theme.colors.neutral900)\r\n    : undefined};\n    }\n  }\n`;\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${props => props.theme.spaceScale.spacing06};\n  height: ${props => props.theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: flex-start;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, icon, index, label, labelStyle, style, testId, topLevel, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { selectable, hasIcons, itemToFocus, handleExpandedChange, isTopLevelSelectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef }),\r\n            React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                        if (!isDisabled) {\r\n                            onExpandedClicked(event);\r\n                        }\r\n                    }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true })))),\r\n                shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                    hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                    treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                    labelText,\r\n                    treeItemAdditionalContent))),\r\n            React.Children.map(children, (child, index) => {\r\n                return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                    React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                        index,\r\n                        key: child.props.itemId,\r\n                        itemDepth,\r\n                        parentDepth,\r\n                        topLevel: false,\r\n                    })))) : (child);\r\n            }))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17068
|
+
}, ";width:100%;" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAgGuC","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => getHoverBackground({\r\n    isDisabled: props.isDisabled,\r\n    hoverColor: props.hoverColor,\r\n    isInverse: props.isInverse,\r\n    theme: props.theme,\r\n})};\n    }\n  }\n`;\r\nfunction getHoverBackground({ isDisabled, hoverColor, isInverse, theme }) {\r\n    if (isDisabled)\r\n        return undefined;\r\n    if (hoverColor)\r\n        return hoverColor;\r\n    const transparency = isInverse ? 0.8 : 0.95;\r\n    return transparentize(transparency, theme.colors.neutral900);\r\n}\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => props.color ||\r\n    getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n  height: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: ${props => (props.hasCustomIconSize ? 'center' : 'flex-start')};\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index, label, labelStyle, style, testId, topLevel, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { expandIconStyles, handleExpandedChange, hasIcons, itemToFocus, isTopLevelSelectable, selectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(\"div\", { style: treeItemStyles },\r\n            React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef, hoverColor: hoverColor }),\r\n                React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, hasCustomIconSize: !!expandIconStyles?.size, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                    hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), size: expandIconStyles?.size, color: expandIconStyles?.color, \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                            if (!isDisabled) {\r\n                                onExpandedClicked(event);\r\n                            }\r\n                        }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })))),\r\n                    shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                        hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                        treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                        labelText,\r\n                        treeItemAdditionalContent))),\r\n                React.Children.map(children, (child, index) => {\r\n                    return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                            index,\r\n                            key: child.props.itemId,\r\n                            itemDepth,\r\n                            parentDepth,\r\n                            topLevel: false,\r\n                        })))) : (child);\r\n                })))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17050
17069
|
var StyledExpandWrapper = /*#__PURE__*/_styled("div", {
|
|
17051
17070
|
target: "e1xiryew3",
|
|
17052
17071
|
label: "StyledExpandWrapper"
|
|
17053
17072
|
})("display:inline-block;vertical-align:middle;margin-right:", function (props) {
|
|
17054
17073
|
return props.theme.spaceScale.spacing03;
|
|
17055
17074
|
}, ";color:", function (props) {
|
|
17056
|
-
return getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme);
|
|
17057
|
-
}, ";border-radius:0;width:", function (
|
|
17058
|
-
|
|
17059
|
-
|
|
17060
|
-
return
|
|
17061
|
-
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AA6FuC","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => !props.isDisabled\r\n    ? props.isInverse\r\n        ? transparentize(0.8, props.theme.colors.neutral900)\r\n        : transparentize(0.95, props.theme.colors.neutral900)\r\n    : undefined};\n    }\n  }\n`;\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${props => props.theme.spaceScale.spacing06};\n  height: ${props => props.theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: flex-start;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, icon, index, label, labelStyle, style, testId, topLevel, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { selectable, hasIcons, itemToFocus, handleExpandedChange, isTopLevelSelectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef }),\r\n            React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                        if (!isDisabled) {\r\n                            onExpandedClicked(event);\r\n                        }\r\n                    }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true })))),\r\n                shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                    hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                    treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                    labelText,\r\n                    treeItemAdditionalContent))),\r\n            React.Children.map(children, (child, index) => {\r\n                return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                    React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                        index,\r\n                        key: child.props.itemId,\r\n                        itemDepth,\r\n                        parentDepth,\r\n                        topLevel: false,\r\n                    })))) : (child);\r\n            }))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17075
|
+
return props.color || getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme);
|
|
17076
|
+
}, ";border-radius:0;width:", function (_ref2) {
|
|
17077
|
+
var size = _ref2.size,
|
|
17078
|
+
theme = _ref2.theme;
|
|
17079
|
+
return size !== undefined ? size + "px" : theme.spaceScale.spacing06;
|
|
17080
|
+
}, ";height:", function (_ref3) {
|
|
17081
|
+
var size = _ref3.size,
|
|
17082
|
+
theme = _ref3.theme;
|
|
17083
|
+
return size !== undefined ? size + "px" : theme.spaceScale.spacing06;
|
|
17084
|
+
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAsGuC","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => getHoverBackground({\r\n    isDisabled: props.isDisabled,\r\n    hoverColor: props.hoverColor,\r\n    isInverse: props.isInverse,\r\n    theme: props.theme,\r\n})};\n    }\n  }\n`;\r\nfunction getHoverBackground({ isDisabled, hoverColor, isInverse, theme }) {\r\n    if (isDisabled)\r\n        return undefined;\r\n    if (hoverColor)\r\n        return hoverColor;\r\n    const transparency = isInverse ? 0.8 : 0.95;\r\n    return transparentize(transparency, theme.colors.neutral900);\r\n}\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => props.color ||\r\n    getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n  height: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: ${props => (props.hasCustomIconSize ? 'center' : 'flex-start')};\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index, label, labelStyle, style, testId, topLevel, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { expandIconStyles, handleExpandedChange, hasIcons, itemToFocus, isTopLevelSelectable, selectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(\"div\", { style: treeItemStyles },\r\n            React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef, hoverColor: hoverColor }),\r\n                React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, hasCustomIconSize: !!expandIconStyles?.size, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                    hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), size: expandIconStyles?.size, color: expandIconStyles?.color, \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                            if (!isDisabled) {\r\n                                onExpandedClicked(event);\r\n                            }\r\n                        }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })))),\r\n                    shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                        hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                        treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                        labelText,\r\n                        treeItemAdditionalContent))),\r\n                React.Children.map(children, (child, index) => {\r\n                    return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                            index,\r\n                            key: child.props.itemId,\r\n                            itemDepth,\r\n                            parentDepth,\r\n                            topLevel: false,\r\n                        })))) : (child);\r\n                })))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17062
17085
|
var StyledCheckboxWrapper = /*#__PURE__*/_styled("div", {
|
|
17063
17086
|
target: "e1xiryew2",
|
|
17064
17087
|
label: "StyledCheckboxWrapper"
|
|
@@ -17068,40 +17091,45 @@ var StyledCheckboxWrapper = /*#__PURE__*/_styled("div", {
|
|
|
17068
17091
|
return props.hasAdditionalContent ? 'flex' : 'inline-flex';
|
|
17069
17092
|
}, ";flex-direction:column;width:", function (props) {
|
|
17070
17093
|
return "calc(100% - " + props.theme.spaceScale.spacing03 + ")";
|
|
17071
|
-
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAsGyC","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => !props.isDisabled\r\n    ? props.isInverse\r\n        ? transparentize(0.8, props.theme.colors.neutral900)\r\n        : transparentize(0.95, props.theme.colors.neutral900)\r\n    : undefined};\n    }\n  }\n`;\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${props => props.theme.spaceScale.spacing06};\n  height: ${props => props.theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: flex-start;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, icon, index, label, labelStyle, style, testId, topLevel, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { selectable, hasIcons, itemToFocus, handleExpandedChange, isTopLevelSelectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef }),\r\n            React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                        if (!isDisabled) {\r\n                            onExpandedClicked(event);\r\n                        }\r\n                    }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true })))),\r\n                shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                    hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                    treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                    labelText,\r\n                    treeItemAdditionalContent))),\r\n            React.Children.map(children, (child, index) => {\r\n                return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                    React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                        index,\r\n                        key: child.props.itemId,\r\n                        itemDepth,\r\n                        parentDepth,\r\n                        topLevel: false,\r\n                    })))) : (child);\r\n            }))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17094
|
+
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAgHyC","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => getHoverBackground({\r\n    isDisabled: props.isDisabled,\r\n    hoverColor: props.hoverColor,\r\n    isInverse: props.isInverse,\r\n    theme: props.theme,\r\n})};\n    }\n  }\n`;\r\nfunction getHoverBackground({ isDisabled, hoverColor, isInverse, theme }) {\r\n    if (isDisabled)\r\n        return undefined;\r\n    if (hoverColor)\r\n        return hoverColor;\r\n    const transparency = isInverse ? 0.8 : 0.95;\r\n    return transparentize(transparency, theme.colors.neutral900);\r\n}\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => props.color ||\r\n    getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n  height: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: ${props => (props.hasCustomIconSize ? 'center' : 'flex-start')};\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index, label, labelStyle, style, testId, topLevel, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { expandIconStyles, handleExpandedChange, hasIcons, itemToFocus, isTopLevelSelectable, selectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(\"div\", { style: treeItemStyles },\r\n            React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef, hoverColor: hoverColor }),\r\n                React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, hasCustomIconSize: !!expandIconStyles?.size, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                    hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), size: expandIconStyles?.size, color: expandIconStyles?.color, \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                            if (!isDisabled) {\r\n                                onExpandedClicked(event);\r\n                            }\r\n                        }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })))),\r\n                    shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                        hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                        treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                        labelText,\r\n                        treeItemAdditionalContent))),\r\n                React.Children.map(children, (child, index) => {\r\n                    return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                            index,\r\n                            key: child.props.itemId,\r\n                            itemDepth,\r\n                            parentDepth,\r\n                            topLevel: false,\r\n                        })))) : (child);\r\n                })))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17072
17095
|
var StyledItemWrapper = /*#__PURE__*/_styled("div", {
|
|
17073
17096
|
target: "e1xiryew1",
|
|
17074
17097
|
label: "StyledItemWrapper"
|
|
17075
17098
|
})("display:flex;flex-direction:", function (props) {
|
|
17076
17099
|
return props.hasAdditionalContent ? 'column' : 'row';
|
|
17077
|
-
}, ";align-items:
|
|
17100
|
+
}, ";align-items:", function (props) {
|
|
17101
|
+
return props.hasCustomIconSize ? 'center' : 'flex-start';
|
|
17102
|
+
}, ";cursor:", function (props) {
|
|
17078
17103
|
return getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType);
|
|
17079
|
-
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AA6GqC","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => !props.isDisabled\r\n    ? props.isInverse\r\n        ? transparentize(0.8, props.theme.colors.neutral900)\r\n        : transparentize(0.95, props.theme.colors.neutral900)\r\n    : undefined};\n    }\n  }\n`;\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${props => props.theme.spaceScale.spacing06};\n  height: ${props => props.theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: flex-start;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, icon, index, label, labelStyle, style, testId, topLevel, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { selectable, hasIcons, itemToFocus, handleExpandedChange, isTopLevelSelectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef }),\r\n            React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                        if (!isDisabled) {\r\n                            onExpandedClicked(event);\r\n                        }\r\n                    }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true })))),\r\n                shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                    hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                    treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                    labelText,\r\n                    treeItemAdditionalContent))),\r\n            React.Children.map(children, (child, index) => {\r\n                return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                    React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                        index,\r\n                        key: child.props.itemId,\r\n                        itemDepth,\r\n                        parentDepth,\r\n                        topLevel: false,\r\n                    })))) : (child);\r\n            }))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17104
|
+
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAuHqC","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => getHoverBackground({\r\n    isDisabled: props.isDisabled,\r\n    hoverColor: props.hoverColor,\r\n    isInverse: props.isInverse,\r\n    theme: props.theme,\r\n})};\n    }\n  }\n`;\r\nfunction getHoverBackground({ isDisabled, hoverColor, isInverse, theme }) {\r\n    if (isDisabled)\r\n        return undefined;\r\n    if (hoverColor)\r\n        return hoverColor;\r\n    const transparency = isInverse ? 0.8 : 0.95;\r\n    return transparentize(transparency, theme.colors.neutral900);\r\n}\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => props.color ||\r\n    getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n  height: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: ${props => (props.hasCustomIconSize ? 'center' : 'flex-start')};\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index, label, labelStyle, style, testId, topLevel, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { expandIconStyles, handleExpandedChange, hasIcons, itemToFocus, isTopLevelSelectable, selectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(\"div\", { style: treeItemStyles },\r\n            React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef, hoverColor: hoverColor }),\r\n                React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, hasCustomIconSize: !!expandIconStyles?.size, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                    hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), size: expandIconStyles?.size, color: expandIconStyles?.color, \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                            if (!isDisabled) {\r\n                                onExpandedClicked(event);\r\n                            }\r\n                        }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })))),\r\n                    shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                        hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                        treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                        labelText,\r\n                        treeItemAdditionalContent))),\r\n                React.Children.map(children, (child, index) => {\r\n                    return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                            index,\r\n                            key: child.props.itemId,\r\n                            itemDepth,\r\n                            parentDepth,\r\n                            topLevel: false,\r\n                        })))) : (child);\r\n                })))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17080
17105
|
var AdditionalContentWrapper$1 = /*#__PURE__*/_styled("div", {
|
|
17081
17106
|
target: "e1xiryew0",
|
|
17082
17107
|
label: "AdditionalContentWrapper"
|
|
17083
17108
|
})("margin-bottom:", function (props) {
|
|
17084
17109
|
return props.theme.spaceScale.spacing05;
|
|
17085
|
-
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAmH4C","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => !props.isDisabled\r\n    ? props.isInverse\r\n        ? transparentize(0.8, props.theme.colors.neutral900)\r\n        : transparentize(0.95, props.theme.colors.neutral900)\r\n    : undefined};\n    }\n  }\n`;\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${props => props.theme.spaceScale.spacing06};\n  height: ${props => props.theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: flex-start;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, icon, index, label, labelStyle, style, testId, topLevel, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { selectable, hasIcons, itemToFocus, handleExpandedChange, isTopLevelSelectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef }),\r\n            React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                        if (!isDisabled) {\r\n                            onExpandedClicked(event);\r\n                        }\r\n                    }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true })))),\r\n                shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                    hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                    treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                    labelText,\r\n                    treeItemAdditionalContent))),\r\n            React.Children.map(children, (child, index) => {\r\n                return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                    React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                        index,\r\n                        key: child.props.itemId,\r\n                        itemDepth,\r\n                        parentDepth,\r\n                        topLevel: false,\r\n                    })))) : (child);\r\n            }))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17110
|
+
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AA6H4C","file":"TreeItem.tsx","sourcesContent":["import * as React from 'react';\r\nimport { css } from '@emotion/react';\r\nimport styled from '@emotion/styled';\r\nimport { transparentize } from 'polished';\r\nimport { ArticleIcon, ChevronRightIcon, ExpandMoreIcon, FolderIcon, } from 'react-magma-icons';\r\nimport { TreeItemContext } from './TreeItemContext';\r\nimport { TreeViewContext } from './TreeViewContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { mergeRefs } from '../../utils';\r\nconst StyledTreeItem = styled.li `\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral700};\n  list-style-type: none;\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType)};\n  position: relative;\n  margin-bottom: 0;\n\n  padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth)};\n\n  &:focus {\n    outline: none;\n\n    & > *:first-child {\n      outline-offset: -2px;\n      outline: 2px solid\n        ${props => props.isInverse\r\n    ? props.theme.colors.focusInverse\r\n    : props.theme.colors.focus};\n    }\n  }\n\n  > div:first-of-type {\n    background: ${props => props.selected && props.isInverse\r\n    ? transparentize(0.7, props.theme.colors.neutral900)\r\n    : props.selected &&\r\n        transparentize(0.92, props.theme.colors.neutral900)};\n    position: relative;\n\n    padding-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true)};\n    padding-block-end: ${props => props.theme.spaceScale.spacing02};\n    padding-block-start: ${props => props.theme.spaceScale.spacing02};\n    padding-right: ${props => props.theme.spaceScale.spacing02};\n\n    ${props => props.selected &&\r\n    css `\n        &:before {\n          position: absolute;\n          background-color: ${props.isInverse\r\n        ? props.theme.colors.tertiary500\r\n        : props.theme.colors.primary500};\n          block-size: 100%;\n          content: '';\n          inline-size: ${props.theme.spaceScale.spacing02};\n          inset-block-start: 0;\n          inset-inline-start: 0;\n        }\n      `}\n    &:hover {\n      background: ${props => getHoverBackground({\r\n    isDisabled: props.isDisabled,\r\n    hoverColor: props.hoverColor,\r\n    isInverse: props.isInverse,\r\n    theme: props.theme,\r\n})};\n    }\n  }\n`;\r\nfunction getHoverBackground({ isDisabled, hoverColor, isInverse, theme }) {\r\n    if (isDisabled)\r\n        return undefined;\r\n    if (hoverColor)\r\n        return hoverColor;\r\n    const transparency = isInverse ? 0.8 : 0.95;\r\n    return transparentize(transparency, theme.colors.neutral900);\r\n}\r\nconst IconWrapper = styled.span `\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  margin-left: 0;\n\n  svg {\n    height: ${props => props.theme.iconSizes.medium}px;\n    width: ${props => props.theme.iconSizes.medium}px;\n    vertical-align: middle;\n  }\n`;\r\nconst StyledLabelWrapper = styled.span `\n  display: flex;\n  align-items: flex-start;\n  color: ${props => getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  width: 100%;\n`;\r\nconst StyledExpandWrapper = styled.div `\n  display: inline-block;\n  vertical-align: middle;\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  color: ${props => props.color ||\r\n    getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme)};\n  border-radius: 0;\n  width: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n  height: ${({ size, theme }) => size !== undefined ? `${size}px` : theme.spaceScale.spacing06};\n`;\r\nconst StyledCheckboxWrapper = styled.div `\n  margin-right: ${props => props.theme.spaceScale.spacing03};\n  vertical-align: middle;\n  display: ${props => (props.hasAdditionalContent ? 'flex' : 'inline-flex')};\n  flex-direction: column;\n  width: ${props => `calc(100% - ${props.theme.spaceScale.spacing03})`};\n`;\r\nconst StyledItemWrapper = styled.div `\n  display: flex;\n  flex-direction: ${props => (props.hasAdditionalContent ? 'column' : 'row')};\n  align-items: ${props => (props.hasCustomIconSize ? 'center' : 'flex-start')};\n  cursor: ${props => getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType)};\n`;\r\nconst AdditionalContentWrapper = styled.div `\n  margin-bottom: ${props => props.theme.spaceScale.spacing05};\n`;\r\nexport const TreeItem = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index, label, labelStyle, style, testId, topLevel, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    const { expandIconStyles, handleExpandedChange, hasIcons, itemToFocus, isTopLevelSelectable, selectable, } = React.useContext(TreeViewContext);\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(props, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, parentDepth, ref, selectedItems, } = contextValue;\r\n    const nodeType = hasOwnTreeItems ? TreeNodeType.branch : TreeNodeType.leaf;\r\n    const selectedItem = selectable === TreeViewSelectable.single\r\n        ? selectedItems?.[0]?.itemId === itemId\r\n        : null;\r\n    const ariaCheckedValue = selectable === TreeViewSelectable.multi\r\n        ? checkedStatus === IndeterminateCheckboxStatus.indeterminate\r\n            ? 'mixed'\r\n            : checkedStatus === IndeterminateCheckboxStatus.checked\r\n        : null;\r\n    const [isInsideTreeItem, setIsInsideTreeItem] = React.useState(false);\r\n    const treeItemRef = React.useRef(null);\r\n    const focusTrapElement = useFocusLock(isInsideTreeItem);\r\n    const interactiveElements = 'button, [role=\"button\"], input, select, textarea, a[href], [tabindex]:not([tabindex=\"-1\"])';\r\n    const getInteractiveElements = (container, selector) => {\r\n        return Array.from(container.querySelectorAll(selector)).filter(el => !el.hasAttribute('tabindex') ||\r\n            (el.tabIndex !== undefined && el.tabIndex >= 0));\r\n    };\r\n    /**\r\n     * This function allows for keyboard navigation within the label and additional content of a tree item.\r\n     *\r\n     * You can navigate through interactive elements using the `Tab` key, activate them with `Enter` or `Space`,\r\n     * and exit outside and focus the whole tree item with `Escape`.\r\n     * **/\r\n    const handleLabelAndAdditionalContentKeyDown = (event) => {\r\n        const { key, target, currentTarget, shiftKey } = event;\r\n        const currentElement = target;\r\n        const isEnter = key === 'Enter';\r\n        const isSpace = key === ' ';\r\n        const isEscape = key === 'Escape';\r\n        const isTab = key === 'Tab';\r\n        const isActivationKey = isEnter || isSpace;\r\n        const interactiveElement = currentElement.closest(interactiveElements);\r\n        // If the key is `Tab`, we navigate through interactive elements inside the tree item\r\n        if (isTab && isInsideTreeItem) {\r\n            event.preventDefault();\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            // Filter list of interactive elements which are only included for current tree item\r\n            const currentTreeItemInteractiveElements = interactiveElementsList.filter(el => {\r\n                const closestTreeItem = el.closest('[role=\"treeitem\"]');\r\n                return closestTreeItem === treeItemRef.current;\r\n            });\r\n            const currentIndex = currentTreeItemInteractiveElements.indexOf(currentElement);\r\n            const direction = shiftKey ? -1 : 1;\r\n            const total = currentTreeItemInteractiveElements.length;\r\n            const nextIndex = (currentIndex + direction + total) % total;\r\n            const elementToFocus = currentTreeItemInteractiveElements[nextIndex];\r\n            if (elementToFocus) {\r\n                setTimeout(() => elementToFocus.focus(), 0);\r\n            }\r\n            return;\r\n        }\r\n        // Pressing `Enter` or `Space` on an interactive element will trigger its click event\r\n        if (isActivationKey && interactiveElement) {\r\n            event.preventDefault();\r\n            interactiveElement.click();\r\n        }\r\n        // Moves focus outside the tree item and focuses the tree item itself when `Escape` is pressed\r\n        if (isEscape) {\r\n            event.preventDefault();\r\n            event.stopPropagation();\r\n            setIsInsideTreeItem(false);\r\n            const treeItemNode = treeItemRef.current;\r\n            if (treeItemNode) {\r\n                treeItemNode.focus();\r\n            }\r\n            return;\r\n        }\r\n    };\r\n    const handleOnClick = (event) => {\r\n        if (isDisabled) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        const currentElement = event.target;\r\n        const interactiveElement = currentElement.closest('button, [role=\"button\"], a[href], input, select, textarea, [role=\"menuitem\"]');\r\n        // Preventing selecting the item when clicking on interactive elements when `selectable` is `single`\r\n        if (interactiveElement) {\r\n            event.stopPropagation();\r\n            return;\r\n        }\r\n        if (selectable === TreeViewSelectable.single) {\r\n            handleClick(event, itemId);\r\n        }\r\n    };\r\n    const defaultIcon = nodeType === TreeNodeType.branch ? (React.createElement(FolderIcon, { \"aria-hidden\": true })) : (React.createElement(ArticleIcon, { \"aria-hidden\": true }));\r\n    const labelText = (React.createElement(StyledLabelWrapper, { theme: theme, isDisabled: isDisabled, isInverse: isInverse, style: labelStyle, id: `${itemId}-label`, \"data-testid\": `${testId || itemId}-label` },\r\n        hasIcons && (React.createElement(IconWrapper, { isInverse: isInverse, theme: theme, isDisabled: isDisabled, \"data-testid\": `${testId || itemId}-icon` }, icon || defaultIcon)),\r\n        label));\r\n    const treeItemAdditionalContent = additionalContent ? (React.createElement(AdditionalContentWrapper, { theme: theme, id: `${itemId}-additionalcontentwrapper`, \"data-testid\": `${testId ?? itemId}-additionalcontentwrapper` }, additionalContent)) : null;\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = {\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: { marginRight: theme.spaceScale.spacing03 },\r\n        labelStyle: {\r\n            padding: 0,\r\n            width: '100%',\r\n        },\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    };\r\n    const onExpandedClicked = (event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    };\r\n    const tabIndex = React.useMemo(() => {\r\n        if (isDisabled) {\r\n            return undefined;\r\n        }\r\n        return itemToFocus === itemId ? 0 : -1;\r\n    }, [isDisabled, itemToFocus, itemId]);\r\n    const shouldShowCheckbox = selectable === TreeViewSelectable.multi &&\r\n        (isTopLevelSelectable !== false || !topLevel);\r\n    /**\r\n     * This function allows for keyboard navigation within the tree item.\r\n     *\r\n     * Pressing `Ctrl + Enter` or `Command + Enter` focuses the first interactive element within the tree item\r\n     * and locks focus inside the tree item.\r\n     *\r\n     * If the focus is within the label or additional content, it handles key events for interaction elements.\r\n     *\r\n     * It also handles the other keys to trigger the click event on the tree item.\r\n     * **/\r\n    const onKeyDownHandler = (event) => {\r\n        const { key, target, currentTarget } = event;\r\n        const isEnter = key === 'Enter';\r\n        const isCtrlOrCommand = event.ctrlKey || event.metaKey;\r\n        const isCtrlEnter = isCtrlOrCommand && isEnter;\r\n        // If the key is Ctrl + Enter or Command + Enter, focus the first interactive element\r\n        // and lock focus inside the tree item\r\n        if (isCtrlEnter && target === currentTarget) {\r\n            setIsInsideTreeItem(true);\r\n            const interactiveElementsList = getInteractiveElements(currentTarget, interactiveElements);\r\n            const elementToFocus = interactiveElementsList[0];\r\n            if (elementToFocus) {\r\n                setTimeout(() => {\r\n                    elementToFocus.focus();\r\n                }, 0);\r\n            }\r\n            return;\r\n        }\r\n        // Ensure valid CSS selectors by escaping special characters (e.g., periods in itemId)\r\n        const safeItemId = CSS.escape(itemId);\r\n        const isWithinLabelOrAdditionalContent = target.closest(`#${safeItemId}-label, #${safeItemId}-additionalcontentwrapper`);\r\n        // If the target is within the label or additional content, handle key events for those areas\r\n        if (isWithinLabelOrAdditionalContent) {\r\n            handleLabelAndAdditionalContentKeyDown(event);\r\n            return;\r\n        }\r\n        // If the target is the tree item itself, handle key down for the tree item\r\n        if (target === currentTarget) {\r\n            handleKeyDown(event);\r\n            return;\r\n        }\r\n    };\r\n    return (React.createElement(TreeItemContext.Provider, { value: contextValue },\r\n        React.createElement(\"div\", { style: treeItemStyles },\r\n            React.createElement(StyledTreeItem, Object.assign({}, rest, { \"aria-expanded\": hasOwnTreeItems ? expanded : null, \"aria-selected\": selectedItem, \"aria-checked\": shouldShowCheckbox ? ariaCheckedValue : null, \"data-testid\": testId, depth: itemDepth, hasOwnTreeItems: hasOwnTreeItems, id: itemId, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, role: \"treeitem\", selectableType: selectable, selected: selectedItem, theme: theme, tabIndex: tabIndex, onKeyDown: onKeyDownHandler, ref: treeItemRef, hoverColor: hoverColor }),\r\n                React.createElement(StyledItemWrapper, { \"data-testid\": `${testId ?? itemId}-itemwrapper`, depth: itemDepth, hasAdditionalContent: !!additionalContent, hasCustomIconSize: !!expandIconStyles?.size, id: `${itemId}-itemwrapper`, isDisabled: isDisabled, isInverse: isInverse, nodeType: nodeType, selectable: selectable, style: style, theme: theme, ref: mergeRefs(ref, focusTrapElement), onClick: handleOnClick },\r\n                    hasOwnTreeItems && (React.createElement(StyledExpandWrapper, { \"aria-hidden\": Boolean(!expanded), size: expandIconStyles?.size, color: expandIconStyles?.color, \"data-testid\": `${testId || itemId}-expand`, isDisabled: isDisabled, isInverse: isInverse, onClick: event => {\r\n                            if (!isDisabled) {\r\n                                onExpandedClicked(event);\r\n                            }\r\n                        }, theme: theme }, expanded ? (React.createElement(ExpandMoreIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })) : (React.createElement(ChevronRightIcon, { \"aria-hidden\": true, size: expandIconStyles?.size })))),\r\n                    shouldShowCheckbox ? (React.createElement(StyledCheckboxWrapper, { hasAdditionalContent: !!additionalContent, theme: theme },\r\n                        hasOwnTreeItems ? (React.createElement(IndeterminateCheckbox, Object.assign({}, checkboxProps, { status: checkedStatus }))) : (React.createElement(Checkbox, Object.assign({}, checkboxProps, { checked: checkedStatusToBoolean(checkedStatus) }))),\r\n                        treeItemAdditionalContent)) : (React.createElement(React.Fragment, null,\r\n                        labelText,\r\n                        treeItemAdditionalContent))),\r\n                React.Children.map(children, (child, index) => {\r\n                    return child?.type === TreeItem ? (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" }, React.cloneElement(child, {\r\n                            index,\r\n                            key: child.props.itemId,\r\n                            itemDepth,\r\n                            parentDepth,\r\n                            topLevel: false,\r\n                        })))) : (child);\r\n                })))));\r\n});\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
17086
17111
|
var TreeItem = /*#__PURE__*/React.forwardRef(function (props, forwardedRef) {
|
|
17087
17112
|
var _selectedItems$;
|
|
17088
17113
|
var additionalContent = props.additionalContent,
|
|
17089
17114
|
children = props.children,
|
|
17115
|
+
hoverColor = props.hoverColor,
|
|
17090
17116
|
icon = props.icon,
|
|
17091
17117
|
label = props.label,
|
|
17092
17118
|
labelStyle = props.labelStyle,
|
|
17093
17119
|
style = props.style,
|
|
17094
17120
|
testId = props.testId,
|
|
17095
17121
|
topLevel = props.topLevel,
|
|
17122
|
+
treeItemStyles = props.treeItemStyles,
|
|
17096
17123
|
rest = _objectWithoutPropertiesLoose(props, _excluded$1I);
|
|
17097
17124
|
var theme = React.useContext(ThemeContext);
|
|
17098
17125
|
var isInverse = useIsInverse();
|
|
17099
17126
|
var _React$useContext = React.useContext(TreeViewContext),
|
|
17100
|
-
|
|
17127
|
+
expandIconStyles = _React$useContext.expandIconStyles,
|
|
17128
|
+
handleExpandedChange = _React$useContext.handleExpandedChange,
|
|
17101
17129
|
hasIcons = _React$useContext.hasIcons,
|
|
17102
17130
|
itemToFocus = _React$useContext.itemToFocus,
|
|
17103
|
-
|
|
17104
|
-
|
|
17131
|
+
isTopLevelSelectable = _React$useContext.isTopLevelSelectable,
|
|
17132
|
+
selectable = _React$useContext.selectable;
|
|
17105
17133
|
var _useTreeItem = useTreeItem(props, forwardedRef),
|
|
17106
17134
|
contextValue = _useTreeItem.contextValue,
|
|
17107
17135
|
handleClick = _useTreeItem.handleClick,
|
|
@@ -17299,6 +17327,8 @@ var TreeItem = /*#__PURE__*/React.forwardRef(function (props, forwardedRef) {
|
|
|
17299
17327
|
};
|
|
17300
17328
|
return React.createElement(TreeItemContext.Provider, {
|
|
17301
17329
|
value: contextValue
|
|
17330
|
+
}, React.createElement("div", {
|
|
17331
|
+
style: treeItemStyles
|
|
17302
17332
|
}, React.createElement(StyledTreeItem, Object.assign({}, rest, {
|
|
17303
17333
|
"aria-expanded": hasOwnTreeItems ? expanded : null,
|
|
17304
17334
|
"aria-selected": selectedItem,
|
|
@@ -17316,11 +17346,13 @@ var TreeItem = /*#__PURE__*/React.forwardRef(function (props, forwardedRef) {
|
|
|
17316
17346
|
theme: theme,
|
|
17317
17347
|
tabIndex: tabIndex,
|
|
17318
17348
|
onKeyDown: onKeyDownHandler,
|
|
17319
|
-
ref: treeItemRef
|
|
17349
|
+
ref: treeItemRef,
|
|
17350
|
+
hoverColor: hoverColor
|
|
17320
17351
|
}), React.createElement(StyledItemWrapper, {
|
|
17321
17352
|
"data-testid": (testId != null ? testId : itemId) + "-itemwrapper",
|
|
17322
17353
|
depth: itemDepth,
|
|
17323
17354
|
hasAdditionalContent: !!additionalContent,
|
|
17355
|
+
hasCustomIconSize: !!(expandIconStyles != null && expandIconStyles.size),
|
|
17324
17356
|
id: itemId + "-itemwrapper",
|
|
17325
17357
|
isDisabled: isDisabled,
|
|
17326
17358
|
isInverse: isInverse,
|
|
@@ -17332,6 +17364,8 @@ var TreeItem = /*#__PURE__*/React.forwardRef(function (props, forwardedRef) {
|
|
|
17332
17364
|
onClick: handleOnClick
|
|
17333
17365
|
}, hasOwnTreeItems && React.createElement(StyledExpandWrapper, {
|
|
17334
17366
|
"aria-hidden": Boolean(!expanded),
|
|
17367
|
+
size: expandIconStyles == null ? void 0 : expandIconStyles.size,
|
|
17368
|
+
color: expandIconStyles == null ? void 0 : expandIconStyles.color,
|
|
17335
17369
|
"data-testid": (testId || itemId) + "-expand",
|
|
17336
17370
|
isDisabled: isDisabled,
|
|
17337
17371
|
isInverse: isInverse,
|
|
@@ -17342,9 +17376,11 @@ var TreeItem = /*#__PURE__*/React.forwardRef(function (props, forwardedRef) {
|
|
|
17342
17376
|
},
|
|
17343
17377
|
theme: theme
|
|
17344
17378
|
}, expanded ? React.createElement(reactMagmaIcons.ExpandMoreIcon, {
|
|
17345
|
-
"aria-hidden": true
|
|
17379
|
+
"aria-hidden": true,
|
|
17380
|
+
size: expandIconStyles == null ? void 0 : expandIconStyles.size
|
|
17346
17381
|
}) : React.createElement(reactMagmaIcons.ChevronRightIcon, {
|
|
17347
|
-
"aria-hidden": true
|
|
17382
|
+
"aria-hidden": true,
|
|
17383
|
+
size: expandIconStyles == null ? void 0 : expandIconStyles.size
|
|
17348
17384
|
})), shouldShowCheckbox ? React.createElement(StyledCheckboxWrapper, {
|
|
17349
17385
|
hasAdditionalContent: !!additionalContent,
|
|
17350
17386
|
theme: theme
|
|
@@ -17365,7 +17401,7 @@ var TreeItem = /*#__PURE__*/React.forwardRef(function (props, forwardedRef) {
|
|
|
17365
17401
|
parentDepth: parentDepth,
|
|
17366
17402
|
topLevel: false
|
|
17367
17403
|
}))) : child;
|
|
17368
|
-
})));
|
|
17404
|
+
}))));
|
|
17369
17405
|
});
|
|
17370
17406
|
|
|
17371
17407
|
function useTreeView(props) {
|
|
@@ -17383,7 +17419,8 @@ function useTreeView(props) {
|
|
|
17383
17419
|
apiRef = props.apiRef,
|
|
17384
17420
|
isDisabled = props.isDisabled,
|
|
17385
17421
|
_props$isTopLevelSele = props.isTopLevelSelectable,
|
|
17386
|
-
isTopLevelSelectable = _props$isTopLevelSele === void 0 ? true : _props$isTopLevelSele
|
|
17422
|
+
isTopLevelSelectable = _props$isTopLevelSele === void 0 ? true : _props$isTopLevelSele,
|
|
17423
|
+
expandIconStyles = props.expandIconStyles;
|
|
17387
17424
|
var hasPreselectedItems = Boolean(preselectedItems);
|
|
17388
17425
|
var _React$useState = React.useState(function () {
|
|
17389
17426
|
return getInitialItems({
|
|
@@ -17739,6 +17776,7 @@ function useTreeView(props) {
|
|
|
17739
17776
|
selectItem: selectItem,
|
|
17740
17777
|
handleExpandedChange: handleExpandedChange,
|
|
17741
17778
|
expandedSet: expandedSet,
|
|
17779
|
+
expandIconStyles: expandIconStyles,
|
|
17742
17780
|
isTopLevelSelectable: isTopLevelSelectable
|
|
17743
17781
|
};
|
|
17744
17782
|
return {
|