react-magma-dom 4.11.0-next.18 → 4.11.0-next.20

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/esm/index.js CHANGED
@@ -22240,7 +22240,8 @@ var TreeItemHierarchyContext = /*#__PURE__*/createContext({
22240
22240
  depth: 0,
22241
22241
  parentDepth: 0,
22242
22242
  isTopLevel: true,
22243
- index: 0
22243
+ index: 0,
22244
+ isVirtualized: false
22244
22245
  });
22245
22246
 
22246
22247
  var TreeViewSelectable;
@@ -22297,7 +22298,7 @@ var TreeNodeType;
22297
22298
  * The label element (the div inside the li) gets additional spacing.
22298
22299
  * In order to highlight the entire line, we need to negate the value for margin.
22299
22300
  */
22300
- function calculateOffset(type, depth, labelElem, negative) {
22301
+ function calculateOffset(type, depth, labelElem, negative, isVirtualized) {
22301
22302
  if (depth === void 0) {
22302
22303
  depth = 0;
22303
22304
  }
@@ -22307,6 +22308,9 @@ function calculateOffset(type, depth, labelElem, negative) {
22307
22308
  if (negative === void 0) {
22308
22309
  negative = false;
22309
22310
  }
22311
+ if (isVirtualized === void 0) {
22312
+ isVirtualized = false;
22313
+ }
22310
22314
  var padding = 0;
22311
22315
  if (type === TreeNodeType.leaf) {
22312
22316
  if (labelElem) {
@@ -22316,6 +22320,10 @@ function calculateOffset(type, depth, labelElem, negative) {
22316
22320
  }
22317
22321
  } else if (depth === 0) {
22318
22322
  padding = 40;
22323
+ } else if (isVirtualized) {
22324
+ // For virtualized items, calculate cumulative padding
22325
+ // Base padding (40px) + (depth * 24px per level after first)
22326
+ padding = 40 + (depth - 1) * 24;
22319
22327
  } else {
22320
22328
  padding = 56;
22321
22329
  }
@@ -22327,6 +22335,10 @@ function calculateOffset(type, depth, labelElem, negative) {
22327
22335
  }
22328
22336
  } else if (depth === 0) {
22329
22337
  padding = 8;
22338
+ } else if (isVirtualized) {
22339
+ // For virtualized items, calculate cumulative padding
22340
+ // Base padding (8px) + (depth * 24px per level after first)
22341
+ padding = 8 + (depth - 1) * 24;
22330
22342
  } else {
22331
22343
  padding = 24;
22332
22344
  }
@@ -23317,15 +23329,15 @@ var StyledTreeItem = /*#__PURE__*/_styled("li", {
23317
23329
  }, ";list-style-type:none;cursor:", function (props) {
23318
23330
  return getTreeItemWrapperCursor(props.isDisabled, props.selectableType, props.nodeType);
23319
23331
  }, ";position:relative;margin-bottom:0;padding-inline-start:", function (props) {
23320
- return calculateOffset(props.nodeType, props.depth);
23332
+ return calculateOffset(props.nodeType, props.depth, false, false, props.isVirtualized);
23321
23333
  }, ";&:focus{outline:none;&>*:first-child{outline-offset:-2px;outline:2px solid ", function (props) {
23322
23334
  return props.isInverse ? props.theme.colors.focusInverse : props.theme.colors.focus;
23323
23335
  }, ";}}>div:first-of-type{background:", function (props) {
23324
23336
  return props.selected && props.isInverse ? curriedTransparentize(0.7, props.theme.colors.neutral900) : props.selected && curriedTransparentize(0.92, props.theme.colors.neutral900);
23325
23337
  }, ";position:relative;padding-inline-start:", function (props) {
23326
- return calculateOffset(props.nodeType, props.depth, true);
23338
+ return calculateOffset(props.nodeType, props.depth, true, false, props.isVirtualized);
23327
23339
  }, ";margin-inline-start:", function (props) {
23328
- return calculateOffset(props.nodeType, props.depth, true, true);
23340
+ return calculateOffset(props.nodeType, props.depth, true, true, props.isVirtualized);
23329
23341
  }, ";padding-block-end:", function (props) {
23330
23342
  return props.theme.spaceScale.spacing02;
23331
23343
  }, ";padding-block-start:", function (props) {
@@ -23333,7 +23345,7 @@ var StyledTreeItem = /*#__PURE__*/_styled("li", {
23333
23345
  }, ";padding-right:", function (props) {
23334
23346
  return props.theme.spaceScale.spacing02;
23335
23347
  }, ";", function (props) {
23336
- return props.selected && /*#__PURE__*/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;" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23348
+ return props.selected && /*#__PURE__*/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;" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23337
23349
  }, " &:hover{background:", function (props) {
23338
23350
  return getHoverBackground({
23339
23351
  isDisabled: props.isDisabled,
@@ -23341,7 +23353,7 @@ var StyledTreeItem = /*#__PURE__*/_styled("li", {
23341
23353
  isInverse: props.isInverse,
23342
23354
  theme: props.theme
23343
23355
  });
23344
- }, ";}}" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23356
+ }, ";}}" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23345
23357
  function getHoverBackground(_ref) {
23346
23358
  var isDisabled = _ref.isDisabled,
23347
23359
  hoverColor = _ref.hoverColor,
@@ -23363,13 +23375,13 @@ var IconWrapper$8 = /*#__PURE__*/_styled("span", {
23363
23375
  return props.theme.iconSizes.medium;
23364
23376
  }, "px;width:", function (props) {
23365
23377
  return props.theme.iconSizes.medium;
23366
- }, "px;vertical-align:middle;}" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23378
+ }, "px;vertical-align:middle;}" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23367
23379
  var StyledLabelWrapper = /*#__PURE__*/_styled("span", {
23368
23380
  target: "e1xiryew4",
23369
23381
  label: "StyledLabelWrapper"
23370
23382
  })("display:flex;align-items:flex-start;color:", function (props) {
23371
23383
  return getTreeItemLabelColor(props.isInverse, props.isDisabled, props.theme);
23372
- }, ";width:100%;" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23384
+ }, ";width:100%;" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23373
23385
  var StyledExpandWrapper = /*#__PURE__*/_styled("div", {
23374
23386
  target: "e1xiryew3",
23375
23387
  label: "StyledExpandWrapper"
@@ -23385,7 +23397,7 @@ var StyledExpandWrapper = /*#__PURE__*/_styled("div", {
23385
23397
  var size = _ref3.size,
23386
23398
  theme = _ref3.theme;
23387
23399
  return size !== undefined ? size + "px" : theme.spaceScale.spacing06;
23388
- }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23400
+ }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23389
23401
  var StyledCheckboxWrapper = /*#__PURE__*/_styled("div", {
23390
23402
  target: "e1xiryew2",
23391
23403
  label: "StyledCheckboxWrapper"
@@ -23395,7 +23407,7 @@ var StyledCheckboxWrapper = /*#__PURE__*/_styled("div", {
23395
23407
  return props.hasAdditionalContent ? 'flex' : 'inline-flex';
23396
23408
  }, ";flex-direction:column;width:", function (props) {
23397
23409
  return "calc(100% - " + props.theme.spaceScale.spacing03 + ")";
23398
- }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23410
+ }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23399
23411
  var StyledItemWrapper = /*#__PURE__*/_styled("div", {
23400
23412
  target: "e1xiryew1",
23401
23413
  label: "StyledItemWrapper"
@@ -23405,13 +23417,13 @@ var StyledItemWrapper = /*#__PURE__*/_styled("div", {
23405
23417
  return props.hasCustomIconSize ? 'center' : 'flex-start';
23406
23418
  }, ";cursor:", function (props) {
23407
23419
  return getTreeItemWrapperCursor(props.isDisabled, props.selectable, props.nodeType);
23408
- }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23420
+ }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23409
23421
  var AdditionalContentWrapper$1 = /*#__PURE__*/_styled("div", {
23410
23422
  target: "e1xiryew0",
23411
23423
  label: "AdditionalContentWrapper"
23412
23424
  })("margin-bottom:", function (props) {
23413
23425
  return props.theme.spaceScale.spacing05;
23414
- }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23426
+ }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
23415
23427
  var TreeItemComponent = /*#__PURE__*/forwardRef(function (props, forwardedRef) {
23416
23428
  var _selectedItems$;
23417
23429
  var additionalContent = props.additionalContent,
@@ -23672,6 +23684,7 @@ var TreeItemComponent = /*#__PURE__*/forwardRef(function (props, forwardedRef) {
23672
23684
  id: itemId,
23673
23685
  isDisabled: isDisabled,
23674
23686
  isInverse: isInverse,
23687
+ isVirtualized: hierarchyContext.isVirtualized,
23675
23688
  nodeType: nodeType,
23676
23689
  role: "treeitem",
23677
23690
  selectableType: selectable,
@@ -24401,8 +24414,7 @@ function useTreeView(props) {
24401
24414
  };
24402
24415
  }
24403
24416
 
24404
- var _excluded$1M = ["ariaLabel", "ariaLabelledBy", "children", "isInverse", "onExpandedChange", "onSelectedItemChange", "selectable", "testId", "apiRef", "enableVirtualization", "estimateSize", "overscan"];
24405
- 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)."; }
24417
+ var _excluded$1M = ["ariaLabel", "ariaLabelledBy", "children", "isInverse", "onExpandedChange", "onSelectedItemChange", "selectable", "testId", "apiRef", "enableVirtualization"];
24406
24418
  var StyledTreeView = /*#__PURE__*/_styled("ul", {
24407
24419
  target: "e1tyeayj2",
24408
24420
  label: "StyledTreeView"
@@ -24410,26 +24422,21 @@ var StyledTreeView = /*#__PURE__*/_styled("ul", {
24410
24422
  return props.isInverse ? props.theme.colors.neutral100 : props.theme.colors.neutral;
24411
24423
  }, ";position:", function (props) {
24412
24424
  return props.isVirtualized ? 'relative' : 'static';
24413
- }, ";ul{padding:0;margin:0;li{margin:0;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# 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                    itemSize: child.props.itemSize,\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    const rowVirtualizer = useVirtual({\r\n        size: flattenedItems.length,\r\n        parentRef,\r\n        estimateSize: React.useCallback((index) => {\r\n            return flattenedItems[index]?.itemSize ?? estimateSize;\r\n        }, [flattenedItems, 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"]} */"));
24425
+ }, ";ul{padding:0;margin:0;li{margin:0;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
24414
24426
  var VirtualContainer = /*#__PURE__*/_styled("div", {
24415
24427
  target: "e1tyeayj1",
24416
24428
  label: "VirtualContainer"
24417
- })(process.env.NODE_ENV === "production" ? {
24418
- name: "n48rgu",
24419
- styles: "width:100%;position:relative"
24420
- } : {
24421
- name: "n48rgu",
24422
- 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                    itemSize: child.props.itemSize,\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    const rowVirtualizer = useVirtual({\r\n        size: flattenedItems.length,\r\n        parentRef,\r\n        estimateSize: React.useCallback((index) => {\r\n            return flattenedItems[index]?.itemSize ?? estimateSize;\r\n        }, [flattenedItems, 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"]} */",
24423
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__$H
24424
- });
24429
+ })("width:100%;position:relative;height:", function (props) {
24430
+ return props.height;
24431
+ }, "px;" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
24425
24432
  var VirtualItem = /*#__PURE__*/_styled("div", {
24426
24433
  target: "e1tyeayj0",
24427
24434
  label: "VirtualItem"
24428
- })("position:absolute;top:0;left:0;width:100%;height:", function (props) {
24435
+ })("position:absolute;top:0;left:0;width:100%;min-height:", function (props) {
24429
24436
  return props.height;
24430
24437
  }, "px;transform:", function (props) {
24431
- return props.transform;
24432
- }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# 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                    itemSize: child.props.itemSize,\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    const rowVirtualizer = useVirtual({\r\n        size: flattenedItems.length,\r\n        parentRef,\r\n        estimateSize: React.useCallback((index) => {\r\n            return flattenedItems[index]?.itemSize ?? estimateSize;\r\n        }, [flattenedItems, 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"]} */"));
24438
+ return "translateY(" + props.transform + "px)";
24439
+ }, ";&:focus-within{z-index:1;}&[data-testid='popoverContent']{z-index:1;}" + (process.env.NODE_ENV === "production" ? "" : "/*# 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"]} */"));
24433
24440
  var TreeView = /*#__PURE__*/forwardRef(function (props, _ref) {
24434
24441
  var ariaLabel = props.ariaLabel,
24435
24442
  ariaLabelledBy = props.ariaLabelledBy,
@@ -24439,10 +24446,6 @@ var TreeView = /*#__PURE__*/forwardRef(function (props, _ref) {
24439
24446
  testId = props.testId,
24440
24447
  _props$enableVirtuali = props.enableVirtualization,
24441
24448
  enableVirtualization = _props$enableVirtuali === void 0 ? false : _props$enableVirtuali,
24442
- _props$estimateSize = props.estimateSize,
24443
- estimateSize = _props$estimateSize === void 0 ? 40 : _props$estimateSize,
24444
- _props$overscan = props.overscan,
24445
- overscan = _props$overscan === void 0 ? 5 : _props$overscan,
24446
24449
  rest = _objectWithoutPropertiesLoose(props, _excluded$1M);
24447
24450
  var theme = useContext(ThemeContext);
24448
24451
  var isInverse = useIsInverse(isInverseProp);
@@ -24460,11 +24463,13 @@ var TreeView = /*#__PURE__*/forwardRef(function (props, _ref) {
24460
24463
  };
24461
24464
  }, [isInverse]);
24462
24465
  var parentRef = useRef(null);
24466
+ var itemHeightsByIdRef = useRef(new Map());
24467
+ var measurementRefs = useRef(new Map());
24463
24468
  // Flatten tree structure for virtualization
24464
24469
  var flattenedItems = useMemo(function () {
24465
24470
  if (!enableVirtualization) return [];
24466
24471
  var items = [];
24467
- var _flatten = function flatten(childrenToFlatten, depth, parentIndex) {
24472
+ var _flatten = function flatten(childrenToFlatten, depth) {
24468
24473
  if (depth === void 0) {
24469
24474
  depth = 0;
24470
24475
  }
@@ -24473,7 +24478,8 @@ var TreeView = /*#__PURE__*/forwardRef(function (props, _ref) {
24473
24478
  if (!isValidElement(child) || child.type !== TreeItem) {
24474
24479
  return;
24475
24480
  }
24476
- var itemKey = "tree-item-" + depth + "-" + index + "-" + items.length;
24481
+ var itemId = child.props.itemId;
24482
+ var itemKey = itemId + "-" + depth;
24477
24483
  // Clone the child without its children to prevent double rendering
24478
24484
  var childWithoutNested = cloneElement(child, _extends({}, child.props, {
24479
24485
  children: null
@@ -24483,10 +24489,9 @@ var TreeView = /*#__PURE__*/forwardRef(function (props, _ref) {
24483
24489
  depth: depth,
24484
24490
  index: index,
24485
24491
  key: itemKey,
24486
- itemSize: child.props.itemSize
24492
+ itemId: itemId
24487
24493
  });
24488
24494
  // Check if item has children and is expanded
24489
- var itemId = child.props.itemId;
24490
24495
  var isExpanded = (_expansionContextValu = expansionContextValue.expandedSet) == null ? void 0 : _expansionContextValu.has(itemId);
24491
24496
  if (isExpanded && child.props.children) {
24492
24497
  _flatten(child.props.children, depth + 1);
@@ -24500,16 +24505,60 @@ var TreeView = /*#__PURE__*/forwardRef(function (props, _ref) {
24500
24505
  size: flattenedItems.length,
24501
24506
  parentRef: parentRef,
24502
24507
  estimateSize: useCallback(function (index) {
24503
- var _flattenedItems$index, _flattenedItems$index2;
24504
- return (_flattenedItems$index = (_flattenedItems$index2 = flattenedItems[index]) == null ? void 0 : _flattenedItems$index2.itemSize) != null ? _flattenedItems$index : estimateSize;
24505
- }, [flattenedItems, estimateSize]),
24506
- overscan: overscan
24508
+ var _flattenedItems$index, _itemHeightsByIdRef$c;
24509
+ var itemId = (_flattenedItems$index = flattenedItems[index]) == null ? void 0 : _flattenedItems$index.itemId;
24510
+ if (!itemId) {
24511
+ return 32;
24512
+ }
24513
+ return (_itemHeightsByIdRef$c = itemHeightsByIdRef.current.get(itemId)) != null ? _itemHeightsByIdRef$c : 32;
24514
+ }, [flattenedItems]),
24515
+ overscan: 6
24507
24516
  });
24508
- // Process children without cloneElement - use context instead
24509
- var processedChildren = useMemo(function () {
24510
- if (enableVirtualization) {
24511
- return null; // Handled by virtualizer below
24517
+ // Measure item heights after render
24518
+ useEffect(function () {
24519
+ if (!enableVirtualization) {
24520
+ measurementRefs.current.clear();
24521
+ itemHeightsByIdRef.current.clear();
24522
+ return;
24512
24523
  }
24524
+ var measureHeights = function measureHeights() {
24525
+ var hasChanges = false;
24526
+ measurementRefs.current.forEach(function (element, itemId) {
24527
+ if (element) {
24528
+ var height = element.offsetHeight;
24529
+ var currentHeight = itemHeightsByIdRef.current.get(itemId);
24530
+ if (currentHeight !== height && height > 0) {
24531
+ itemHeightsByIdRef.current.set(itemId, height);
24532
+ hasChanges = true;
24533
+ }
24534
+ }
24535
+ });
24536
+ if (hasChanges) {
24537
+ rowVirtualizer.measure();
24538
+ }
24539
+ };
24540
+ var timeoutId = setTimeout(function () {
24541
+ measureHeights();
24542
+ }, 0);
24543
+ if (typeof window !== 'undefined' && 'ResizeObserver' in window) {
24544
+ var resizeObserver = new window.ResizeObserver(function () {
24545
+ measureHeights();
24546
+ });
24547
+ measurementRefs.current.forEach(function (element) {
24548
+ if (element) {
24549
+ resizeObserver.observe(element);
24550
+ }
24551
+ });
24552
+ return function () {
24553
+ clearTimeout(timeoutId);
24554
+ resizeObserver.disconnect();
24555
+ };
24556
+ }
24557
+ return function () {
24558
+ clearTimeout(timeoutId);
24559
+ };
24560
+ }, [enableVirtualization, flattenedItems, rowVirtualizer]);
24561
+ var processedChildren = useMemo(function () {
24513
24562
  var treeItemIndex = 0;
24514
24563
  return Children.map(children, function (child) {
24515
24564
  if (!isValidElement(child)) {
@@ -24531,8 +24580,29 @@ var TreeView = /*#__PURE__*/forwardRef(function (props, _ref) {
24531
24580
  }
24532
24581
  return null;
24533
24582
  });
24534
- }, [children, enableVirtualization]);
24535
- var virtualItems = enableVirtualization ? rowVirtualizer.virtualItems : [];
24583
+ }, [children]);
24584
+ useEffect(function () {
24585
+ // Force measurement on children change
24586
+ var timeoutId = setTimeout(function () {
24587
+ var hasChanges = false;
24588
+ measurementRefs.current.forEach(function (element, itemId) {
24589
+ if (element) {
24590
+ var height = element.offsetHeight;
24591
+ var currentHeight = itemHeightsByIdRef.current.get(itemId);
24592
+ if (currentHeight !== height && height > 0) {
24593
+ itemHeightsByIdRef.current.set(itemId, height);
24594
+ hasChanges = true;
24595
+ }
24596
+ }
24597
+ });
24598
+ if (hasChanges) {
24599
+ rowVirtualizer.measure();
24600
+ }
24601
+ }, 100); // Small delay to ensure DOM is updated
24602
+ return function () {
24603
+ return clearTimeout(timeoutId);
24604
+ };
24605
+ }, [children, rowVirtualizer]);
24536
24606
  return createElement(InverseContext.Provider, {
24537
24607
  value: inverseContextValue
24538
24608
  }, createElement(TreeViewSelectionContext.Provider, {
@@ -24557,27 +24627,29 @@ var TreeView = /*#__PURE__*/forwardRef(function (props, _ref) {
24557
24627
  parentRef.current = mergedRefs;
24558
24628
  },
24559
24629
  role: "tree",
24560
- theme: theme,
24561
- style: _extends({}, rest.style, enableVirtualization ? {
24562
- height: '400px',
24563
- overflow: 'auto'
24564
- } : {})
24630
+ theme: theme
24565
24631
  }), enableVirtualization ? createElement(VirtualContainer, {
24566
- style: {
24567
- height: rowVirtualizer.totalSize + "px"
24568
- }
24569
- }, virtualItems.map(function (virtualItem) {
24632
+ height: rowVirtualizer.totalSize
24633
+ }, rowVirtualizer.virtualItems.map(function (virtualItem) {
24570
24634
  var item = flattenedItems[virtualItem.index];
24571
24635
  var hierarchyValue = {
24572
24636
  depth: item.depth,
24573
24637
  parentDepth: Math.max(0, item.depth - 1),
24574
24638
  isTopLevel: item.depth === 0,
24575
- index: item.index
24639
+ index: item.index,
24640
+ isVirtualized: true
24576
24641
  };
24577
24642
  return createElement(VirtualItem, {
24578
24643
  key: item.key,
24579
24644
  height: virtualItem.size,
24580
- transform: "translateY(" + virtualItem.start + "px)"
24645
+ transform: virtualItem.start,
24646
+ ref: function ref(el) {
24647
+ if (el) {
24648
+ measurementRefs.current.set(item.itemId, el);
24649
+ } else {
24650
+ measurementRefs.current["delete"](item.itemId);
24651
+ }
24652
+ }
24581
24653
  }, createElement(TreeItemHierarchyContext.Provider, {
24582
24654
  value: hierarchyValue
24583
24655
  }, item.child));