react-magma-dom 4.11.0-next.17 → 4.11.0-next.19
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/TreeItemHierarchyContext.d.ts +1 -0
- package/dist/components/TreeView/TreeView.d.ts +0 -10
- package/dist/components/TreeView/utils.d.ts +1 -1
- package/dist/esm/index.js +145 -88
- package/dist/esm/index.js.map +1 -1
- package/dist/properties.json +148 -116
- package/dist/react-magma-dom.cjs.development.js +145 -85
- 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
|
@@ -17258,7 +17258,8 @@ var TreeItemHierarchyContext = /*#__PURE__*/React.createContext({
|
|
|
17258
17258
|
depth: 0,
|
|
17259
17259
|
parentDepth: 0,
|
|
17260
17260
|
isTopLevel: true,
|
|
17261
|
-
index: 0
|
|
17261
|
+
index: 0,
|
|
17262
|
+
isVirtualized: false
|
|
17262
17263
|
});
|
|
17263
17264
|
|
|
17264
17265
|
(function (TreeViewSelectable) {
|
|
@@ -17313,7 +17314,7 @@ var TreeViewSelectionContext = /*#__PURE__*/React.createContext({
|
|
|
17313
17314
|
* The label element (the div inside the li) gets additional spacing.
|
|
17314
17315
|
* In order to highlight the entire line, we need to negate the value for margin.
|
|
17315
17316
|
*/
|
|
17316
|
-
function calculateOffset(type, depth, labelElem, negative) {
|
|
17317
|
+
function calculateOffset(type, depth, labelElem, negative, isVirtualized) {
|
|
17317
17318
|
if (depth === void 0) {
|
|
17318
17319
|
depth = 0;
|
|
17319
17320
|
}
|
|
@@ -17323,6 +17324,9 @@ function calculateOffset(type, depth, labelElem, negative) {
|
|
|
17323
17324
|
if (negative === void 0) {
|
|
17324
17325
|
negative = false;
|
|
17325
17326
|
}
|
|
17327
|
+
if (isVirtualized === void 0) {
|
|
17328
|
+
isVirtualized = false;
|
|
17329
|
+
}
|
|
17326
17330
|
var padding = 0;
|
|
17327
17331
|
if (type === exports.TreeNodeType.leaf) {
|
|
17328
17332
|
if (labelElem) {
|
|
@@ -17332,6 +17336,10 @@ function calculateOffset(type, depth, labelElem, negative) {
|
|
|
17332
17336
|
}
|
|
17333
17337
|
} else if (depth === 0) {
|
|
17334
17338
|
padding = 40;
|
|
17339
|
+
} else if (isVirtualized) {
|
|
17340
|
+
// For virtualized items, calculate cumulative padding
|
|
17341
|
+
// Base padding (40px) + (depth * 24px per level after first)
|
|
17342
|
+
padding = 40 + (depth - 1) * 24;
|
|
17335
17343
|
} else {
|
|
17336
17344
|
padding = 56;
|
|
17337
17345
|
}
|
|
@@ -17343,6 +17351,10 @@ function calculateOffset(type, depth, labelElem, negative) {
|
|
|
17343
17351
|
}
|
|
17344
17352
|
} else if (depth === 0) {
|
|
17345
17353
|
padding = 8;
|
|
17354
|
+
} else if (isVirtualized) {
|
|
17355
|
+
// For virtualized items, calculate cumulative padding
|
|
17356
|
+
// Base padding (8px) + (depth * 24px per level after first)
|
|
17357
|
+
padding = 8 + (depth - 1) * 24;
|
|
17346
17358
|
} else {
|
|
17347
17359
|
padding = 24;
|
|
17348
17360
|
}
|
|
@@ -18333,15 +18345,15 @@ var StyledTreeItem = /*#__PURE__*/_styled("li", {
|
|
|
18333
18345
|
}, ";list-style-type:none;cursor:", function (props) {
|
|
18334
18346
|
return getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType);
|
|
18335
18347
|
}, ";position:relative;margin-bottom:0;padding-inline-start:", function (props) {
|
|
18336
|
-
return calculateOffset(props.nodeType, props.depth);
|
|
18348
|
+
return calculateOffset(props.nodeType, props.depth, false, false, props.isVirtualized);
|
|
18337
18349
|
}, ";&:focus{outline:none;&>*:first-child{outline-offset:-2px;outline:2px solid ", function (props) {
|
|
18338
18350
|
return props.isInverse ? props.theme.colors.focusInverse : props.theme.colors.focus;
|
|
18339
18351
|
}, ";}}>div:first-of-type{background:", function (props) {
|
|
18340
18352
|
return props.selected && props.isInverse ? polished.transparentize(0.7, props.theme.colors.neutral900) : props.selected && polished.transparentize(0.92, props.theme.colors.neutral900);
|
|
18341
18353
|
}, ";position:relative;padding-inline-start:", function (props) {
|
|
18342
|
-
return calculateOffset(props.nodeType, props.depth, true);
|
|
18354
|
+
return calculateOffset(props.nodeType, props.depth, true, false, props.isVirtualized);
|
|
18343
18355
|
}, ";margin-inline-start:", function (props) {
|
|
18344
|
-
return calculateOffset(props.nodeType, props.depth, true, true);
|
|
18356
|
+
return calculateOffset(props.nodeType, props.depth, true, true, props.isVirtualized);
|
|
18345
18357
|
}, ";padding-block-end:", function (props) {
|
|
18346
18358
|
return props.theme.spaceScale.spacing02;
|
|
18347
18359
|
}, ";padding-block-start:", function (props) {
|
|
@@ -18349,7 +18361,7 @@ var StyledTreeItem = /*#__PURE__*/_styled("li", {
|
|
|
18349
18361
|
}, ";padding-right:", function (props) {
|
|
18350
18362
|
return props.theme.spaceScale.spacing02;
|
|
18351
18363
|
}, ";", function (props) {
|
|
18352
|
-
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":"AAyDQ","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18364
|
+
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":"AAyDQ","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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, false, false, props.isVirtualized)};\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, false, props.isVirtualized)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true, props.isVirtualized)};\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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, isVirtualized: hierarchyContext.isVirtualized, 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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18353
18365
|
}, " &:hover{background:", function (props) {
|
|
18354
18366
|
return getHoverBackground({
|
|
18355
18367
|
isDisabled: props.isDisabled,
|
|
@@ -18357,7 +18369,7 @@ var StyledTreeItem = /*#__PURE__*/_styled("li", {
|
|
|
18357
18369
|
isInverse: props.isInverse,
|
|
18358
18370
|
theme: props.theme
|
|
18359
18371
|
});
|
|
18360
|
-
}, ";}}" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAoBiC","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18372
|
+
}, ";}}" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAoBiC","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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, false, false, props.isVirtualized)};\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, false, props.isVirtualized)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true, props.isVirtualized)};\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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, isVirtualized: hierarchyContext.isVirtualized, 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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18361
18373
|
function getHoverBackground(_ref) {
|
|
18362
18374
|
var isDisabled = _ref.isDisabled,
|
|
18363
18375
|
hoverColor = _ref.hoverColor,
|
|
@@ -18379,13 +18391,13 @@ var IconWrapper$8 = /*#__PURE__*/_styled("span", {
|
|
|
18379
18391
|
return props.theme.iconSizes.medium;
|
|
18380
18392
|
}, "px;width:", function (props) {
|
|
18381
18393
|
return props.theme.iconSizes.medium;
|
|
18382
|
-
}, "px;vertical-align:middle;}" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAwFgC","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18394
|
+
}, "px;vertical-align:middle;}" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAwFgC","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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, false, false, props.isVirtualized)};\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, false, props.isVirtualized)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true, props.isVirtualized)};\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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, isVirtualized: hierarchyContext.isVirtualized, 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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18383
18395
|
var StyledLabelWrapper = /*#__PURE__*/_styled("span", {
|
|
18384
18396
|
target: "e1xiryew4",
|
|
18385
18397
|
label: "StyledLabelWrapper"
|
|
18386
18398
|
})("display:flex;align-items:flex-start;color:", function (props) {
|
|
18387
18399
|
return getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme);
|
|
18388
|
-
}, ";width:100%;" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAmGuC","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18400
|
+
}, ";width:100%;" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAmGuC","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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, false, false, props.isVirtualized)};\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, false, props.isVirtualized)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true, props.isVirtualized)};\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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, isVirtualized: hierarchyContext.isVirtualized, 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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18389
18401
|
var StyledExpandWrapper = /*#__PURE__*/_styled("div", {
|
|
18390
18402
|
target: "e1xiryew3",
|
|
18391
18403
|
label: "StyledExpandWrapper"
|
|
@@ -18401,7 +18413,7 @@ var StyledExpandWrapper = /*#__PURE__*/_styled("div", {
|
|
|
18401
18413
|
var size = _ref3.size,
|
|
18402
18414
|
theme = _ref3.theme;
|
|
18403
18415
|
return size !== undefined ? size + "px" : theme.spaceScale.spacing06;
|
|
18404
|
-
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAyGuC","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18416
|
+
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAyGuC","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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, false, false, props.isVirtualized)};\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, false, props.isVirtualized)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true, props.isVirtualized)};\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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, isVirtualized: hierarchyContext.isVirtualized, 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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18405
18417
|
var StyledCheckboxWrapper = /*#__PURE__*/_styled("div", {
|
|
18406
18418
|
target: "e1xiryew2",
|
|
18407
18419
|
label: "StyledCheckboxWrapper"
|
|
@@ -18411,7 +18423,7 @@ var StyledCheckboxWrapper = /*#__PURE__*/_styled("div", {
|
|
|
18411
18423
|
return props.hasAdditionalContent ? 'flex' : 'inline-flex';
|
|
18412
18424
|
}, ";flex-direction:column;width:", function (props) {
|
|
18413
18425
|
return "calc(100% - " + props.theme.spaceScale.spacing03 + ")";
|
|
18414
|
-
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAmHyC","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18426
|
+
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAmHyC","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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, false, false, props.isVirtualized)};\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, false, props.isVirtualized)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true, props.isVirtualized)};\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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, isVirtualized: hierarchyContext.isVirtualized, 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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18415
18427
|
var StyledItemWrapper = /*#__PURE__*/_styled("div", {
|
|
18416
18428
|
target: "e1xiryew1",
|
|
18417
18429
|
label: "StyledItemWrapper"
|
|
@@ -18421,13 +18433,13 @@ var StyledItemWrapper = /*#__PURE__*/_styled("div", {
|
|
|
18421
18433
|
return props.hasCustomIconSize ? 'center' : 'flex-start';
|
|
18422
18434
|
}, ";cursor:", function (props) {
|
|
18423
18435
|
return getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType);
|
|
18424
|
-
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AA0HqC","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18436
|
+
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AA0HqC","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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, false, false, props.isVirtualized)};\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, false, props.isVirtualized)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true, props.isVirtualized)};\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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, isVirtualized: hierarchyContext.isVirtualized, 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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18425
18437
|
var AdditionalContentWrapper$1 = /*#__PURE__*/_styled("div", {
|
|
18426
18438
|
target: "e1xiryew0",
|
|
18427
18439
|
label: "AdditionalContentWrapper"
|
|
18428
18440
|
})("margin-bottom:", function (props) {
|
|
18429
18441
|
return props.theme.spaceScale.spacing05;
|
|
18430
|
-
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAgI4C","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18442
|
+
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeItem.tsx"],"names":[],"mappings":"AAgI4C","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 { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { checkedStatusToBoolean, useTreeItem, } from './useTreeItem';\r\nimport { calculateOffset, getTreeItemLabelColor, getTreeItemWrapperCursor, TreeNodeType, } from './utils';\r\nimport { useFocusLock } from '../../hooks/useFocusLock';\r\nimport { useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nimport { mergeRefs } from '../../utils';\r\nimport { Checkbox } from '../Checkbox';\r\nimport { IndeterminateCheckbox, IndeterminateCheckboxStatus, } from '../IndeterminateCheckbox';\r\nimport { Transition } from '../Transition';\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, false, false, props.isVirtualized)};\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, false, props.isVirtualized)};\n    margin-inline-start: ${props => calculateOffset(props.nodeType, props.depth, true, true, props.isVirtualized)};\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\nconst TreeItemComponent = React.forwardRef((props, forwardedRef) => {\r\n    const { additionalContent, children, hoverColor, icon, index: indexProp, label, labelStyle, style, testId, topLevel: topLevelProp, treeItemStyles, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse();\r\n    // Read hierarchy information from context (reduces cloneElement overhead)\r\n    const hierarchyContext = React.useContext(TreeItemHierarchyContext);\r\n    // Use context values if props are not provided (for backward compatibility)\r\n    const index = indexProp !== undefined ? indexProp : hierarchyContext.index;\r\n    const topLevel = topLevelProp !== undefined ? topLevelProp : hierarchyContext.isTopLevel;\r\n    // Consume split contexts for reduced re-render scope\r\n    const { itemToFocus } = React.useContext(TreeViewSelectionContext);\r\n    const { handleExpandedChange } = React.useContext(TreeViewExpansionContext);\r\n    const { expandIconStyles, hasIcons, isTopLevelSelectable, selectable } = React.useContext(TreeViewConfigContext);\r\n    // Pass the resolved values to useTreeItem\r\n    const propsWithHierarchy = {\r\n        ...props,\r\n        index,\r\n        topLevel,\r\n        itemDepth: hierarchyContext.depth,\r\n        parentDepth: hierarchyContext.parentDepth,\r\n    };\r\n    const { contextValue, handleClick, handleKeyDown } = useTreeItem(propsWithHierarchy, forwardedRef);\r\n    const { isDisabled } = contextValue;\r\n    const { checkboxChangeHandler, checkedStatus, expanded, hasOwnTreeItems, itemDepth, itemId, 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 = React.useCallback((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 = React.useCallback((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    }, [getInteractiveElements, interactiveElements, isInsideTreeItem]);\r\n    const handleOnClick = React.useCallback((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    }, [isDisabled, selectable, handleClick, itemId]);\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    // Memoize inline style objects to prevent unnecessary re-renders\r\n    const checkboxInputStyle = React.useMemo(() => ({ marginRight: theme.spaceScale.spacing03 }), [theme.spaceScale.spacing03]);\r\n    const checkboxLabelStyle = React.useMemo(() => ({\r\n        padding: 0,\r\n        width: '100%',\r\n    }), []);\r\n    // Props shared by Checkbox and IndeterminateCheckbox\r\n    const checkboxProps = React.useMemo(() => ({\r\n        disabled: isDisabled,\r\n        hideFocus: true,\r\n        id: `${itemId}-checkbox`,\r\n        inputStyle: checkboxInputStyle,\r\n        labelStyle: checkboxLabelStyle,\r\n        labelText: labelText,\r\n        onChange: checkboxChangeHandler,\r\n        tabIndex: -1,\r\n        testId: `${itemId}-checkbox`,\r\n    }), [\r\n        isDisabled,\r\n        itemId,\r\n        checkboxInputStyle,\r\n        checkboxLabelStyle,\r\n        labelText,\r\n        checkboxChangeHandler,\r\n    ]);\r\n    const onExpandedClicked = React.useCallback((event) => {\r\n        event.preventDefault();\r\n        handleExpandedChange(event, itemId);\r\n    }, [handleExpandedChange, itemId]);\r\n    const handleExpandClick = React.useCallback((event) => {\r\n        if (!isDisabled) {\r\n            onExpandedClicked(event);\r\n        }\r\n    }, [isDisabled, onExpandedClicked]);\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 = React.useCallback((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        getInteractiveElements,\r\n        interactiveElements,\r\n        itemId,\r\n        handleLabelAndAdditionalContentKeyDown,\r\n        handleKeyDown,\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, isVirtualized: hierarchyContext.isVirtualized, 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: handleExpandClick, 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, childIndex) => {\r\n                    if (child?.type !== TreeItem) {\r\n                        return child;\r\n                    }\r\n                    // Pass hierarchy info through context instead of cloneElement\r\n                    const nestedHierarchyValue = {\r\n                        depth: itemDepth + 1,\r\n                        parentDepth: itemDepth,\r\n                        isTopLevel: false,\r\n                        index: childIndex,\r\n                    };\r\n                    return (React.createElement(Transition, { isOpen: expanded, unmountOnExit: true },\r\n                        React.createElement(\"ul\", { role: \"group\" },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { key: child.props.itemId, value: nestedHierarchyValue }, child))));\r\n                })))));\r\n});\r\n/**\r\n * Custom comparison function for React.memo\r\n * Only re-render TreeItem when relevant props change\r\n */\r\nfunction arePropsEqual(prevProps, nextProps) {\r\n    // Check if itemId changed\r\n    if (prevProps.itemId !== nextProps.itemId) {\r\n        return false;\r\n    }\r\n    // Check if label changed\r\n    if (prevProps.label !== nextProps.label) {\r\n        return false;\r\n    }\r\n    // Check if disabled state changed\r\n    if (prevProps.isDisabled !== nextProps.isDisabled) {\r\n        return false;\r\n    }\r\n    // Check if icon changed\r\n    if (prevProps.icon !== nextProps.icon) {\r\n        return false;\r\n    }\r\n    // Check if additional content changed\r\n    if (prevProps.additionalContent !== nextProps.additionalContent) {\r\n        return false;\r\n    }\r\n    // Check if hover color changed\r\n    if (prevProps.hoverColor !== nextProps.hoverColor) {\r\n        return false;\r\n    }\r\n    // Check if children changed (for nested tree items)\r\n    // Using type assertion since children comes from React.HTMLAttributes\r\n    if (prevProps.children !== nextProps.children) {\r\n        return false;\r\n    }\r\n    // Check if styles changed\r\n    // Using type assertion since style comes from React.HTMLAttributes\r\n    if (prevProps.style !== nextProps.style) {\r\n        return false;\r\n    }\r\n    if (prevProps.labelStyle !== nextProps.labelStyle) {\r\n        return false;\r\n    }\r\n    if (prevProps.treeItemStyles !== nextProps.treeItemStyles) {\r\n        return false;\r\n    }\r\n    // All relevant props are equal, skip re-render\r\n    return true;\r\n}\r\n/**\r\n * Memoized TreeItem component\r\n * Only re-renders when relevant props change, improving performance for large trees\r\n */\r\nexport const TreeItem = React.memo(TreeItemComponent, arePropsEqual);\r\n//# sourceMappingURL=TreeItem.js.map"]} */"));
|
|
18431
18443
|
var TreeItemComponent = /*#__PURE__*/React.forwardRef(function (props, forwardedRef) {
|
|
18432
18444
|
var _selectedItems$;
|
|
18433
18445
|
var additionalContent = props.additionalContent,
|
|
@@ -18688,6 +18700,7 @@ var TreeItemComponent = /*#__PURE__*/React.forwardRef(function (props, forwarded
|
|
|
18688
18700
|
id: itemId,
|
|
18689
18701
|
isDisabled: isDisabled,
|
|
18690
18702
|
isInverse: isInverse,
|
|
18703
|
+
isVirtualized: hierarchyContext.isVirtualized,
|
|
18691
18704
|
nodeType: nodeType,
|
|
18692
18705
|
role: "treeitem",
|
|
18693
18706
|
selectableType: selectable,
|
|
@@ -18965,17 +18978,21 @@ function treeViewReducer(state, action) {
|
|
|
18965
18978
|
if (item.itemId === newItem.parentId) {
|
|
18966
18979
|
item.hasOwnTreeItems = true;
|
|
18967
18980
|
if (_checkParents3) {
|
|
18968
|
-
|
|
18981
|
+
var allChildren = [].concat(state.items, [newItem]).filter(function (child) {
|
|
18982
|
+
return child.parentId === item.itemId;
|
|
18983
|
+
});
|
|
18984
|
+
var checkedChildren = allChildren.filter(function (child) {
|
|
18985
|
+
return child.checkedStatus === exports.IndeterminateCheckboxStatus.checked;
|
|
18986
|
+
});
|
|
18987
|
+
var uncheckedChildren = allChildren.filter(function (child) {
|
|
18988
|
+
return child.checkedStatus === exports.IndeterminateCheckboxStatus.unchecked;
|
|
18989
|
+
});
|
|
18990
|
+
if (checkedChildren.length === allChildren.length) {
|
|
18991
|
+
item.checkedStatus = exports.IndeterminateCheckboxStatus.checked;
|
|
18992
|
+
} else if (uncheckedChildren.length === allChildren.length) {
|
|
18993
|
+
item.checkedStatus = exports.IndeterminateCheckboxStatus.unchecked;
|
|
18994
|
+
} else {
|
|
18969
18995
|
item.checkedStatus = exports.IndeterminateCheckboxStatus.indeterminate;
|
|
18970
|
-
} else if (item.checkedStatus === exports.IndeterminateCheckboxStatus.indeterminate && newItem.checkedStatus === exports.IndeterminateCheckboxStatus.checked) {
|
|
18971
|
-
var allChildrenChecked = [].concat(state.items, [newItem]).filter(function (child) {
|
|
18972
|
-
return child.parentId === item.itemId;
|
|
18973
|
-
}).every(function (child) {
|
|
18974
|
-
return child.checkedStatus === exports.IndeterminateCheckboxStatus.checked;
|
|
18975
|
-
});
|
|
18976
|
-
if (allChildrenChecked) {
|
|
18977
|
-
item.checkedStatus = exports.IndeterminateCheckboxStatus.checked;
|
|
18978
|
-
}
|
|
18979
18996
|
}
|
|
18980
18997
|
}
|
|
18981
18998
|
}
|
|
@@ -19323,27 +19340,7 @@ function useTreeView(props) {
|
|
|
19323
19340
|
selectable: selectable
|
|
19324
19341
|
}
|
|
19325
19342
|
});
|
|
19326
|
-
|
|
19327
|
-
if (newItem.parentId) {
|
|
19328
|
-
// We need to get the updated items after ADD_ITEM dispatch
|
|
19329
|
-
// This is a limitation of the current approach - we'll handle it in the next effect
|
|
19330
|
-
dispatch({
|
|
19331
|
-
type: 'SET_ITEMS',
|
|
19332
|
-
payload: {
|
|
19333
|
-
items: getInitialItems({
|
|
19334
|
-
children: children,
|
|
19335
|
-
preselectedItems: selectedItems,
|
|
19336
|
-
checkParents: checkParents,
|
|
19337
|
-
checkChildren: false,
|
|
19338
|
-
selectable: selectable,
|
|
19339
|
-
isDisabled: isDisabled,
|
|
19340
|
-
isTopLevelSelectable: isTopLevelSelectable,
|
|
19341
|
-
items: [].concat(items, [newItem])
|
|
19342
|
-
})
|
|
19343
|
-
}
|
|
19344
|
-
});
|
|
19345
|
-
}
|
|
19346
|
-
}, [checkParents, children, isDisabled, isTopLevelSelectable, items, selectable, selectedItems]);
|
|
19343
|
+
}, [checkParents, selectable]);
|
|
19347
19344
|
React.useEffect(function () {
|
|
19348
19345
|
if (apiRef) {
|
|
19349
19346
|
apiRef.current = {
|
|
@@ -19433,8 +19430,7 @@ function useTreeView(props) {
|
|
|
19433
19430
|
};
|
|
19434
19431
|
}
|
|
19435
19432
|
|
|
19436
|
-
var _excluded$1M = ["ariaLabel", "ariaLabelledBy", "children", "isInverse", "onExpandedChange", "onSelectedItemChange", "selectable", "testId", "apiRef", "enableVirtualization"
|
|
19437
|
-
function _EMOTION_STRINGIFIED_CSS_ERROR__$H() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
|
|
19433
|
+
var _excluded$1M = ["ariaLabel", "ariaLabelledBy", "children", "isInverse", "onExpandedChange", "onSelectedItemChange", "selectable", "testId", "apiRef", "enableVirtualization"];
|
|
19438
19434
|
var StyledTreeView = /*#__PURE__*/_styled("ul", {
|
|
19439
19435
|
target: "e1tyeayj2",
|
|
19440
19436
|
label: "StyledTreeView"
|
|
@@ -19442,23 +19438,21 @@ var StyledTreeView = /*#__PURE__*/_styled("ul", {
|
|
|
19442
19438
|
return props.isInverse ? props.theme.colors.neutral100 : props.theme.colors.neutral;
|
|
19443
19439
|
}, ";position:", function (props) {
|
|
19444
19440
|
return props.isVirtualized ? 'relative' : 'static';
|
|
19445
|
-
}, ";ul{padding:0;margin:0;li{margin:0;}}" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeView.tsx"],"names":[],"mappings":"AAaiC","file":"TreeView.tsx","sourcesContent":["import * as React from 'react';\r\nimport styled from '@emotion/styled';\r\nimport { useVirtual } from 'react-virtual';\r\nimport { TreeItem } from './TreeItem';\r\nimport { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { useTreeItem } from './useTreeItem';\r\nimport { useTreeView } from './useTreeView';\r\nimport { InverseContext, useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nconst StyledTreeView = styled.ul `\n  padding: 0;\n  margin: 0;\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral};\n  position: ${props => (props.isVirtualized ? 'relative' : 'static')};\n  ul {\n    padding: 0;\n    margin: 0;\n    li {\n      margin: 0;\n    }\n  }\n`;\r\nconst VirtualContainer = styled.div `\n  width: 100%;\n  position: relative;\n`;\r\nconst VirtualItem = styled.div `\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  height: ${props => props.height}px;\n  transform: ${props => props.transform};\n`;\r\nexport const TreeView = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, ariaLabelledBy, children, isInverse: isInverseProp, onExpandedChange, onSelectedItemChange, selectable, testId, apiRef, enableVirtualization = false, estimateSize = 40, overscan = 5, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse(isInverseProp);\r\n    const { selectionContextValue, expansionContextValue, configContextValue } = useTreeView(props);\r\n    useTreeItem({ label: ariaLabel, itemId: '' }, ref);\r\n    const inverseContextValue = React.useMemo(() => ({ isInverse }), [isInverse]);\r\n    const parentRef = React.useRef(null);\r\n    // Flatten tree structure for virtualization\r\n    const flattenedItems = React.useMemo(() => {\r\n        if (!enableVirtualization)\r\n            return [];\r\n        const items = [];\r\n        const flatten = (childrenToFlatten, depth = 0, parentIndex = 0) => {\r\n            React.Children.forEach(childrenToFlatten, (child, index) => {\r\n                if (!React.isValidElement(child) || child.type !== TreeItem) {\r\n                    return;\r\n                }\r\n                const itemKey = `tree-item-${depth}-${index}-${items.length}`;\r\n                // Clone the child without its children to prevent double rendering\r\n                const childWithoutNested = React.cloneElement(child, {\r\n                    ...child.props,\r\n                    children: null,\r\n                });\r\n                items.push({\r\n                    child: childWithoutNested,\r\n                    depth,\r\n                    index,\r\n                    key: itemKey,\r\n                });\r\n                // Check if item has children and is expanded\r\n                const itemId = child.props.itemId;\r\n                const isExpanded = expansionContextValue.expandedSet?.has(itemId);\r\n                if (isExpanded && child.props.children) {\r\n                    flatten(child.props.children, depth + 1, index);\r\n                }\r\n            });\r\n        };\r\n        flatten(children);\r\n        return items;\r\n    }, [children, enableVirtualization, expansionContextValue.expandedSet]);\r\n    // Setup virtualizer\r\n    const rowVirtualizer = useVirtual({\r\n        size: flattenedItems.length,\r\n        parentRef,\r\n        estimateSize: React.useCallback(() => estimateSize, [estimateSize]),\r\n        overscan,\r\n    });\r\n    // Process children without cloneElement - use context instead\r\n    const processedChildren = React.useMemo(() => {\r\n        if (enableVirtualization) {\r\n            return null; // Handled by virtualizer below\r\n        }\r\n        let treeItemIndex = 0;\r\n        return React.Children.map(children, child => {\r\n            if (!React.isValidElement(child)) {\r\n                return null;\r\n            }\r\n            if (child.type === TreeItem) {\r\n                const currentIndex = treeItemIndex++;\r\n                const hierarchyValue = {\r\n                    depth: 0,\r\n                    parentDepth: 0,\r\n                    isTopLevel: true,\r\n                    index: currentIndex,\r\n                };\r\n                // Wrap in context provider instead of cloning\r\n                return (React.createElement(TreeItemHierarchyContext.Provider, { key: `tree-item-${currentIndex}`, value: hierarchyValue }, child));\r\n            }\r\n            return null;\r\n        });\r\n    }, [children, enableVirtualization]);\r\n    const virtualItems = enableVirtualization\r\n        ? rowVirtualizer.virtualItems\r\n        : [];\r\n    return (React.createElement(InverseContext.Provider, { value: inverseContextValue },\r\n        React.createElement(TreeViewSelectionContext.Provider, { value: selectionContextValue },\r\n            React.createElement(TreeViewExpansionContext.Provider, { value: expansionContextValue },\r\n                React.createElement(TreeViewConfigContext.Provider, { value: configContextValue },\r\n                    React.createElement(StyledTreeView, Object.assign({}, rest, { \"aria-label\": ariaLabel, \"aria-labelledby\": ariaLabelledBy, \"aria-multiselectable\": selectable === TreeViewSelectable.multi, \"data-testid\": testId, isInverse: isInverse, isVirtualized: enableVirtualization, ref: mergedRefs => {\r\n                            if (typeof ref === 'function') {\r\n                                ref(mergedRefs);\r\n                            }\r\n                            else if (ref) {\r\n                                ref.current =\r\n                                    mergedRefs;\r\n                            }\r\n                            parentRef.current = mergedRefs;\r\n                        }, role: \"tree\", theme: theme, style: {\r\n                            ...rest.style,\r\n                            ...(enableVirtualization\r\n                                ? { height: '400px', overflow: 'auto' }\r\n                                : {}),\r\n                        } }), enableVirtualization ? (React.createElement(VirtualContainer, { style: {\r\n                            height: `${rowVirtualizer.totalSize}px`,\r\n                        } }, virtualItems.map(virtualItem => {\r\n                        const item = flattenedItems[virtualItem.index];\r\n                        const hierarchyValue = {\r\n                            depth: item.depth,\r\n                            parentDepth: Math.max(0, item.depth - 1),\r\n                            isTopLevel: item.depth === 0,\r\n                            index: item.index,\r\n                        };\r\n                        return (React.createElement(VirtualItem, { key: item.key, height: virtualItem.size, transform: `translateY(${virtualItem.start}px)` },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { value: hierarchyValue }, item.child)));\r\n                    }))) : (processedChildren)))))));\r\n});\r\n//# sourceMappingURL=TreeView.js.map"]} */"));
|
|
19441
|
+
}, ";ul{padding:0;margin:0;li{margin:0;}}" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeView.tsx"],"names":[],"mappings":"AAaiC","file":"TreeView.tsx","sourcesContent":["import * as React from 'react';\r\nimport styled from '@emotion/styled';\r\nimport { useVirtual } from 'react-virtual';\r\nimport { TreeItem } from './TreeItem';\r\nimport { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { useTreeItem } from './useTreeItem';\r\nimport { useTreeView } from './useTreeView';\r\nimport { InverseContext, useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nconst StyledTreeView = styled.ul `\n  padding: 0;\n  margin: 0;\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral};\n  position: ${props => (props.isVirtualized ? 'relative' : 'static')};\n  ul {\n    padding: 0;\n    margin: 0;\n    li {\n      margin: 0;\n    }\n  }\n`;\r\nconst VirtualContainer = styled.div `\n  width: 100%;\n  position: relative;\n  height: ${props => props.height}px;\n`;\r\nconst VirtualItem = styled.div `\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-height: ${props => props.height}px;\n  transform: ${props => `translateY(${props.transform}px)`};\n\n  /* Ensure dropdowns and other floating elements appear above items */\n  &:focus-within {\n    z-index: 1;\n  }\n  &[data-testid='popoverContent'] {\n    z-index: 1;\n  }\n`;\r\nexport const TreeView = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, ariaLabelledBy, children, isInverse: isInverseProp, onExpandedChange, onSelectedItemChange, selectable, testId, apiRef, enableVirtualization = false, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse(isInverseProp);\r\n    const { selectionContextValue, expansionContextValue, configContextValue } = useTreeView(props);\r\n    useTreeItem({ label: ariaLabel, itemId: '' }, ref);\r\n    const inverseContextValue = React.useMemo(() => ({ isInverse }), [isInverse]);\r\n    const parentRef = React.useRef(null);\r\n    const itemHeightsByIdRef = React.useRef(new Map());\r\n    const measurementRefs = React.useRef(new Map());\r\n    // Flatten tree structure for virtualization\r\n    const flattenedItems = React.useMemo(() => {\r\n        if (!enableVirtualization)\r\n            return [];\r\n        const items = [];\r\n        const flatten = (childrenToFlatten, depth = 0) => {\r\n            React.Children.forEach(childrenToFlatten, (child, index) => {\r\n                if (!React.isValidElement(child) || child.type !== TreeItem) {\r\n                    return;\r\n                }\r\n                const itemId = child.props.itemId;\r\n                const itemKey = `${itemId}-${depth}`;\r\n                // Clone the child without its children to prevent double rendering\r\n                const childWithoutNested = React.cloneElement(child, {\r\n                    ...child.props,\r\n                    children: null,\r\n                });\r\n                items.push({\r\n                    child: childWithoutNested,\r\n                    depth,\r\n                    index,\r\n                    key: itemKey,\r\n                    itemId,\r\n                });\r\n                // Check if item has children and is expanded\r\n                const isExpanded = expansionContextValue.expandedSet?.has(itemId);\r\n                if (isExpanded && child.props.children) {\r\n                    flatten(child.props.children, depth + 1);\r\n                }\r\n            });\r\n        };\r\n        flatten(children);\r\n        return items;\r\n    }, [children, enableVirtualization, expansionContextValue.expandedSet]);\r\n    const rowVirtualizer = useVirtual({\r\n        size: flattenedItems.length,\r\n        parentRef,\r\n        estimateSize: React.useCallback((index) => {\r\n            const itemId = flattenedItems[index]?.itemId;\r\n            if (!itemId) {\r\n                return 32;\r\n            }\r\n            return itemHeightsByIdRef.current.get(itemId) ?? 32;\r\n        }, [flattenedItems]),\r\n        overscan: 6,\r\n    });\r\n    // Measure item heights after render\r\n    React.useEffect(() => {\r\n        if (!enableVirtualization) {\r\n            measurementRefs.current.clear();\r\n            itemHeightsByIdRef.current.clear();\r\n            return;\r\n        }\r\n        const measureHeights = () => {\r\n            let hasChanges = false;\r\n            measurementRefs.current.forEach((element, itemId) => {\r\n                if (element) {\r\n                    const height = element.offsetHeight;\r\n                    const currentHeight = itemHeightsByIdRef.current.get(itemId);\r\n                    if (currentHeight !== height && height > 0) {\r\n                        itemHeightsByIdRef.current.set(itemId, height);\r\n                        hasChanges = true;\r\n                    }\r\n                }\r\n            });\r\n            if (hasChanges) {\r\n                rowVirtualizer.measure();\r\n            }\r\n        };\r\n        const timeoutId = setTimeout(() => {\r\n            measureHeights();\r\n        }, 0);\r\n        if (typeof window !== 'undefined' && 'ResizeObserver' in window) {\r\n            const resizeObserver = new window.ResizeObserver(() => {\r\n                measureHeights();\r\n            });\r\n            measurementRefs.current.forEach(element => {\r\n                if (element) {\r\n                    resizeObserver.observe(element);\r\n                }\r\n            });\r\n            return () => {\r\n                clearTimeout(timeoutId);\r\n                resizeObserver.disconnect();\r\n            };\r\n        }\r\n        return () => {\r\n            clearTimeout(timeoutId);\r\n        };\r\n    }, [enableVirtualization, flattenedItems, rowVirtualizer]);\r\n    const processedChildren = React.useMemo(() => {\r\n        let treeItemIndex = 0;\r\n        return React.Children.map(children, child => {\r\n            if (!React.isValidElement(child)) {\r\n                return null;\r\n            }\r\n            if (child.type === TreeItem) {\r\n                const currentIndex = treeItemIndex++;\r\n                const hierarchyValue = {\r\n                    depth: 0,\r\n                    parentDepth: 0,\r\n                    isTopLevel: true,\r\n                    index: currentIndex,\r\n                };\r\n                // Wrap in context provider instead of cloning\r\n                return (React.createElement(TreeItemHierarchyContext.Provider, { key: `tree-item-${currentIndex}`, value: hierarchyValue }, child));\r\n            }\r\n            return null;\r\n        });\r\n    }, [children]);\r\n    React.useEffect(() => {\r\n        // Force measurement on children change\r\n        const timeoutId = setTimeout(() => {\r\n            let hasChanges = false;\r\n            measurementRefs.current.forEach((element, itemId) => {\r\n                if (element) {\r\n                    const height = element.offsetHeight;\r\n                    const currentHeight = itemHeightsByIdRef.current.get(itemId);\r\n                    if (currentHeight !== height && height > 0) {\r\n                        itemHeightsByIdRef.current.set(itemId, height);\r\n                        hasChanges = true;\r\n                    }\r\n                }\r\n            });\r\n            if (hasChanges) {\r\n                rowVirtualizer.measure();\r\n            }\r\n        }, 100); // Small delay to ensure DOM is updated\r\n        return () => clearTimeout(timeoutId);\r\n    }, [children, rowVirtualizer]);\r\n    return (React.createElement(InverseContext.Provider, { value: inverseContextValue },\r\n        React.createElement(TreeViewSelectionContext.Provider, { value: selectionContextValue },\r\n            React.createElement(TreeViewExpansionContext.Provider, { value: expansionContextValue },\r\n                React.createElement(TreeViewConfigContext.Provider, { value: configContextValue },\r\n                    React.createElement(StyledTreeView, Object.assign({}, rest, { \"aria-label\": ariaLabel, \"aria-labelledby\": ariaLabelledBy, \"aria-multiselectable\": selectable === TreeViewSelectable.multi, \"data-testid\": testId, isInverse: isInverse, isVirtualized: enableVirtualization, ref: mergedRefs => {\r\n                            if (typeof ref === 'function') {\r\n                                ref(mergedRefs);\r\n                            }\r\n                            else if (ref) {\r\n                                ref.current =\r\n                                    mergedRefs;\r\n                            }\r\n                            parentRef.current = mergedRefs;\r\n                        }, role: \"tree\", theme: theme }), enableVirtualization ? (React.createElement(VirtualContainer, { height: rowVirtualizer.totalSize }, rowVirtualizer.virtualItems.map(virtualItem => {\r\n                        const item = flattenedItems[virtualItem.index];\r\n                        const hierarchyValue = {\r\n                            depth: item.depth,\r\n                            parentDepth: Math.max(0, item.depth - 1),\r\n                            isTopLevel: item.depth === 0,\r\n                            index: item.index,\r\n                            isVirtualized: true,\r\n                        };\r\n                        return (React.createElement(VirtualItem, { key: item.key, height: virtualItem.size, transform: virtualItem.start, ref: (el) => {\r\n                                if (el) {\r\n                                    measurementRefs.current.set(item.itemId, el);\r\n                                }\r\n                                else {\r\n                                    measurementRefs.current.delete(item.itemId);\r\n                                }\r\n                            } },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { value: hierarchyValue }, item.child)));\r\n                    }))) : (processedChildren)))))));\r\n});\r\n//# sourceMappingURL=TreeView.js.map"]} */"));
|
|
19446
19442
|
var VirtualContainer = /*#__PURE__*/_styled("div", {
|
|
19447
19443
|
target: "e1tyeayj1",
|
|
19448
19444
|
label: "VirtualContainer"
|
|
19449
|
-
})( {
|
|
19450
|
-
|
|
19451
|
-
styles: "width:100%;position:relative/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeView.tsx"],"names":[],"mappings":"AA4BoC","file":"TreeView.tsx","sourcesContent":["import * as React from 'react';\r\nimport styled from '@emotion/styled';\r\nimport { useVirtual } from 'react-virtual';\r\nimport { TreeItem } from './TreeItem';\r\nimport { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { useTreeItem } from './useTreeItem';\r\nimport { useTreeView } from './useTreeView';\r\nimport { InverseContext, useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nconst StyledTreeView = styled.ul `\n  padding: 0;\n  margin: 0;\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral};\n  position: ${props => (props.isVirtualized ? 'relative' : 'static')};\n  ul {\n    padding: 0;\n    margin: 0;\n    li {\n      margin: 0;\n    }\n  }\n`;\r\nconst VirtualContainer = styled.div `\n  width: 100%;\n  position: relative;\n`;\r\nconst VirtualItem = styled.div `\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  height: ${props => props.height}px;\n  transform: ${props => props.transform};\n`;\r\nexport const TreeView = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, ariaLabelledBy, children, isInverse: isInverseProp, onExpandedChange, onSelectedItemChange, selectable, testId, apiRef, enableVirtualization = false, estimateSize = 40, overscan = 5, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse(isInverseProp);\r\n    const { selectionContextValue, expansionContextValue, configContextValue } = useTreeView(props);\r\n    useTreeItem({ label: ariaLabel, itemId: '' }, ref);\r\n    const inverseContextValue = React.useMemo(() => ({ isInverse }), [isInverse]);\r\n    const parentRef = React.useRef(null);\r\n    // Flatten tree structure for virtualization\r\n    const flattenedItems = React.useMemo(() => {\r\n        if (!enableVirtualization)\r\n            return [];\r\n        const items = [];\r\n        const flatten = (childrenToFlatten, depth = 0, parentIndex = 0) => {\r\n            React.Children.forEach(childrenToFlatten, (child, index) => {\r\n                if (!React.isValidElement(child) || child.type !== TreeItem) {\r\n                    return;\r\n                }\r\n                const itemKey = `tree-item-${depth}-${index}-${items.length}`;\r\n                // Clone the child without its children to prevent double rendering\r\n                const childWithoutNested = React.cloneElement(child, {\r\n                    ...child.props,\r\n                    children: null,\r\n                });\r\n                items.push({\r\n                    child: childWithoutNested,\r\n                    depth,\r\n                    index,\r\n                    key: itemKey,\r\n                });\r\n                // Check if item has children and is expanded\r\n                const itemId = child.props.itemId;\r\n                const isExpanded = expansionContextValue.expandedSet?.has(itemId);\r\n                if (isExpanded && child.props.children) {\r\n                    flatten(child.props.children, depth + 1, index);\r\n                }\r\n            });\r\n        };\r\n        flatten(children);\r\n        return items;\r\n    }, [children, enableVirtualization, expansionContextValue.expandedSet]);\r\n    // Setup virtualizer\r\n    const rowVirtualizer = useVirtual({\r\n        size: flattenedItems.length,\r\n        parentRef,\r\n        estimateSize: React.useCallback(() => estimateSize, [estimateSize]),\r\n        overscan,\r\n    });\r\n    // Process children without cloneElement - use context instead\r\n    const processedChildren = React.useMemo(() => {\r\n        if (enableVirtualization) {\r\n            return null; // Handled by virtualizer below\r\n        }\r\n        let treeItemIndex = 0;\r\n        return React.Children.map(children, child => {\r\n            if (!React.isValidElement(child)) {\r\n                return null;\r\n            }\r\n            if (child.type === TreeItem) {\r\n                const currentIndex = treeItemIndex++;\r\n                const hierarchyValue = {\r\n                    depth: 0,\r\n                    parentDepth: 0,\r\n                    isTopLevel: true,\r\n                    index: currentIndex,\r\n                };\r\n                // Wrap in context provider instead of cloning\r\n                return (React.createElement(TreeItemHierarchyContext.Provider, { key: `tree-item-${currentIndex}`, value: hierarchyValue }, child));\r\n            }\r\n            return null;\r\n        });\r\n    }, [children, enableVirtualization]);\r\n    const virtualItems = enableVirtualization\r\n        ? rowVirtualizer.virtualItems\r\n        : [];\r\n    return (React.createElement(InverseContext.Provider, { value: inverseContextValue },\r\n        React.createElement(TreeViewSelectionContext.Provider, { value: selectionContextValue },\r\n            React.createElement(TreeViewExpansionContext.Provider, { value: expansionContextValue },\r\n                React.createElement(TreeViewConfigContext.Provider, { value: configContextValue },\r\n                    React.createElement(StyledTreeView, Object.assign({}, rest, { \"aria-label\": ariaLabel, \"aria-labelledby\": ariaLabelledBy, \"aria-multiselectable\": selectable === TreeViewSelectable.multi, \"data-testid\": testId, isInverse: isInverse, isVirtualized: enableVirtualization, ref: mergedRefs => {\r\n                            if (typeof ref === 'function') {\r\n                                ref(mergedRefs);\r\n                            }\r\n                            else if (ref) {\r\n                                ref.current =\r\n                                    mergedRefs;\r\n                            }\r\n                            parentRef.current = mergedRefs;\r\n                        }, role: \"tree\", theme: theme, style: {\r\n                            ...rest.style,\r\n                            ...(enableVirtualization\r\n                                ? { height: '400px', overflow: 'auto' }\r\n                                : {}),\r\n                        } }), enableVirtualization ? (React.createElement(VirtualContainer, { style: {\r\n                            height: `${rowVirtualizer.totalSize}px`,\r\n                        } }, virtualItems.map(virtualItem => {\r\n                        const item = flattenedItems[virtualItem.index];\r\n                        const hierarchyValue = {\r\n                            depth: item.depth,\r\n                            parentDepth: Math.max(0, item.depth - 1),\r\n                            isTopLevel: item.depth === 0,\r\n                            index: item.index,\r\n                        };\r\n                        return (React.createElement(VirtualItem, { key: item.key, height: virtualItem.size, transform: `translateY(${virtualItem.start}px)` },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { value: hierarchyValue }, item.child)));\r\n                    }))) : (processedChildren)))))));\r\n});\r\n//# sourceMappingURL=TreeView.js.map"]} */",
|
|
19452
|
-
toString: _EMOTION_STRINGIFIED_CSS_ERROR__$H
|
|
19453
|
-
});
|
|
19445
|
+
})("width:100%;position:relative;height:", function (props) {
|
|
19446
|
+
return props.height;
|
|
19447
|
+
}, "px;" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeView.tsx"],"names":[],"mappings":"AA4BoC","file":"TreeView.tsx","sourcesContent":["import * as React from 'react';\r\nimport styled from '@emotion/styled';\r\nimport { useVirtual } from 'react-virtual';\r\nimport { TreeItem } from './TreeItem';\r\nimport { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { useTreeItem } from './useTreeItem';\r\nimport { useTreeView } from './useTreeView';\r\nimport { InverseContext, useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nconst StyledTreeView = styled.ul `\n  padding: 0;\n  margin: 0;\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral};\n  position: ${props => (props.isVirtualized ? 'relative' : 'static')};\n  ul {\n    padding: 0;\n    margin: 0;\n    li {\n      margin: 0;\n    }\n  }\n`;\r\nconst VirtualContainer = styled.div `\n  width: 100%;\n  position: relative;\n  height: ${props => props.height}px;\n`;\r\nconst VirtualItem = styled.div `\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-height: ${props => props.height}px;\n  transform: ${props => `translateY(${props.transform}px)`};\n\n  /* Ensure dropdowns and other floating elements appear above items */\n  &:focus-within {\n    z-index: 1;\n  }\n  &[data-testid='popoverContent'] {\n    z-index: 1;\n  }\n`;\r\nexport const TreeView = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, ariaLabelledBy, children, isInverse: isInverseProp, onExpandedChange, onSelectedItemChange, selectable, testId, apiRef, enableVirtualization = false, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse(isInverseProp);\r\n    const { selectionContextValue, expansionContextValue, configContextValue } = useTreeView(props);\r\n    useTreeItem({ label: ariaLabel, itemId: '' }, ref);\r\n    const inverseContextValue = React.useMemo(() => ({ isInverse }), [isInverse]);\r\n    const parentRef = React.useRef(null);\r\n    const itemHeightsByIdRef = React.useRef(new Map());\r\n    const measurementRefs = React.useRef(new Map());\r\n    // Flatten tree structure for virtualization\r\n    const flattenedItems = React.useMemo(() => {\r\n        if (!enableVirtualization)\r\n            return [];\r\n        const items = [];\r\n        const flatten = (childrenToFlatten, depth = 0) => {\r\n            React.Children.forEach(childrenToFlatten, (child, index) => {\r\n                if (!React.isValidElement(child) || child.type !== TreeItem) {\r\n                    return;\r\n                }\r\n                const itemId = child.props.itemId;\r\n                const itemKey = `${itemId}-${depth}`;\r\n                // Clone the child without its children to prevent double rendering\r\n                const childWithoutNested = React.cloneElement(child, {\r\n                    ...child.props,\r\n                    children: null,\r\n                });\r\n                items.push({\r\n                    child: childWithoutNested,\r\n                    depth,\r\n                    index,\r\n                    key: itemKey,\r\n                    itemId,\r\n                });\r\n                // Check if item has children and is expanded\r\n                const isExpanded = expansionContextValue.expandedSet?.has(itemId);\r\n                if (isExpanded && child.props.children) {\r\n                    flatten(child.props.children, depth + 1);\r\n                }\r\n            });\r\n        };\r\n        flatten(children);\r\n        return items;\r\n    }, [children, enableVirtualization, expansionContextValue.expandedSet]);\r\n    const rowVirtualizer = useVirtual({\r\n        size: flattenedItems.length,\r\n        parentRef,\r\n        estimateSize: React.useCallback((index) => {\r\n            const itemId = flattenedItems[index]?.itemId;\r\n            if (!itemId) {\r\n                return 32;\r\n            }\r\n            return itemHeightsByIdRef.current.get(itemId) ?? 32;\r\n        }, [flattenedItems]),\r\n        overscan: 6,\r\n    });\r\n    // Measure item heights after render\r\n    React.useEffect(() => {\r\n        if (!enableVirtualization) {\r\n            measurementRefs.current.clear();\r\n            itemHeightsByIdRef.current.clear();\r\n            return;\r\n        }\r\n        const measureHeights = () => {\r\n            let hasChanges = false;\r\n            measurementRefs.current.forEach((element, itemId) => {\r\n                if (element) {\r\n                    const height = element.offsetHeight;\r\n                    const currentHeight = itemHeightsByIdRef.current.get(itemId);\r\n                    if (currentHeight !== height && height > 0) {\r\n                        itemHeightsByIdRef.current.set(itemId, height);\r\n                        hasChanges = true;\r\n                    }\r\n                }\r\n            });\r\n            if (hasChanges) {\r\n                rowVirtualizer.measure();\r\n            }\r\n        };\r\n        const timeoutId = setTimeout(() => {\r\n            measureHeights();\r\n        }, 0);\r\n        if (typeof window !== 'undefined' && 'ResizeObserver' in window) {\r\n            const resizeObserver = new window.ResizeObserver(() => {\r\n                measureHeights();\r\n            });\r\n            measurementRefs.current.forEach(element => {\r\n                if (element) {\r\n                    resizeObserver.observe(element);\r\n                }\r\n            });\r\n            return () => {\r\n                clearTimeout(timeoutId);\r\n                resizeObserver.disconnect();\r\n            };\r\n        }\r\n        return () => {\r\n            clearTimeout(timeoutId);\r\n        };\r\n    }, [enableVirtualization, flattenedItems, rowVirtualizer]);\r\n    const processedChildren = React.useMemo(() => {\r\n        let treeItemIndex = 0;\r\n        return React.Children.map(children, child => {\r\n            if (!React.isValidElement(child)) {\r\n                return null;\r\n            }\r\n            if (child.type === TreeItem) {\r\n                const currentIndex = treeItemIndex++;\r\n                const hierarchyValue = {\r\n                    depth: 0,\r\n                    parentDepth: 0,\r\n                    isTopLevel: true,\r\n                    index: currentIndex,\r\n                };\r\n                // Wrap in context provider instead of cloning\r\n                return (React.createElement(TreeItemHierarchyContext.Provider, { key: `tree-item-${currentIndex}`, value: hierarchyValue }, child));\r\n            }\r\n            return null;\r\n        });\r\n    }, [children]);\r\n    React.useEffect(() => {\r\n        // Force measurement on children change\r\n        const timeoutId = setTimeout(() => {\r\n            let hasChanges = false;\r\n            measurementRefs.current.forEach((element, itemId) => {\r\n                if (element) {\r\n                    const height = element.offsetHeight;\r\n                    const currentHeight = itemHeightsByIdRef.current.get(itemId);\r\n                    if (currentHeight !== height && height > 0) {\r\n                        itemHeightsByIdRef.current.set(itemId, height);\r\n                        hasChanges = true;\r\n                    }\r\n                }\r\n            });\r\n            if (hasChanges) {\r\n                rowVirtualizer.measure();\r\n            }\r\n        }, 100); // Small delay to ensure DOM is updated\r\n        return () => clearTimeout(timeoutId);\r\n    }, [children, rowVirtualizer]);\r\n    return (React.createElement(InverseContext.Provider, { value: inverseContextValue },\r\n        React.createElement(TreeViewSelectionContext.Provider, { value: selectionContextValue },\r\n            React.createElement(TreeViewExpansionContext.Provider, { value: expansionContextValue },\r\n                React.createElement(TreeViewConfigContext.Provider, { value: configContextValue },\r\n                    React.createElement(StyledTreeView, Object.assign({}, rest, { \"aria-label\": ariaLabel, \"aria-labelledby\": ariaLabelledBy, \"aria-multiselectable\": selectable === TreeViewSelectable.multi, \"data-testid\": testId, isInverse: isInverse, isVirtualized: enableVirtualization, ref: mergedRefs => {\r\n                            if (typeof ref === 'function') {\r\n                                ref(mergedRefs);\r\n                            }\r\n                            else if (ref) {\r\n                                ref.current =\r\n                                    mergedRefs;\r\n                            }\r\n                            parentRef.current = mergedRefs;\r\n                        }, role: \"tree\", theme: theme }), enableVirtualization ? (React.createElement(VirtualContainer, { height: rowVirtualizer.totalSize }, rowVirtualizer.virtualItems.map(virtualItem => {\r\n                        const item = flattenedItems[virtualItem.index];\r\n                        const hierarchyValue = {\r\n                            depth: item.depth,\r\n                            parentDepth: Math.max(0, item.depth - 1),\r\n                            isTopLevel: item.depth === 0,\r\n                            index: item.index,\r\n                            isVirtualized: true,\r\n                        };\r\n                        return (React.createElement(VirtualItem, { key: item.key, height: virtualItem.size, transform: virtualItem.start, ref: (el) => {\r\n                                if (el) {\r\n                                    measurementRefs.current.set(item.itemId, el);\r\n                                }\r\n                                else {\r\n                                    measurementRefs.current.delete(item.itemId);\r\n                                }\r\n                            } },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { value: hierarchyValue }, item.child)));\r\n                    }))) : (processedChildren)))))));\r\n});\r\n//# sourceMappingURL=TreeView.js.map"]} */"));
|
|
19454
19448
|
var VirtualItem = /*#__PURE__*/_styled("div", {
|
|
19455
19449
|
target: "e1tyeayj0",
|
|
19456
19450
|
label: "VirtualItem"
|
|
19457
|
-
})("position:absolute;top:0;left:0;width:100%;height:", function (props) {
|
|
19451
|
+
})("position:absolute;top:0;left:0;width:100%;min-height:", function (props) {
|
|
19458
19452
|
return props.height;
|
|
19459
19453
|
}, "px;transform:", function (props) {
|
|
19460
|
-
return props.transform;
|
|
19461
|
-
}, ";" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeView.tsx"],"names":[],"mappings":"AAgC+B","file":"TreeView.tsx","sourcesContent":["import * as React from 'react';\r\nimport styled from '@emotion/styled';\r\nimport { useVirtual } from 'react-virtual';\r\nimport { TreeItem } from './TreeItem';\r\nimport { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { useTreeItem } from './useTreeItem';\r\nimport { useTreeView } from './useTreeView';\r\nimport { InverseContext, useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nconst StyledTreeView = styled.ul `\n  padding: 0;\n  margin: 0;\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral};\n  position: ${props => (props.isVirtualized ? 'relative' : 'static')};\n  ul {\n    padding: 0;\n    margin: 0;\n    li {\n      margin: 0;\n    }\n  }\n`;\r\nconst VirtualContainer = styled.div `\n  width: 100%;\n  position: relative;\n`;\r\nconst VirtualItem = styled.div `\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  height: ${props => props.height}px;\n  transform: ${props => props.transform};\n`;\r\nexport const TreeView = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, ariaLabelledBy, children, isInverse: isInverseProp, onExpandedChange, onSelectedItemChange, selectable, testId, apiRef, enableVirtualization = false, estimateSize = 40, overscan = 5, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse(isInverseProp);\r\n    const { selectionContextValue, expansionContextValue, configContextValue } = useTreeView(props);\r\n    useTreeItem({ label: ariaLabel, itemId: '' }, ref);\r\n    const inverseContextValue = React.useMemo(() => ({ isInverse }), [isInverse]);\r\n    const parentRef = React.useRef(null);\r\n    // Flatten tree structure for virtualization\r\n    const flattenedItems = React.useMemo(() => {\r\n        if (!enableVirtualization)\r\n            return [];\r\n        const items = [];\r\n        const flatten = (childrenToFlatten, depth = 0, parentIndex = 0) => {\r\n            React.Children.forEach(childrenToFlatten, (child, index) => {\r\n                if (!React.isValidElement(child) || child.type !== TreeItem) {\r\n                    return;\r\n                }\r\n                const itemKey = `tree-item-${depth}-${index}-${items.length}`;\r\n                // Clone the child without its children to prevent double rendering\r\n                const childWithoutNested = React.cloneElement(child, {\r\n                    ...child.props,\r\n                    children: null,\r\n                });\r\n                items.push({\r\n                    child: childWithoutNested,\r\n                    depth,\r\n                    index,\r\n                    key: itemKey,\r\n                });\r\n                // Check if item has children and is expanded\r\n                const itemId = child.props.itemId;\r\n                const isExpanded = expansionContextValue.expandedSet?.has(itemId);\r\n                if (isExpanded && child.props.children) {\r\n                    flatten(child.props.children, depth + 1, index);\r\n                }\r\n            });\r\n        };\r\n        flatten(children);\r\n        return items;\r\n    }, [children, enableVirtualization, expansionContextValue.expandedSet]);\r\n    // Setup virtualizer\r\n    const rowVirtualizer = useVirtual({\r\n        size: flattenedItems.length,\r\n        parentRef,\r\n        estimateSize: React.useCallback(() => estimateSize, [estimateSize]),\r\n        overscan,\r\n    });\r\n    // Process children without cloneElement - use context instead\r\n    const processedChildren = React.useMemo(() => {\r\n        if (enableVirtualization) {\r\n            return null; // Handled by virtualizer below\r\n        }\r\n        let treeItemIndex = 0;\r\n        return React.Children.map(children, child => {\r\n            if (!React.isValidElement(child)) {\r\n                return null;\r\n            }\r\n            if (child.type === TreeItem) {\r\n                const currentIndex = treeItemIndex++;\r\n                const hierarchyValue = {\r\n                    depth: 0,\r\n                    parentDepth: 0,\r\n                    isTopLevel: true,\r\n                    index: currentIndex,\r\n                };\r\n                // Wrap in context provider instead of cloning\r\n                return (React.createElement(TreeItemHierarchyContext.Provider, { key: `tree-item-${currentIndex}`, value: hierarchyValue }, child));\r\n            }\r\n            return null;\r\n        });\r\n    }, [children, enableVirtualization]);\r\n    const virtualItems = enableVirtualization\r\n        ? rowVirtualizer.virtualItems\r\n        : [];\r\n    return (React.createElement(InverseContext.Provider, { value: inverseContextValue },\r\n        React.createElement(TreeViewSelectionContext.Provider, { value: selectionContextValue },\r\n            React.createElement(TreeViewExpansionContext.Provider, { value: expansionContextValue },\r\n                React.createElement(TreeViewConfigContext.Provider, { value: configContextValue },\r\n                    React.createElement(StyledTreeView, Object.assign({}, rest, { \"aria-label\": ariaLabel, \"aria-labelledby\": ariaLabelledBy, \"aria-multiselectable\": selectable === TreeViewSelectable.multi, \"data-testid\": testId, isInverse: isInverse, isVirtualized: enableVirtualization, ref: mergedRefs => {\r\n                            if (typeof ref === 'function') {\r\n                                ref(mergedRefs);\r\n                            }\r\n                            else if (ref) {\r\n                                ref.current =\r\n                                    mergedRefs;\r\n                            }\r\n                            parentRef.current = mergedRefs;\r\n                        }, role: \"tree\", theme: theme, style: {\r\n                            ...rest.style,\r\n                            ...(enableVirtualization\r\n                                ? { height: '400px', overflow: 'auto' }\r\n                                : {}),\r\n                        } }), enableVirtualization ? (React.createElement(VirtualContainer, { style: {\r\n                            height: `${rowVirtualizer.totalSize}px`,\r\n                        } }, virtualItems.map(virtualItem => {\r\n                        const item = flattenedItems[virtualItem.index];\r\n                        const hierarchyValue = {\r\n                            depth: item.depth,\r\n                            parentDepth: Math.max(0, item.depth - 1),\r\n                            isTopLevel: item.depth === 0,\r\n                            index: item.index,\r\n                        };\r\n                        return (React.createElement(VirtualItem, { key: item.key, height: virtualItem.size, transform: `translateY(${virtualItem.start}px)` },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { value: hierarchyValue }, item.child)));\r\n                    }))) : (processedChildren)))))));\r\n});\r\n//# sourceMappingURL=TreeView.js.map"]} */"));
|
|
19454
|
+
return "translateY(" + props.transform + "px)";
|
|
19455
|
+
}, ";&:focus-within{z-index:1;}&[data-testid='popoverContent']{z-index:1;}" + ( "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["TreeView.tsx"],"names":[],"mappings":"AAiC+B","file":"TreeView.tsx","sourcesContent":["import * as React from 'react';\r\nimport styled from '@emotion/styled';\r\nimport { useVirtual } from 'react-virtual';\r\nimport { TreeItem } from './TreeItem';\r\nimport { TreeItemHierarchyContext } from './TreeItemHierarchyContext';\r\nimport { TreeViewConfigContext } from './TreeViewConfigContext';\r\nimport { TreeViewExpansionContext } from './TreeViewExpansionContext';\r\nimport { TreeViewSelectionContext } from './TreeViewSelectionContext';\r\nimport { TreeViewSelectable } from './types';\r\nimport { useTreeItem } from './useTreeItem';\r\nimport { useTreeView } from './useTreeView';\r\nimport { InverseContext, useIsInverse } from '../../inverse';\r\nimport { ThemeContext } from '../../theme/ThemeContext';\r\nconst StyledTreeView = styled.ul `\n  padding: 0;\n  margin: 0;\n  color: ${props => props.isInverse\r\n    ? props.theme.colors.neutral100\r\n    : props.theme.colors.neutral};\n  position: ${props => (props.isVirtualized ? 'relative' : 'static')};\n  ul {\n    padding: 0;\n    margin: 0;\n    li {\n      margin: 0;\n    }\n  }\n`;\r\nconst VirtualContainer = styled.div `\n  width: 100%;\n  position: relative;\n  height: ${props => props.height}px;\n`;\r\nconst VirtualItem = styled.div `\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-height: ${props => props.height}px;\n  transform: ${props => `translateY(${props.transform}px)`};\n\n  /* Ensure dropdowns and other floating elements appear above items */\n  &:focus-within {\n    z-index: 1;\n  }\n  &[data-testid='popoverContent'] {\n    z-index: 1;\n  }\n`;\r\nexport const TreeView = React.forwardRef((props, ref) => {\r\n    const { ariaLabel, ariaLabelledBy, children, isInverse: isInverseProp, onExpandedChange, onSelectedItemChange, selectable, testId, apiRef, enableVirtualization = false, ...rest } = props;\r\n    const theme = React.useContext(ThemeContext);\r\n    const isInverse = useIsInverse(isInverseProp);\r\n    const { selectionContextValue, expansionContextValue, configContextValue } = useTreeView(props);\r\n    useTreeItem({ label: ariaLabel, itemId: '' }, ref);\r\n    const inverseContextValue = React.useMemo(() => ({ isInverse }), [isInverse]);\r\n    const parentRef = React.useRef(null);\r\n    const itemHeightsByIdRef = React.useRef(new Map());\r\n    const measurementRefs = React.useRef(new Map());\r\n    // Flatten tree structure for virtualization\r\n    const flattenedItems = React.useMemo(() => {\r\n        if (!enableVirtualization)\r\n            return [];\r\n        const items = [];\r\n        const flatten = (childrenToFlatten, depth = 0) => {\r\n            React.Children.forEach(childrenToFlatten, (child, index) => {\r\n                if (!React.isValidElement(child) || child.type !== TreeItem) {\r\n                    return;\r\n                }\r\n                const itemId = child.props.itemId;\r\n                const itemKey = `${itemId}-${depth}`;\r\n                // Clone the child without its children to prevent double rendering\r\n                const childWithoutNested = React.cloneElement(child, {\r\n                    ...child.props,\r\n                    children: null,\r\n                });\r\n                items.push({\r\n                    child: childWithoutNested,\r\n                    depth,\r\n                    index,\r\n                    key: itemKey,\r\n                    itemId,\r\n                });\r\n                // Check if item has children and is expanded\r\n                const isExpanded = expansionContextValue.expandedSet?.has(itemId);\r\n                if (isExpanded && child.props.children) {\r\n                    flatten(child.props.children, depth + 1);\r\n                }\r\n            });\r\n        };\r\n        flatten(children);\r\n        return items;\r\n    }, [children, enableVirtualization, expansionContextValue.expandedSet]);\r\n    const rowVirtualizer = useVirtual({\r\n        size: flattenedItems.length,\r\n        parentRef,\r\n        estimateSize: React.useCallback((index) => {\r\n            const itemId = flattenedItems[index]?.itemId;\r\n            if (!itemId) {\r\n                return 32;\r\n            }\r\n            return itemHeightsByIdRef.current.get(itemId) ?? 32;\r\n        }, [flattenedItems]),\r\n        overscan: 6,\r\n    });\r\n    // Measure item heights after render\r\n    React.useEffect(() => {\r\n        if (!enableVirtualization) {\r\n            measurementRefs.current.clear();\r\n            itemHeightsByIdRef.current.clear();\r\n            return;\r\n        }\r\n        const measureHeights = () => {\r\n            let hasChanges = false;\r\n            measurementRefs.current.forEach((element, itemId) => {\r\n                if (element) {\r\n                    const height = element.offsetHeight;\r\n                    const currentHeight = itemHeightsByIdRef.current.get(itemId);\r\n                    if (currentHeight !== height && height > 0) {\r\n                        itemHeightsByIdRef.current.set(itemId, height);\r\n                        hasChanges = true;\r\n                    }\r\n                }\r\n            });\r\n            if (hasChanges) {\r\n                rowVirtualizer.measure();\r\n            }\r\n        };\r\n        const timeoutId = setTimeout(() => {\r\n            measureHeights();\r\n        }, 0);\r\n        if (typeof window !== 'undefined' && 'ResizeObserver' in window) {\r\n            const resizeObserver = new window.ResizeObserver(() => {\r\n                measureHeights();\r\n            });\r\n            measurementRefs.current.forEach(element => {\r\n                if (element) {\r\n                    resizeObserver.observe(element);\r\n                }\r\n            });\r\n            return () => {\r\n                clearTimeout(timeoutId);\r\n                resizeObserver.disconnect();\r\n            };\r\n        }\r\n        return () => {\r\n            clearTimeout(timeoutId);\r\n        };\r\n    }, [enableVirtualization, flattenedItems, rowVirtualizer]);\r\n    const processedChildren = React.useMemo(() => {\r\n        let treeItemIndex = 0;\r\n        return React.Children.map(children, child => {\r\n            if (!React.isValidElement(child)) {\r\n                return null;\r\n            }\r\n            if (child.type === TreeItem) {\r\n                const currentIndex = treeItemIndex++;\r\n                const hierarchyValue = {\r\n                    depth: 0,\r\n                    parentDepth: 0,\r\n                    isTopLevel: true,\r\n                    index: currentIndex,\r\n                };\r\n                // Wrap in context provider instead of cloning\r\n                return (React.createElement(TreeItemHierarchyContext.Provider, { key: `tree-item-${currentIndex}`, value: hierarchyValue }, child));\r\n            }\r\n            return null;\r\n        });\r\n    }, [children]);\r\n    React.useEffect(() => {\r\n        // Force measurement on children change\r\n        const timeoutId = setTimeout(() => {\r\n            let hasChanges = false;\r\n            measurementRefs.current.forEach((element, itemId) => {\r\n                if (element) {\r\n                    const height = element.offsetHeight;\r\n                    const currentHeight = itemHeightsByIdRef.current.get(itemId);\r\n                    if (currentHeight !== height && height > 0) {\r\n                        itemHeightsByIdRef.current.set(itemId, height);\r\n                        hasChanges = true;\r\n                    }\r\n                }\r\n            });\r\n            if (hasChanges) {\r\n                rowVirtualizer.measure();\r\n            }\r\n        }, 100); // Small delay to ensure DOM is updated\r\n        return () => clearTimeout(timeoutId);\r\n    }, [children, rowVirtualizer]);\r\n    return (React.createElement(InverseContext.Provider, { value: inverseContextValue },\r\n        React.createElement(TreeViewSelectionContext.Provider, { value: selectionContextValue },\r\n            React.createElement(TreeViewExpansionContext.Provider, { value: expansionContextValue },\r\n                React.createElement(TreeViewConfigContext.Provider, { value: configContextValue },\r\n                    React.createElement(StyledTreeView, Object.assign({}, rest, { \"aria-label\": ariaLabel, \"aria-labelledby\": ariaLabelledBy, \"aria-multiselectable\": selectable === TreeViewSelectable.multi, \"data-testid\": testId, isInverse: isInverse, isVirtualized: enableVirtualization, ref: mergedRefs => {\r\n                            if (typeof ref === 'function') {\r\n                                ref(mergedRefs);\r\n                            }\r\n                            else if (ref) {\r\n                                ref.current =\r\n                                    mergedRefs;\r\n                            }\r\n                            parentRef.current = mergedRefs;\r\n                        }, role: \"tree\", theme: theme }), enableVirtualization ? (React.createElement(VirtualContainer, { height: rowVirtualizer.totalSize }, rowVirtualizer.virtualItems.map(virtualItem => {\r\n                        const item = flattenedItems[virtualItem.index];\r\n                        const hierarchyValue = {\r\n                            depth: item.depth,\r\n                            parentDepth: Math.max(0, item.depth - 1),\r\n                            isTopLevel: item.depth === 0,\r\n                            index: item.index,\r\n                            isVirtualized: true,\r\n                        };\r\n                        return (React.createElement(VirtualItem, { key: item.key, height: virtualItem.size, transform: virtualItem.start, ref: (el) => {\r\n                                if (el) {\r\n                                    measurementRefs.current.set(item.itemId, el);\r\n                                }\r\n                                else {\r\n                                    measurementRefs.current.delete(item.itemId);\r\n                                }\r\n                            } },\r\n                            React.createElement(TreeItemHierarchyContext.Provider, { value: hierarchyValue }, item.child)));\r\n                    }))) : (processedChildren)))))));\r\n});\r\n//# sourceMappingURL=TreeView.js.map"]} */"));
|
|
19462
19456
|
var TreeView = /*#__PURE__*/React.forwardRef(function (props, _ref) {
|
|
19463
19457
|
var ariaLabel = props.ariaLabel,
|
|
19464
19458
|
ariaLabelledBy = props.ariaLabelledBy,
|
|
@@ -19468,10 +19462,6 @@ var TreeView = /*#__PURE__*/React.forwardRef(function (props, _ref) {
|
|
|
19468
19462
|
testId = props.testId,
|
|
19469
19463
|
_props$enableVirtuali = props.enableVirtualization,
|
|
19470
19464
|
enableVirtualization = _props$enableVirtuali === void 0 ? false : _props$enableVirtuali,
|
|
19471
|
-
_props$estimateSize = props.estimateSize,
|
|
19472
|
-
estimateSize = _props$estimateSize === void 0 ? 40 : _props$estimateSize,
|
|
19473
|
-
_props$overscan = props.overscan,
|
|
19474
|
-
overscan = _props$overscan === void 0 ? 5 : _props$overscan,
|
|
19475
19465
|
rest = _objectWithoutPropertiesLoose(props, _excluded$1M);
|
|
19476
19466
|
var theme = React.useContext(ThemeContext);
|
|
19477
19467
|
var isInverse = useIsInverse(isInverseProp);
|
|
@@ -19489,11 +19479,13 @@ var TreeView = /*#__PURE__*/React.forwardRef(function (props, _ref) {
|
|
|
19489
19479
|
};
|
|
19490
19480
|
}, [isInverse]);
|
|
19491
19481
|
var parentRef = React.useRef(null);
|
|
19482
|
+
var itemHeightsByIdRef = React.useRef(new Map());
|
|
19483
|
+
var measurementRefs = React.useRef(new Map());
|
|
19492
19484
|
// Flatten tree structure for virtualization
|
|
19493
19485
|
var flattenedItems = React.useMemo(function () {
|
|
19494
19486
|
if (!enableVirtualization) return [];
|
|
19495
19487
|
var items = [];
|
|
19496
|
-
var _flatten = function flatten(childrenToFlatten, depth
|
|
19488
|
+
var _flatten = function flatten(childrenToFlatten, depth) {
|
|
19497
19489
|
if (depth === void 0) {
|
|
19498
19490
|
depth = 0;
|
|
19499
19491
|
}
|
|
@@ -19502,7 +19494,8 @@ var TreeView = /*#__PURE__*/React.forwardRef(function (props, _ref) {
|
|
|
19502
19494
|
if (!React.isValidElement(child) || child.type !== TreeItem) {
|
|
19503
19495
|
return;
|
|
19504
19496
|
}
|
|
19505
|
-
var
|
|
19497
|
+
var itemId = child.props.itemId;
|
|
19498
|
+
var itemKey = itemId + "-" + depth;
|
|
19506
19499
|
// Clone the child without its children to prevent double rendering
|
|
19507
19500
|
var childWithoutNested = React.cloneElement(child, _extends({}, child.props, {
|
|
19508
19501
|
children: null
|
|
@@ -19511,10 +19504,10 @@ var TreeView = /*#__PURE__*/React.forwardRef(function (props, _ref) {
|
|
|
19511
19504
|
child: childWithoutNested,
|
|
19512
19505
|
depth: depth,
|
|
19513
19506
|
index: index,
|
|
19514
|
-
key: itemKey
|
|
19507
|
+
key: itemKey,
|
|
19508
|
+
itemId: itemId
|
|
19515
19509
|
});
|
|
19516
19510
|
// Check if item has children and is expanded
|
|
19517
|
-
var itemId = child.props.itemId;
|
|
19518
19511
|
var isExpanded = (_expansionContextValu = expansionContextValue.expandedSet) == null ? void 0 : _expansionContextValu.has(itemId);
|
|
19519
19512
|
if (isExpanded && child.props.children) {
|
|
19520
19513
|
_flatten(child.props.children, depth + 1);
|
|
@@ -19524,20 +19517,64 @@ var TreeView = /*#__PURE__*/React.forwardRef(function (props, _ref) {
|
|
|
19524
19517
|
_flatten(children);
|
|
19525
19518
|
return items;
|
|
19526
19519
|
}, [children, enableVirtualization, expansionContextValue.expandedSet]);
|
|
19527
|
-
// Setup virtualizer
|
|
19528
19520
|
var rowVirtualizer = reactVirtual.useVirtual({
|
|
19529
19521
|
size: flattenedItems.length,
|
|
19530
19522
|
parentRef: parentRef,
|
|
19531
|
-
estimateSize: React.useCallback(function () {
|
|
19532
|
-
|
|
19533
|
-
|
|
19534
|
-
|
|
19523
|
+
estimateSize: React.useCallback(function (index) {
|
|
19524
|
+
var _flattenedItems$index, _itemHeightsByIdRef$c;
|
|
19525
|
+
var itemId = (_flattenedItems$index = flattenedItems[index]) == null ? void 0 : _flattenedItems$index.itemId;
|
|
19526
|
+
if (!itemId) {
|
|
19527
|
+
return 32;
|
|
19528
|
+
}
|
|
19529
|
+
return (_itemHeightsByIdRef$c = itemHeightsByIdRef.current.get(itemId)) != null ? _itemHeightsByIdRef$c : 32;
|
|
19530
|
+
}, [flattenedItems]),
|
|
19531
|
+
overscan: 6
|
|
19535
19532
|
});
|
|
19536
|
-
//
|
|
19537
|
-
|
|
19538
|
-
if (enableVirtualization) {
|
|
19539
|
-
|
|
19533
|
+
// Measure item heights after render
|
|
19534
|
+
React.useEffect(function () {
|
|
19535
|
+
if (!enableVirtualization) {
|
|
19536
|
+
measurementRefs.current.clear();
|
|
19537
|
+
itemHeightsByIdRef.current.clear();
|
|
19538
|
+
return;
|
|
19540
19539
|
}
|
|
19540
|
+
var measureHeights = function measureHeights() {
|
|
19541
|
+
var hasChanges = false;
|
|
19542
|
+
measurementRefs.current.forEach(function (element, itemId) {
|
|
19543
|
+
if (element) {
|
|
19544
|
+
var height = element.offsetHeight;
|
|
19545
|
+
var currentHeight = itemHeightsByIdRef.current.get(itemId);
|
|
19546
|
+
if (currentHeight !== height && height > 0) {
|
|
19547
|
+
itemHeightsByIdRef.current.set(itemId, height);
|
|
19548
|
+
hasChanges = true;
|
|
19549
|
+
}
|
|
19550
|
+
}
|
|
19551
|
+
});
|
|
19552
|
+
if (hasChanges) {
|
|
19553
|
+
rowVirtualizer.measure();
|
|
19554
|
+
}
|
|
19555
|
+
};
|
|
19556
|
+
var timeoutId = setTimeout(function () {
|
|
19557
|
+
measureHeights();
|
|
19558
|
+
}, 0);
|
|
19559
|
+
if (typeof window !== 'undefined' && 'ResizeObserver' in window) {
|
|
19560
|
+
var resizeObserver = new window.ResizeObserver(function () {
|
|
19561
|
+
measureHeights();
|
|
19562
|
+
});
|
|
19563
|
+
measurementRefs.current.forEach(function (element) {
|
|
19564
|
+
if (element) {
|
|
19565
|
+
resizeObserver.observe(element);
|
|
19566
|
+
}
|
|
19567
|
+
});
|
|
19568
|
+
return function () {
|
|
19569
|
+
clearTimeout(timeoutId);
|
|
19570
|
+
resizeObserver.disconnect();
|
|
19571
|
+
};
|
|
19572
|
+
}
|
|
19573
|
+
return function () {
|
|
19574
|
+
clearTimeout(timeoutId);
|
|
19575
|
+
};
|
|
19576
|
+
}, [enableVirtualization, flattenedItems, rowVirtualizer]);
|
|
19577
|
+
var processedChildren = React.useMemo(function () {
|
|
19541
19578
|
var treeItemIndex = 0;
|
|
19542
19579
|
return React.Children.map(children, function (child) {
|
|
19543
19580
|
if (!React.isValidElement(child)) {
|
|
@@ -19559,8 +19596,29 @@ var TreeView = /*#__PURE__*/React.forwardRef(function (props, _ref) {
|
|
|
19559
19596
|
}
|
|
19560
19597
|
return null;
|
|
19561
19598
|
});
|
|
19562
|
-
}, [children
|
|
19563
|
-
|
|
19599
|
+
}, [children]);
|
|
19600
|
+
React.useEffect(function () {
|
|
19601
|
+
// Force measurement on children change
|
|
19602
|
+
var timeoutId = setTimeout(function () {
|
|
19603
|
+
var hasChanges = false;
|
|
19604
|
+
measurementRefs.current.forEach(function (element, itemId) {
|
|
19605
|
+
if (element) {
|
|
19606
|
+
var height = element.offsetHeight;
|
|
19607
|
+
var currentHeight = itemHeightsByIdRef.current.get(itemId);
|
|
19608
|
+
if (currentHeight !== height && height > 0) {
|
|
19609
|
+
itemHeightsByIdRef.current.set(itemId, height);
|
|
19610
|
+
hasChanges = true;
|
|
19611
|
+
}
|
|
19612
|
+
}
|
|
19613
|
+
});
|
|
19614
|
+
if (hasChanges) {
|
|
19615
|
+
rowVirtualizer.measure();
|
|
19616
|
+
}
|
|
19617
|
+
}, 100); // Small delay to ensure DOM is updated
|
|
19618
|
+
return function () {
|
|
19619
|
+
return clearTimeout(timeoutId);
|
|
19620
|
+
};
|
|
19621
|
+
}, [children, rowVirtualizer]);
|
|
19564
19622
|
return React.createElement(InverseContext.Provider, {
|
|
19565
19623
|
value: inverseContextValue
|
|
19566
19624
|
}, React.createElement(TreeViewSelectionContext.Provider, {
|
|
@@ -19585,27 +19643,29 @@ var TreeView = /*#__PURE__*/React.forwardRef(function (props, _ref) {
|
|
|
19585
19643
|
parentRef.current = mergedRefs;
|
|
19586
19644
|
},
|
|
19587
19645
|
role: "tree",
|
|
19588
|
-
theme: theme
|
|
19589
|
-
style: _extends({}, rest.style, enableVirtualization ? {
|
|
19590
|
-
height: '400px',
|
|
19591
|
-
overflow: 'auto'
|
|
19592
|
-
} : {})
|
|
19646
|
+
theme: theme
|
|
19593
19647
|
}), enableVirtualization ? React.createElement(VirtualContainer, {
|
|
19594
|
-
|
|
19595
|
-
|
|
19596
|
-
}
|
|
19597
|
-
}, virtualItems.map(function (virtualItem) {
|
|
19648
|
+
height: rowVirtualizer.totalSize
|
|
19649
|
+
}, rowVirtualizer.virtualItems.map(function (virtualItem) {
|
|
19598
19650
|
var item = flattenedItems[virtualItem.index];
|
|
19599
19651
|
var hierarchyValue = {
|
|
19600
19652
|
depth: item.depth,
|
|
19601
19653
|
parentDepth: Math.max(0, item.depth - 1),
|
|
19602
19654
|
isTopLevel: item.depth === 0,
|
|
19603
|
-
index: item.index
|
|
19655
|
+
index: item.index,
|
|
19656
|
+
isVirtualized: true
|
|
19604
19657
|
};
|
|
19605
19658
|
return React.createElement(VirtualItem, {
|
|
19606
19659
|
key: item.key,
|
|
19607
19660
|
height: virtualItem.size,
|
|
19608
|
-
transform:
|
|
19661
|
+
transform: virtualItem.start,
|
|
19662
|
+
ref: function ref(el) {
|
|
19663
|
+
if (el) {
|
|
19664
|
+
measurementRefs.current.set(item.itemId, el);
|
|
19665
|
+
} else {
|
|
19666
|
+
measurementRefs.current["delete"](item.itemId);
|
|
19667
|
+
}
|
|
19668
|
+
}
|
|
19609
19669
|
}, React.createElement(TreeItemHierarchyContext.Provider, {
|
|
19610
19670
|
value: hierarchyValue
|
|
19611
19671
|
}, item.child));
|