@flodesk/grain 11.71.1 → 11.72.0
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/es/foundational/menu.js
CHANGED
|
@@ -11,7 +11,7 @@ import { Global, css } from '@emotion/react';
|
|
|
11
11
|
import { getColor, getDimension, getRadius, getShadow, getSpace, getTransition } from '../utilities';
|
|
12
12
|
import { Box, Icon, Text } from '../components';
|
|
13
13
|
import { IconCheck } from '../icons';
|
|
14
|
-
import { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';
|
|
14
|
+
import { autoUpdate, flip, size, useFloating, shift } from '@floating-ui/react-dom-interactions';
|
|
15
15
|
import { Transition } from '@headlessui/react';
|
|
16
16
|
import { types } from '../types';
|
|
17
17
|
import { jsx as ___EmotionJSX } from "@emotion/react";
|
|
@@ -35,7 +35,8 @@ export const placementsMap = {
|
|
|
35
35
|
export const useMenuPosition = _ref => {
|
|
36
36
|
let {
|
|
37
37
|
menuWidth,
|
|
38
|
-
menuPlacement
|
|
38
|
+
menuPlacement,
|
|
39
|
+
enableShift
|
|
39
40
|
} = _ref;
|
|
40
41
|
const {
|
|
41
42
|
x,
|
|
@@ -62,6 +63,8 @@ export const useMenuPosition = _ref => {
|
|
|
62
63
|
});
|
|
63
64
|
},
|
|
64
65
|
padding: 24
|
|
66
|
+
}), enableShift && shift({
|
|
67
|
+
padding: 24
|
|
65
68
|
})]
|
|
66
69
|
});
|
|
67
70
|
const floatingStyles = {
|
|
@@ -119,7 +122,7 @@ const MenuCardUl = /*#__PURE__*/_styled("ul", process.env.NODE_ENV === "producti
|
|
|
119
122
|
isDisabled
|
|
120
123
|
} = _ref7;
|
|
121
124
|
return isDisabled && "pointer-events: none;";
|
|
122
|
-
}, " &:focus{outline:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AA8G4B","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, {\n  Fragment,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useRef,\n} from 'react';\nimport PropTypes from 'prop-types';\nimport { Global, css } from '@emotion/react';\nimport {\n  getColor,\n  getDimension,\n  getRadius,\n  getShadow,\n  getSpace,\n  getTransition,\n} from '../utilities';\nimport { Box, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\nimport { types } from '../types';\nimport styled from '@emotion/styled';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = `\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(function MenuCardTransition(\n  { children, show, ...props },\n  ref,\n) {\n  return (\n    <Transition\n      as={Fragment}\n      enterFrom=\"menuCardEnterFrom\"\n      enterTo=\"menuCardEnterTo\"\n      leaveFrom=\"menuCardLeaveFrom\"\n      leaveTo=\"menuCardLeaveTo\"\n      ref={ref}\n      show={show}\n      {...props}\n    >\n      {children}\n    </Transition>\n  );\n});\n\nconst MenuCardUl = styled.ul`\n  padding: ${cardPaddingPx};\n  margin: 0;\n  background: ${getColor('floatingBackground')};\n  z-index: ${({ zIndex }) => zIndex};\n  box-shadow: ${getShadow('m')};\n  border-radius: ${getRadius('m')};\n  overflow: auto;\n  ${({ maxHeight }) => maxHeight && `max-height: ${getSpace(maxHeight)}`};\n  ${({ maxWidth }) => maxWidth && `max-width: ${getSpace(maxWidth)}`};\n  /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n  ${({ isDisabled }) => isDisabled && `pointer-events: none;`}\n\n  &:focus {\n    outline: none;\n  }\n`;\n\nexport const MenuCard = forwardRef(function MenuCard(\n  { children, zIndex = 1, maxHeight, maxWidth, isDisabled, ...props },\n  ref,\n) {\n  const menuCardRef = useRef(ref);\n  useImperativeHandle(ref, () => menuCardRef.current, []);\n\n  const scrollToActive = useCallback(() => {\n    // Note: this function is rely on data-headlessui-state attribute\n    const menuCard = menuCardRef.current;\n    const activeItem = menuCard?.querySelector('[data-headlessui-state=\"selected\"]');\n    if (!menuCard || !activeItem) return;\n    const menuCardRect = menuCard.getBoundingClientRect();\n    const activeItemRect = activeItem.getBoundingClientRect();\n    menuCard.scrollTop = activeItem.offsetTop - menuCardRect.height / 2 + activeItemRect.height / 2;\n  }, []);\n\n  useEffect(() => {\n    scrollToActive();\n  }, [scrollToActive]);\n\n  return (\n    <>\n      <Global styles={menuCardStyles} />\n      <MenuCardUl\n        ref={menuCardRef}\n        zIndex={zIndex}\n        maxHeight={maxHeight}\n        maxWidth={maxWidth}\n        isDisabled={isDisabled}\n        className=\"MenuCard grn-context\"\n        {...props}\n      >\n        {children}\n      </MenuCardUl>\n    </>\n  );\n});\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('overlay'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('contentDanger'),\n    iconColor: getColor('iconDanger'),\n    iconColorActive: getColor('contentDanger'),\n    backgroundColorActive: getColor('overlayDanger'),\n  },\n};\n\nconst MenuItemRoot = styled.li`\n  display: grid;\n  grid-auto-flow: column;\n  gap: 8px;\n  ${({ isDisabled }) => isDisabled && `color: ${getColor('disabledContent')}`};\n  ${({ isDisabled }) => !isDisabled && `cursor: pointer;`};\n  align-items: center;\n  justify-content: start;\n  list-style: none;\n  padding-block: 10px;\n  padding-inline: ${itemXPadding}px;\n  margin: 0;\n  min-height: var(--grn-textbox-height-m);\n  border-radius: ${getRadius('s')};\n  appearance: none;\n  color: ${({ color }) => color};\n\n  [data-experiment-compact-textbox] & {\n    padding-block: ${getSpace('s')};\n  }\n`;\n\nexport const MenuGroupTitle = ({ children }) => (\n  <Box\n    css={css`\n      &:first-of-type {\n        --marginTop: calc(-1 * var(--grn-space-s));\n        --borderColor: transparent;\n      }\n      &:not(:first-of-type) {\n        --marginTop: ${cardPaddingPx};\n        --borderColor: var(--grn-color-border);\n      }\n    `}\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingTop=\"m\"\n    paddingBottom={cardPaddingPx}\n    marginTop=\"var(--marginTop)\"\n    borderSide=\"top\"\n    borderColor=\"var(--borderColor)\"\n    marginX={`-${cardPaddingPx}`}\n    position=\"relative\"\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height=\"1px\" marginY={1} backgroundColor=\"border\" marginX={`-${cardPaddingPx}`} />\n);\n\nexport const MenuItem = forwardRef(function MenuItem(\n  {\n    children,\n    isSelected,\n    isActive,\n    isDisabled,\n    icon,\n    variant = 'neutral',\n    hasEllipsis,\n    checkVariant = 'check',\n    ...props\n  },\n  ref,\n) {\n  const hasCheck = checkVariant === 'check';\n\n  const getColumns = () => {\n    if (Boolean(icon)) return 'auto 1fr';\n    if (hasCheck) return '1fr auto';\n  };\n\n  return (\n    <MenuItemRoot\n      ref={ref}\n      isSelected={isSelected}\n      isDisabled={isDisabled}\n      color={variantStyles[variant].color}\n      hasIcon={Boolean(icon)}\n      style={{\n        background: isActive && variantStyles[variant].backgroundColorActive,\n        gridTemplateColumns: getColumns(),\n      }}\n      {...props}\n    >\n      {icon && (\n        <Icon\n          icon={icon}\n          hasEvenBoundary\n          color={\n            isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n          }\n        />\n      )}\n      <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n        <Text hasEllipsis={hasEllipsis}>{children}</Text>\n      </Box>\n      {hasCheck && (\n        <Box opacity={isSelected ? '1' : '0'}>\n          <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n        </Box>\n      )}\n    </MenuItemRoot>\n  );\n});\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'none']),\n};\n\nMenuCard.propTypes = {\n  children: PropTypes.node,\n  zIndex: types.zIndex,\n  maxHeight: types.dimension,\n  maxWidth: types.dimension,\n  isDisabled: PropTypes.bool,\n};\n\nMenuCardTransition.propTypes = {\n  children: PropTypes.node,\n  show: PropTypes.bool,\n};\n"]} */"));
|
|
125
|
+
}, " &:focus{outline:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AA+G4B","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, {\n  Fragment,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useRef,\n} from 'react';\nimport PropTypes from 'prop-types';\nimport { Global, css } from '@emotion/react';\nimport {\n  getColor,\n  getDimension,\n  getRadius,\n  getShadow,\n  getSpace,\n  getTransition,\n} from '../utilities';\nimport { Box, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { autoUpdate, flip, size, useFloating, shift } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\nimport { types } from '../types';\nimport styled from '@emotion/styled';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement, enableShift }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n      enableShift && shift({ padding: 24 }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = `\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(function MenuCardTransition(\n  { children, show, ...props },\n  ref,\n) {\n  return (\n    <Transition\n      as={Fragment}\n      enterFrom=\"menuCardEnterFrom\"\n      enterTo=\"menuCardEnterTo\"\n      leaveFrom=\"menuCardLeaveFrom\"\n      leaveTo=\"menuCardLeaveTo\"\n      ref={ref}\n      show={show}\n      {...props}\n    >\n      {children}\n    </Transition>\n  );\n});\n\nconst MenuCardUl = styled.ul`\n  padding: ${cardPaddingPx};\n  margin: 0;\n  background: ${getColor('floatingBackground')};\n  z-index: ${({ zIndex }) => zIndex};\n  box-shadow: ${getShadow('m')};\n  border-radius: ${getRadius('m')};\n  overflow: auto;\n  ${({ maxHeight }) => maxHeight && `max-height: ${getSpace(maxHeight)}`};\n  ${({ maxWidth }) => maxWidth && `max-width: ${getSpace(maxWidth)}`};\n  /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n  ${({ isDisabled }) => isDisabled && `pointer-events: none;`}\n\n  &:focus {\n    outline: none;\n  }\n`;\n\nexport const MenuCard = forwardRef(function MenuCard(\n  { children, zIndex = 1, maxHeight, maxWidth, isDisabled, ...props },\n  ref,\n) {\n  const menuCardRef = useRef(ref);\n  useImperativeHandle(ref, () => menuCardRef.current, []);\n\n  const scrollToActive = useCallback(() => {\n    // Note: this function is rely on data-headlessui-state attribute\n    const menuCard = menuCardRef.current;\n    const activeItem = menuCard?.querySelector('[data-headlessui-state=\"selected\"]');\n    if (!menuCard || !activeItem) return;\n    const menuCardRect = menuCard.getBoundingClientRect();\n    const activeItemRect = activeItem.getBoundingClientRect();\n    menuCard.scrollTop = activeItem.offsetTop - menuCardRect.height / 2 + activeItemRect.height / 2;\n  }, []);\n\n  useEffect(() => {\n    scrollToActive();\n  }, [scrollToActive]);\n\n  return (\n    <>\n      <Global styles={menuCardStyles} />\n      <MenuCardUl\n        ref={menuCardRef}\n        zIndex={zIndex}\n        maxHeight={maxHeight}\n        maxWidth={maxWidth}\n        isDisabled={isDisabled}\n        className=\"MenuCard grn-context\"\n        {...props}\n      >\n        {children}\n      </MenuCardUl>\n    </>\n  );\n});\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('overlay'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('contentDanger'),\n    iconColor: getColor('iconDanger'),\n    iconColorActive: getColor('contentDanger'),\n    backgroundColorActive: getColor('overlayDanger'),\n  },\n};\n\nconst MenuItemRoot = styled.li`\n  display: grid;\n  grid-auto-flow: column;\n  gap: 8px;\n  ${({ isDisabled }) => isDisabled && `color: ${getColor('disabledContent')}`};\n  ${({ isDisabled }) => !isDisabled && `cursor: pointer;`};\n  align-items: center;\n  justify-content: start;\n  list-style: none;\n  padding-block: 10px;\n  padding-inline: ${itemXPadding}px;\n  margin: 0;\n  min-height: var(--grn-textbox-height-m);\n  border-radius: ${getRadius('s')};\n  appearance: none;\n  color: ${({ color }) => color};\n\n  [data-experiment-compact-textbox] & {\n    padding-block: ${getSpace('s')};\n  }\n`;\n\nexport const MenuGroupTitle = ({ children }) => (\n  <Box\n    css={css`\n      &:first-of-type {\n        --marginTop: calc(-1 * var(--grn-space-s));\n        --borderColor: transparent;\n      }\n      &:not(:first-of-type) {\n        --marginTop: ${cardPaddingPx};\n        --borderColor: var(--grn-color-border);\n      }\n    `}\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingTop=\"m\"\n    paddingBottom={cardPaddingPx}\n    marginTop=\"var(--marginTop)\"\n    borderSide=\"top\"\n    borderColor=\"var(--borderColor)\"\n    marginX={`-${cardPaddingPx}`}\n    position=\"relative\"\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height=\"1px\" marginY={1} backgroundColor=\"border\" marginX={`-${cardPaddingPx}`} />\n);\n\nexport const MenuItem = forwardRef(function MenuItem(\n  {\n    children,\n    isSelected,\n    isActive,\n    isDisabled,\n    icon,\n    variant = 'neutral',\n    hasEllipsis,\n    checkVariant = 'check',\n    ...props\n  },\n  ref,\n) {\n  const hasCheck = checkVariant === 'check';\n\n  const getColumns = () => {\n    if (Boolean(icon)) return 'auto 1fr';\n    if (hasCheck) return '1fr auto';\n  };\n\n  return (\n    <MenuItemRoot\n      ref={ref}\n      isSelected={isSelected}\n      isDisabled={isDisabled}\n      color={variantStyles[variant].color}\n      hasIcon={Boolean(icon)}\n      style={{\n        background: isActive && variantStyles[variant].backgroundColorActive,\n        gridTemplateColumns: getColumns(),\n      }}\n      {...props}\n    >\n      {icon && (\n        <Icon\n          icon={icon}\n          hasEvenBoundary\n          color={\n            isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n          }\n        />\n      )}\n      <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n        <Text hasEllipsis={hasEllipsis}>{children}</Text>\n      </Box>\n      {hasCheck && (\n        <Box opacity={isSelected ? '1' : '0'}>\n          <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n        </Box>\n      )}\n    </MenuItemRoot>\n  );\n});\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'none']),\n};\n\nMenuCard.propTypes = {\n  children: PropTypes.node,\n  zIndex: types.zIndex,\n  maxHeight: types.dimension,\n  maxWidth: types.dimension,\n  isDisabled: PropTypes.bool,\n};\n\nMenuCardTransition.propTypes = {\n  children: PropTypes.node,\n  show: PropTypes.bool,\n};\n"]} */"));
|
|
123
126
|
export const MenuCard = /*#__PURE__*/forwardRef(function MenuCard(_ref8, ref) {
|
|
124
127
|
let {
|
|
125
128
|
children,
|
|
@@ -186,13 +189,13 @@ const MenuItemRoot = /*#__PURE__*/_styled("li", process.env.NODE_ENV === "produc
|
|
|
186
189
|
color
|
|
187
190
|
} = _ref1;
|
|
188
191
|
return color;
|
|
189
|
-
}, ";[data-experiment-compact-textbox] &{padding-block:", getSpace('s'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AAoL8B","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, {\n  Fragment,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useRef,\n} from 'react';\nimport PropTypes from 'prop-types';\nimport { Global, css } from '@emotion/react';\nimport {\n  getColor,\n  getDimension,\n  getRadius,\n  getShadow,\n  getSpace,\n  getTransition,\n} from '../utilities';\nimport { Box, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\nimport { types } from '../types';\nimport styled from '@emotion/styled';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = `\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(function MenuCardTransition(\n  { children, show, ...props },\n  ref,\n) {\n  return (\n    <Transition\n      as={Fragment}\n      enterFrom=\"menuCardEnterFrom\"\n      enterTo=\"menuCardEnterTo\"\n      leaveFrom=\"menuCardLeaveFrom\"\n      leaveTo=\"menuCardLeaveTo\"\n      ref={ref}\n      show={show}\n      {...props}\n    >\n      {children}\n    </Transition>\n  );\n});\n\nconst MenuCardUl = styled.ul`\n  padding: ${cardPaddingPx};\n  margin: 0;\n  background: ${getColor('floatingBackground')};\n  z-index: ${({ zIndex }) => zIndex};\n  box-shadow: ${getShadow('m')};\n  border-radius: ${getRadius('m')};\n  overflow: auto;\n  ${({ maxHeight }) => maxHeight && `max-height: ${getSpace(maxHeight)}`};\n  ${({ maxWidth }) => maxWidth && `max-width: ${getSpace(maxWidth)}`};\n  /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n  ${({ isDisabled }) => isDisabled && `pointer-events: none;`}\n\n  &:focus {\n    outline: none;\n  }\n`;\n\nexport const MenuCard = forwardRef(function MenuCard(\n  { children, zIndex = 1, maxHeight, maxWidth, isDisabled, ...props },\n  ref,\n) {\n  const menuCardRef = useRef(ref);\n  useImperativeHandle(ref, () => menuCardRef.current, []);\n\n  const scrollToActive = useCallback(() => {\n    // Note: this function is rely on data-headlessui-state attribute\n    const menuCard = menuCardRef.current;\n    const activeItem = menuCard?.querySelector('[data-headlessui-state=\"selected\"]');\n    if (!menuCard || !activeItem) return;\n    const menuCardRect = menuCard.getBoundingClientRect();\n    const activeItemRect = activeItem.getBoundingClientRect();\n    menuCard.scrollTop = activeItem.offsetTop - menuCardRect.height / 2 + activeItemRect.height / 2;\n  }, []);\n\n  useEffect(() => {\n    scrollToActive();\n  }, [scrollToActive]);\n\n  return (\n    <>\n      <Global styles={menuCardStyles} />\n      <MenuCardUl\n        ref={menuCardRef}\n        zIndex={zIndex}\n        maxHeight={maxHeight}\n        maxWidth={maxWidth}\n        isDisabled={isDisabled}\n        className=\"MenuCard grn-context\"\n        {...props}\n      >\n        {children}\n      </MenuCardUl>\n    </>\n  );\n});\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('overlay'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('contentDanger'),\n    iconColor: getColor('iconDanger'),\n    iconColorActive: getColor('contentDanger'),\n    backgroundColorActive: getColor('overlayDanger'),\n  },\n};\n\nconst MenuItemRoot = styled.li`\n  display: grid;\n  grid-auto-flow: column;\n  gap: 8px;\n  ${({ isDisabled }) => isDisabled && `color: ${getColor('disabledContent')}`};\n  ${({ isDisabled }) => !isDisabled && `cursor: pointer;`};\n  align-items: center;\n  justify-content: start;\n  list-style: none;\n  padding-block: 10px;\n  padding-inline: ${itemXPadding}px;\n  margin: 0;\n  min-height: var(--grn-textbox-height-m);\n  border-radius: ${getRadius('s')};\n  appearance: none;\n  color: ${({ color }) => color};\n\n  [data-experiment-compact-textbox] & {\n    padding-block: ${getSpace('s')};\n  }\n`;\n\nexport const MenuGroupTitle = ({ children }) => (\n  <Box\n    css={css`\n      &:first-of-type {\n        --marginTop: calc(-1 * var(--grn-space-s));\n        --borderColor: transparent;\n      }\n      &:not(:first-of-type) {\n        --marginTop: ${cardPaddingPx};\n        --borderColor: var(--grn-color-border);\n      }\n    `}\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingTop=\"m\"\n    paddingBottom={cardPaddingPx}\n    marginTop=\"var(--marginTop)\"\n    borderSide=\"top\"\n    borderColor=\"var(--borderColor)\"\n    marginX={`-${cardPaddingPx}`}\n    position=\"relative\"\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height=\"1px\" marginY={1} backgroundColor=\"border\" marginX={`-${cardPaddingPx}`} />\n);\n\nexport const MenuItem = forwardRef(function MenuItem(\n  {\n    children,\n    isSelected,\n    isActive,\n    isDisabled,\n    icon,\n    variant = 'neutral',\n    hasEllipsis,\n    checkVariant = 'check',\n    ...props\n  },\n  ref,\n) {\n  const hasCheck = checkVariant === 'check';\n\n  const getColumns = () => {\n    if (Boolean(icon)) return 'auto 1fr';\n    if (hasCheck) return '1fr auto';\n  };\n\n  return (\n    <MenuItemRoot\n      ref={ref}\n      isSelected={isSelected}\n      isDisabled={isDisabled}\n      color={variantStyles[variant].color}\n      hasIcon={Boolean(icon)}\n      style={{\n        background: isActive && variantStyles[variant].backgroundColorActive,\n        gridTemplateColumns: getColumns(),\n      }}\n      {...props}\n    >\n      {icon && (\n        <Icon\n          icon={icon}\n          hasEvenBoundary\n          color={\n            isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n          }\n        />\n      )}\n      <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n        <Text hasEllipsis={hasEllipsis}>{children}</Text>\n      </Box>\n      {hasCheck && (\n        <Box opacity={isSelected ? '1' : '0'}>\n          <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n        </Box>\n      )}\n    </MenuItemRoot>\n  );\n});\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'none']),\n};\n\nMenuCard.propTypes = {\n  children: PropTypes.node,\n  zIndex: types.zIndex,\n  maxHeight: types.dimension,\n  maxWidth: types.dimension,\n  isDisabled: PropTypes.bool,\n};\n\nMenuCardTransition.propTypes = {\n  children: PropTypes.node,\n  show: PropTypes.bool,\n};\n"]} */"));
|
|
192
|
+
}, ";[data-experiment-compact-textbox] &{padding-block:", getSpace('s'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AAqL8B","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, {\n  Fragment,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useRef,\n} from 'react';\nimport PropTypes from 'prop-types';\nimport { Global, css } from '@emotion/react';\nimport {\n  getColor,\n  getDimension,\n  getRadius,\n  getShadow,\n  getSpace,\n  getTransition,\n} from '../utilities';\nimport { Box, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { autoUpdate, flip, size, useFloating, shift } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\nimport { types } from '../types';\nimport styled from '@emotion/styled';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement, enableShift }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n      enableShift && shift({ padding: 24 }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = `\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(function MenuCardTransition(\n  { children, show, ...props },\n  ref,\n) {\n  return (\n    <Transition\n      as={Fragment}\n      enterFrom=\"menuCardEnterFrom\"\n      enterTo=\"menuCardEnterTo\"\n      leaveFrom=\"menuCardLeaveFrom\"\n      leaveTo=\"menuCardLeaveTo\"\n      ref={ref}\n      show={show}\n      {...props}\n    >\n      {children}\n    </Transition>\n  );\n});\n\nconst MenuCardUl = styled.ul`\n  padding: ${cardPaddingPx};\n  margin: 0;\n  background: ${getColor('floatingBackground')};\n  z-index: ${({ zIndex }) => zIndex};\n  box-shadow: ${getShadow('m')};\n  border-radius: ${getRadius('m')};\n  overflow: auto;\n  ${({ maxHeight }) => maxHeight && `max-height: ${getSpace(maxHeight)}`};\n  ${({ maxWidth }) => maxWidth && `max-width: ${getSpace(maxWidth)}`};\n  /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n  ${({ isDisabled }) => isDisabled && `pointer-events: none;`}\n\n  &:focus {\n    outline: none;\n  }\n`;\n\nexport const MenuCard = forwardRef(function MenuCard(\n  { children, zIndex = 1, maxHeight, maxWidth, isDisabled, ...props },\n  ref,\n) {\n  const menuCardRef = useRef(ref);\n  useImperativeHandle(ref, () => menuCardRef.current, []);\n\n  const scrollToActive = useCallback(() => {\n    // Note: this function is rely on data-headlessui-state attribute\n    const menuCard = menuCardRef.current;\n    const activeItem = menuCard?.querySelector('[data-headlessui-state=\"selected\"]');\n    if (!menuCard || !activeItem) return;\n    const menuCardRect = menuCard.getBoundingClientRect();\n    const activeItemRect = activeItem.getBoundingClientRect();\n    menuCard.scrollTop = activeItem.offsetTop - menuCardRect.height / 2 + activeItemRect.height / 2;\n  }, []);\n\n  useEffect(() => {\n    scrollToActive();\n  }, [scrollToActive]);\n\n  return (\n    <>\n      <Global styles={menuCardStyles} />\n      <MenuCardUl\n        ref={menuCardRef}\n        zIndex={zIndex}\n        maxHeight={maxHeight}\n        maxWidth={maxWidth}\n        isDisabled={isDisabled}\n        className=\"MenuCard grn-context\"\n        {...props}\n      >\n        {children}\n      </MenuCardUl>\n    </>\n  );\n});\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('overlay'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('contentDanger'),\n    iconColor: getColor('iconDanger'),\n    iconColorActive: getColor('contentDanger'),\n    backgroundColorActive: getColor('overlayDanger'),\n  },\n};\n\nconst MenuItemRoot = styled.li`\n  display: grid;\n  grid-auto-flow: column;\n  gap: 8px;\n  ${({ isDisabled }) => isDisabled && `color: ${getColor('disabledContent')}`};\n  ${({ isDisabled }) => !isDisabled && `cursor: pointer;`};\n  align-items: center;\n  justify-content: start;\n  list-style: none;\n  padding-block: 10px;\n  padding-inline: ${itemXPadding}px;\n  margin: 0;\n  min-height: var(--grn-textbox-height-m);\n  border-radius: ${getRadius('s')};\n  appearance: none;\n  color: ${({ color }) => color};\n\n  [data-experiment-compact-textbox] & {\n    padding-block: ${getSpace('s')};\n  }\n`;\n\nexport const MenuGroupTitle = ({ children }) => (\n  <Box\n    css={css`\n      &:first-of-type {\n        --marginTop: calc(-1 * var(--grn-space-s));\n        --borderColor: transparent;\n      }\n      &:not(:first-of-type) {\n        --marginTop: ${cardPaddingPx};\n        --borderColor: var(--grn-color-border);\n      }\n    `}\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingTop=\"m\"\n    paddingBottom={cardPaddingPx}\n    marginTop=\"var(--marginTop)\"\n    borderSide=\"top\"\n    borderColor=\"var(--borderColor)\"\n    marginX={`-${cardPaddingPx}`}\n    position=\"relative\"\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height=\"1px\" marginY={1} backgroundColor=\"border\" marginX={`-${cardPaddingPx}`} />\n);\n\nexport const MenuItem = forwardRef(function MenuItem(\n  {\n    children,\n    isSelected,\n    isActive,\n    isDisabled,\n    icon,\n    variant = 'neutral',\n    hasEllipsis,\n    checkVariant = 'check',\n    ...props\n  },\n  ref,\n) {\n  const hasCheck = checkVariant === 'check';\n\n  const getColumns = () => {\n    if (Boolean(icon)) return 'auto 1fr';\n    if (hasCheck) return '1fr auto';\n  };\n\n  return (\n    <MenuItemRoot\n      ref={ref}\n      isSelected={isSelected}\n      isDisabled={isDisabled}\n      color={variantStyles[variant].color}\n      hasIcon={Boolean(icon)}\n      style={{\n        background: isActive && variantStyles[variant].backgroundColorActive,\n        gridTemplateColumns: getColumns(),\n      }}\n      {...props}\n    >\n      {icon && (\n        <Icon\n          icon={icon}\n          hasEvenBoundary\n          color={\n            isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n          }\n        />\n      )}\n      <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n        <Text hasEllipsis={hasEllipsis}>{children}</Text>\n      </Box>\n      {hasCheck && (\n        <Box opacity={isSelected ? '1' : '0'}>\n          <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n        </Box>\n      )}\n    </MenuItemRoot>\n  );\n});\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'none']),\n};\n\nMenuCard.propTypes = {\n  children: PropTypes.node,\n  zIndex: types.zIndex,\n  maxHeight: types.dimension,\n  maxWidth: types.dimension,\n  isDisabled: PropTypes.bool,\n};\n\nMenuCardTransition.propTypes = {\n  children: PropTypes.node,\n  show: PropTypes.bool,\n};\n"]} */"));
|
|
190
193
|
export const MenuGroupTitle = _ref10 => {
|
|
191
194
|
let {
|
|
192
195
|
children
|
|
193
196
|
} = _ref10;
|
|
194
197
|
return ___EmotionJSX(Box, {
|
|
195
|
-
css: /*#__PURE__*/css("&:first-of-type{--marginTop:calc(-1 * var(--grn-space-s));--borderColor:transparent;}&:not(:first-of-type){--marginTop:", cardPaddingPx, ";--borderColor:var(--grn-color-border);}" + (process.env.NODE_ENV === "production" ? "" : ";label:MenuGroupTitle;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AA4MY","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, {\n  Fragment,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useRef,\n} from 'react';\nimport PropTypes from 'prop-types';\nimport { Global, css } from '@emotion/react';\nimport {\n  getColor,\n  getDimension,\n  getRadius,\n  getShadow,\n  getSpace,\n  getTransition,\n} from '../utilities';\nimport { Box, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\nimport { types } from '../types';\nimport styled from '@emotion/styled';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = `\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(function MenuCardTransition(\n  { children, show, ...props },\n  ref,\n) {\n  return (\n    <Transition\n      as={Fragment}\n      enterFrom=\"menuCardEnterFrom\"\n      enterTo=\"menuCardEnterTo\"\n      leaveFrom=\"menuCardLeaveFrom\"\n      leaveTo=\"menuCardLeaveTo\"\n      ref={ref}\n      show={show}\n      {...props}\n    >\n      {children}\n    </Transition>\n  );\n});\n\nconst MenuCardUl = styled.ul`\n  padding: ${cardPaddingPx};\n  margin: 0;\n  background: ${getColor('floatingBackground')};\n  z-index: ${({ zIndex }) => zIndex};\n  box-shadow: ${getShadow('m')};\n  border-radius: ${getRadius('m')};\n  overflow: auto;\n  ${({ maxHeight }) => maxHeight && `max-height: ${getSpace(maxHeight)}`};\n  ${({ maxWidth }) => maxWidth && `max-width: ${getSpace(maxWidth)}`};\n  /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n  ${({ isDisabled }) => isDisabled && `pointer-events: none;`}\n\n  &:focus {\n    outline: none;\n  }\n`;\n\nexport const MenuCard = forwardRef(function MenuCard(\n  { children, zIndex = 1, maxHeight, maxWidth, isDisabled, ...props },\n  ref,\n) {\n  const menuCardRef = useRef(ref);\n  useImperativeHandle(ref, () => menuCardRef.current, []);\n\n  const scrollToActive = useCallback(() => {\n    // Note: this function is rely on data-headlessui-state attribute\n    const menuCard = menuCardRef.current;\n    const activeItem = menuCard?.querySelector('[data-headlessui-state=\"selected\"]');\n    if (!menuCard || !activeItem) return;\n    const menuCardRect = menuCard.getBoundingClientRect();\n    const activeItemRect = activeItem.getBoundingClientRect();\n    menuCard.scrollTop = activeItem.offsetTop - menuCardRect.height / 2 + activeItemRect.height / 2;\n  }, []);\n\n  useEffect(() => {\n    scrollToActive();\n  }, [scrollToActive]);\n\n  return (\n    <>\n      <Global styles={menuCardStyles} />\n      <MenuCardUl\n        ref={menuCardRef}\n        zIndex={zIndex}\n        maxHeight={maxHeight}\n        maxWidth={maxWidth}\n        isDisabled={isDisabled}\n        className=\"MenuCard grn-context\"\n        {...props}\n      >\n        {children}\n      </MenuCardUl>\n    </>\n  );\n});\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('overlay'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('contentDanger'),\n    iconColor: getColor('iconDanger'),\n    iconColorActive: getColor('contentDanger'),\n    backgroundColorActive: getColor('overlayDanger'),\n  },\n};\n\nconst MenuItemRoot = styled.li`\n  display: grid;\n  grid-auto-flow: column;\n  gap: 8px;\n  ${({ isDisabled }) => isDisabled && `color: ${getColor('disabledContent')}`};\n  ${({ isDisabled }) => !isDisabled && `cursor: pointer;`};\n  align-items: center;\n  justify-content: start;\n  list-style: none;\n  padding-block: 10px;\n  padding-inline: ${itemXPadding}px;\n  margin: 0;\n  min-height: var(--grn-textbox-height-m);\n  border-radius: ${getRadius('s')};\n  appearance: none;\n  color: ${({ color }) => color};\n\n  [data-experiment-compact-textbox] & {\n    padding-block: ${getSpace('s')};\n  }\n`;\n\nexport const MenuGroupTitle = ({ children }) => (\n  <Box\n    css={css`\n      &:first-of-type {\n        --marginTop: calc(-1 * var(--grn-space-s));\n        --borderColor: transparent;\n      }\n      &:not(:first-of-type) {\n        --marginTop: ${cardPaddingPx};\n        --borderColor: var(--grn-color-border);\n      }\n    `}\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingTop=\"m\"\n    paddingBottom={cardPaddingPx}\n    marginTop=\"var(--marginTop)\"\n    borderSide=\"top\"\n    borderColor=\"var(--borderColor)\"\n    marginX={`-${cardPaddingPx}`}\n    position=\"relative\"\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height=\"1px\" marginY={1} backgroundColor=\"border\" marginX={`-${cardPaddingPx}`} />\n);\n\nexport const MenuItem = forwardRef(function MenuItem(\n  {\n    children,\n    isSelected,\n    isActive,\n    isDisabled,\n    icon,\n    variant = 'neutral',\n    hasEllipsis,\n    checkVariant = 'check',\n    ...props\n  },\n  ref,\n) {\n  const hasCheck = checkVariant === 'check';\n\n  const getColumns = () => {\n    if (Boolean(icon)) return 'auto 1fr';\n    if (hasCheck) return '1fr auto';\n  };\n\n  return (\n    <MenuItemRoot\n      ref={ref}\n      isSelected={isSelected}\n      isDisabled={isDisabled}\n      color={variantStyles[variant].color}\n      hasIcon={Boolean(icon)}\n      style={{\n        background: isActive && variantStyles[variant].backgroundColorActive,\n        gridTemplateColumns: getColumns(),\n      }}\n      {...props}\n    >\n      {icon && (\n        <Icon\n          icon={icon}\n          hasEvenBoundary\n          color={\n            isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n          }\n        />\n      )}\n      <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n        <Text hasEllipsis={hasEllipsis}>{children}</Text>\n      </Box>\n      {hasCheck && (\n        <Box opacity={isSelected ? '1' : '0'}>\n          <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n        </Box>\n      )}\n    </MenuItemRoot>\n  );\n});\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'none']),\n};\n\nMenuCard.propTypes = {\n  children: PropTypes.node,\n  zIndex: types.zIndex,\n  maxHeight: types.dimension,\n  maxWidth: types.dimension,\n  isDisabled: PropTypes.bool,\n};\n\nMenuCardTransition.propTypes = {\n  children: PropTypes.node,\n  show: PropTypes.bool,\n};\n"]} */"),
|
|
198
|
+
css: /*#__PURE__*/css("&:first-of-type{--marginTop:calc(-1 * var(--grn-space-s));--borderColor:transparent;}&:not(:first-of-type){--marginTop:", cardPaddingPx, ";--borderColor:var(--grn-color-border);}" + (process.env.NODE_ENV === "production" ? "" : ";label:MenuGroupTitle;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AA6MY","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, {\n  Fragment,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useRef,\n} from 'react';\nimport PropTypes from 'prop-types';\nimport { Global, css } from '@emotion/react';\nimport {\n  getColor,\n  getDimension,\n  getRadius,\n  getShadow,\n  getSpace,\n  getTransition,\n} from '../utilities';\nimport { Box, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { autoUpdate, flip, size, useFloating, shift } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\nimport { types } from '../types';\nimport styled from '@emotion/styled';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement, enableShift }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n      enableShift && shift({ padding: 24 }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = `\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(function MenuCardTransition(\n  { children, show, ...props },\n  ref,\n) {\n  return (\n    <Transition\n      as={Fragment}\n      enterFrom=\"menuCardEnterFrom\"\n      enterTo=\"menuCardEnterTo\"\n      leaveFrom=\"menuCardLeaveFrom\"\n      leaveTo=\"menuCardLeaveTo\"\n      ref={ref}\n      show={show}\n      {...props}\n    >\n      {children}\n    </Transition>\n  );\n});\n\nconst MenuCardUl = styled.ul`\n  padding: ${cardPaddingPx};\n  margin: 0;\n  background: ${getColor('floatingBackground')};\n  z-index: ${({ zIndex }) => zIndex};\n  box-shadow: ${getShadow('m')};\n  border-radius: ${getRadius('m')};\n  overflow: auto;\n  ${({ maxHeight }) => maxHeight && `max-height: ${getSpace(maxHeight)}`};\n  ${({ maxWidth }) => maxWidth && `max-width: ${getSpace(maxWidth)}`};\n  /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n  ${({ isDisabled }) => isDisabled && `pointer-events: none;`}\n\n  &:focus {\n    outline: none;\n  }\n`;\n\nexport const MenuCard = forwardRef(function MenuCard(\n  { children, zIndex = 1, maxHeight, maxWidth, isDisabled, ...props },\n  ref,\n) {\n  const menuCardRef = useRef(ref);\n  useImperativeHandle(ref, () => menuCardRef.current, []);\n\n  const scrollToActive = useCallback(() => {\n    // Note: this function is rely on data-headlessui-state attribute\n    const menuCard = menuCardRef.current;\n    const activeItem = menuCard?.querySelector('[data-headlessui-state=\"selected\"]');\n    if (!menuCard || !activeItem) return;\n    const menuCardRect = menuCard.getBoundingClientRect();\n    const activeItemRect = activeItem.getBoundingClientRect();\n    menuCard.scrollTop = activeItem.offsetTop - menuCardRect.height / 2 + activeItemRect.height / 2;\n  }, []);\n\n  useEffect(() => {\n    scrollToActive();\n  }, [scrollToActive]);\n\n  return (\n    <>\n      <Global styles={menuCardStyles} />\n      <MenuCardUl\n        ref={menuCardRef}\n        zIndex={zIndex}\n        maxHeight={maxHeight}\n        maxWidth={maxWidth}\n        isDisabled={isDisabled}\n        className=\"MenuCard grn-context\"\n        {...props}\n      >\n        {children}\n      </MenuCardUl>\n    </>\n  );\n});\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('overlay'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('contentDanger'),\n    iconColor: getColor('iconDanger'),\n    iconColorActive: getColor('contentDanger'),\n    backgroundColorActive: getColor('overlayDanger'),\n  },\n};\n\nconst MenuItemRoot = styled.li`\n  display: grid;\n  grid-auto-flow: column;\n  gap: 8px;\n  ${({ isDisabled }) => isDisabled && `color: ${getColor('disabledContent')}`};\n  ${({ isDisabled }) => !isDisabled && `cursor: pointer;`};\n  align-items: center;\n  justify-content: start;\n  list-style: none;\n  padding-block: 10px;\n  padding-inline: ${itemXPadding}px;\n  margin: 0;\n  min-height: var(--grn-textbox-height-m);\n  border-radius: ${getRadius('s')};\n  appearance: none;\n  color: ${({ color }) => color};\n\n  [data-experiment-compact-textbox] & {\n    padding-block: ${getSpace('s')};\n  }\n`;\n\nexport const MenuGroupTitle = ({ children }) => (\n  <Box\n    css={css`\n      &:first-of-type {\n        --marginTop: calc(-1 * var(--grn-space-s));\n        --borderColor: transparent;\n      }\n      &:not(:first-of-type) {\n        --marginTop: ${cardPaddingPx};\n        --borderColor: var(--grn-color-border);\n      }\n    `}\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingTop=\"m\"\n    paddingBottom={cardPaddingPx}\n    marginTop=\"var(--marginTop)\"\n    borderSide=\"top\"\n    borderColor=\"var(--borderColor)\"\n    marginX={`-${cardPaddingPx}`}\n    position=\"relative\"\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height=\"1px\" marginY={1} backgroundColor=\"border\" marginX={`-${cardPaddingPx}`} />\n);\n\nexport const MenuItem = forwardRef(function MenuItem(\n  {\n    children,\n    isSelected,\n    isActive,\n    isDisabled,\n    icon,\n    variant = 'neutral',\n    hasEllipsis,\n    checkVariant = 'check',\n    ...props\n  },\n  ref,\n) {\n  const hasCheck = checkVariant === 'check';\n\n  const getColumns = () => {\n    if (Boolean(icon)) return 'auto 1fr';\n    if (hasCheck) return '1fr auto';\n  };\n\n  return (\n    <MenuItemRoot\n      ref={ref}\n      isSelected={isSelected}\n      isDisabled={isDisabled}\n      color={variantStyles[variant].color}\n      hasIcon={Boolean(icon)}\n      style={{\n        background: isActive && variantStyles[variant].backgroundColorActive,\n        gridTemplateColumns: getColumns(),\n      }}\n      {...props}\n    >\n      {icon && (\n        <Icon\n          icon={icon}\n          hasEvenBoundary\n          color={\n            isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n          }\n        />\n      )}\n      <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n        <Text hasEllipsis={hasEllipsis}>{children}</Text>\n      </Box>\n      {hasCheck && (\n        <Box opacity={isSelected ? '1' : '0'}>\n          <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n        </Box>\n      )}\n    </MenuItemRoot>\n  );\n});\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'none']),\n};\n\nMenuCard.propTypes = {\n  children: PropTypes.node,\n  zIndex: types.zIndex,\n  maxHeight: types.dimension,\n  maxWidth: types.dimension,\n  isDisabled: PropTypes.bool,\n};\n\nMenuCardTransition.propTypes = {\n  children: PropTypes.node,\n  show: PropTypes.bool,\n};\n"]} */"),
|
|
196
199
|
paddingX: "".concat(cardPadding + itemXPadding, "px"),
|
|
197
200
|
paddingTop: "m",
|
|
198
201
|
paddingBottom: cardPaddingPx,
|