@flodesk/grain 7.7.0 → 7.8.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.
@@ -52,7 +52,8 @@ export var Dropdown = function Dropdown(_ref) {
52
52
  onClick: option.onClick,
53
53
  variant: option.variant,
54
54
  hasDivider: option.hasDivider,
55
- hasEllipsis: true
55
+ hasEllipsis: true,
56
+ checkVariant: "none"
56
57
  }, option.content);
57
58
  });
58
59
  }))));
@@ -4,7 +4,7 @@ import "core-js/modules/es.symbol.js";
4
4
  import "core-js/modules/es.object.assign.js";
5
5
  import _styled from "@emotion/styled/base";
6
6
  var _excluded = ["children", "triggerVariant"],
7
- _excluded2 = ["option", "isSelected", "menuItemsHaveEllipsis"],
7
+ _excluded2 = ["option", "isSelected", "menuItemsHaveEllipsis", "allowMultiple"],
8
8
  _excluded3 = ["children", "floating", "hasPortal"];
9
9
  import "core-js/modules/es.array.map.js";
10
10
  import "core-js/modules/es.object.to-string.js";
@@ -38,7 +38,7 @@ var TriggerButton = /*#__PURE__*/_styled("button", process.env.NODE_ENV === "pro
38
38
  } : {
39
39
  target: "e17qd7kh1",
40
40
  label: "TriggerButton"
41
- })(styles.buttonReset, ";", styles.transitions, ";color:inherit;display:flex;align-items:center;gap:8px;border:none;background:", getColor('fade1'), ";padding:0 ", fieldVars.xPadding, ";min-height:", componentVars.textBoxHeight, ";border-radius:", getRadius('s'), ";width:100%;text-align:left;&:hover{background:", getColor('fade2'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/select.jsx"],"names":[],"mappings":"AAoBmC","file":"../../src/components/select.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { Box, Icon, Text, TextButton } from '.';\nimport { IconChevronDown } from '../icons';\nimport { Listbox } from '@headlessui/react';\nimport { getColor, getRadius } from '../utilities';\nimport { defaultProps, types } from '../types';\nimport {\n  componentVars,\n  FieldHint,\n  FieldLabel,\n  fieldVars,\n  MenuCard,\n  MenuItem,\n  styles,\n} from '../foundational';\nimport { useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\n\nconst TriggerButton = styled.button`\n  ${styles.buttonReset};\n  ${styles.transitions};\n\n  color: inherit;\n  display: flex;\n  align-items: center;\n  gap: 8px;\n  border: none;\n  background: ${getColor('fade1')};\n  padding: 0 ${fieldVars.xPadding};\n  min-height: ${componentVars.textBoxHeight};\n  border-radius: ${getRadius('s')};\n  width: 100%;\n  text-align: left;\n\n  &:hover {\n    background: ${getColor('fade2')};\n  }\n`;\n\nconst Trigger = forwardRef(({ children, triggerVariant, ...props }, ref) => (\n  <>\n    {triggerVariant === 'box' && (\n      <TriggerButton ref={ref} type=\"button\" {...props}>\n        <Text hasEllipsis>{children}</Text>\n        <Box marginLeft=\"auto\">\n          <Icon icon={<IconChevronDown />} />\n        </Box>\n      </TriggerButton>\n    )}\n    {triggerVariant === 'text' && (\n      <TextButton ref={ref} icon={<IconChevronDown />} iconPosition=\"right\" {...props}>\n        {children}\n      </TextButton>\n    )}\n  </>\n));\n\nconst getTriggerContent = (isMultiple, selected) => {\n  if (isMultiple) {\n    return selected.map(option => option.content).join(', ');\n  }\n  return selected.content;\n};\n\nconst SelectButton = ({ customTrigger, triggerVariant, isMultiple, selected }) => {\n  const trigger = getTriggerContent(isMultiple, selected);\n  return (\n    <Listbox.Button as={Fragment}>\n      {customTrigger ? (\n        customTrigger(selected)\n      ) : (\n        <Trigger triggerVariant={triggerVariant}>{trigger}</Trigger>\n      )}\n    </Listbox.Button>\n  );\n};\n\nconst Root = styled.div`\n  min-width: 0px;\n`;\n\nconst SelectMenuOption = ({ option, isSelected, menuItemsHaveEllipsis, ...props }) => (\n  <Listbox.Option value={option} as={Fragment} disabled={option.isDisabled} {...props}>\n    {({ active }) => {\n      return (\n        <MenuItem\n          isSelected={isSelected}\n          isActive={active}\n          isDisabled={option.isDisabled}\n          hasEllipsis={menuItemsHaveEllipsis}\n        >\n          {option.content}\n        </MenuItem>\n      );\n    }}\n  </Listbox.Option>\n);\n\nconst SelectInfo = ({ label, hint }) => (\n  <>\n    {(label || hint) && (\n      <Box marginBottom=\"betweenFormControlAndLabel\">\n        {label && <Listbox.Label as={FieldLabel}>{label}</Listbox.Label>}\n        {hint && <FieldHint>{hint}</FieldHint>}\n      </Box>\n    )}\n  </>\n);\n\nconst SelectMenu = ({ children, floating, hasPortal, ...props }) => {\n  const Root = hasPortal ? FloatingPortal : Fragment;\n  return (\n    <Root>\n      <Listbox.Options static className=\"grn-context\" ref={floating} as={MenuCard} {...props}>\n        {children}\n      </Listbox.Options>\n    </Root>\n  );\n};\n\nconst getIsSelected = (isMultiple, option, selected) => {\n  if (isMultiple) {\n    return selected.some(selectedOption => selectedOption.content === option.content);\n  }\n\n  return option.value === selected.value;\n};\n\nexport const Select = ({\n  options,\n  value,\n  onChange,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  trigger: customTrigger,\n  triggerVariant = 'box',\n  label,\n  hint,\n  menuZIndex,\n  menuItemsHaveEllipsis = true,\n  allowMultiple,\n  hasPortal = true,\n}) => {\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n  const rootWidth = triggerVariant === 'text' ? 'fit-content' : undefined;\n\n  const optionMap = useMemo(() => {\n    return new Map(options.map(option => [option.value, option]));\n  }, [options]);\n\n  const selected = useMemo(\n    () => (allowMultiple ? value.map(value => optionMap.get(value)) : optionMap.get(value)),\n    [allowMultiple, optionMap, value],\n  );\n\n  return (\n    <Listbox\n      as={Root}\n      value={selected}\n      onChange={onChange}\n      width={rootWidth}\n      multiple={allowMultiple}\n    >\n      {({ open }) => (\n        <>\n          <SelectInfo label={label} hint={hint} />\n\n          <Box position=\"relative\" ref={reference}>\n            <SelectButton\n              customTrigger={customTrigger}\n              triggerVariant={triggerVariant}\n              isMultiple={allowMultiple}\n              selected={selected}\n            />\n\n            <SelectMenu\n              isOpen={open}\n              placement={menuPlacement}\n              maxHeight={menuMaxHeight}\n              zIndex={menuZIndex}\n              floating={floating}\n              style={floatingStyles}\n              hasPortal={hasPortal}\n            >\n              {options.map((option, index) => {\n                const isSelected = getIsSelected(allowMultiple, option, selected);\n                return (\n                  <SelectMenuOption\n                    isSelected={isSelected}\n                    key={index}\n                    option={option}\n                    hasEllipsis={menuItemsHaveEllipsis}\n                  />\n                );\n              })}\n            </SelectMenu>\n          </Box>\n        </>\n      )}\n    </Listbox>\n  );\n};\n\nSelect.Label = FieldLabel;\nSelect.Hint = FieldHint;\n\nSelect.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.oneOfType([PropTypes.string, PropTypes.array]).isRequired,\n  onChange: PropTypes.func.isRequired,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuZIndex: types.zIndex,\n  menuMaxHeight: types.dimension,\n  trigger: PropTypes.func,\n  triggerVariant: PropTypes.oneOf(['box', 'text']),\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  allowMultiple: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n};\n"]} */"));
41
+ })(styles.buttonReset, ";", styles.transitions, ";color:inherit;display:flex;align-items:center;gap:8px;border:none;background:", getColor('fade1'), ";padding:0 ", fieldVars.xPadding, ";min-height:", componentVars.textBoxHeight, ";border-radius:", getRadius('s'), ";width:100%;text-align:left;&:hover{background:", getColor('fade2'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/select.jsx"],"names":[],"mappings":"AAoBmC","file":"../../src/components/select.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { Box, Icon, Text, TextButton } from '.';\nimport { IconChevronDown } from '../icons';\nimport { Listbox } from '@headlessui/react';\nimport { getColor, getRadius } from '../utilities';\nimport { defaultProps, types } from '../types';\nimport {\n  componentVars,\n  FieldHint,\n  FieldLabel,\n  fieldVars,\n  MenuCard,\n  MenuItem,\n  styles,\n} from '../foundational';\nimport { useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\n\nconst TriggerButton = styled.button`\n  ${styles.buttonReset};\n  ${styles.transitions};\n\n  color: inherit;\n  display: flex;\n  align-items: center;\n  gap: 8px;\n  border: none;\n  background: ${getColor('fade1')};\n  padding: 0 ${fieldVars.xPadding};\n  min-height: ${componentVars.textBoxHeight};\n  border-radius: ${getRadius('s')};\n  width: 100%;\n  text-align: left;\n\n  &:hover {\n    background: ${getColor('fade2')};\n  }\n`;\n\nconst Trigger = forwardRef(({ children, triggerVariant, ...props }, ref) => (\n  <>\n    {triggerVariant === 'box' && (\n      <TriggerButton ref={ref} type=\"button\" {...props}>\n        <Text hasEllipsis>{children}</Text>\n        <Box marginLeft=\"auto\">\n          <Icon icon={<IconChevronDown />} />\n        </Box>\n      </TriggerButton>\n    )}\n    {triggerVariant === 'text' && (\n      <TextButton ref={ref} icon={<IconChevronDown />} iconPosition=\"right\" {...props}>\n        {children}\n      </TextButton>\n    )}\n  </>\n));\n\nconst getTriggerContent = (isMultiple, selected) => {\n  if (!selected) return null;\n  if (isMultiple) {\n    return selected.map(option => option.content).join(', ');\n  }\n  return selected.content;\n};\n\nconst SelectButton = ({ customTrigger, triggerVariant, isMultiple, selected }) => {\n  const trigger = getTriggerContent(isMultiple, selected);\n  return (\n    <Listbox.Button as={Fragment}>\n      {customTrigger ? (\n        customTrigger(selected)\n      ) : (\n        <Trigger triggerVariant={triggerVariant}>{trigger}</Trigger>\n      )}\n    </Listbox.Button>\n  );\n};\n\nconst Root = styled.div`\n  min-width: 0px;\n`;\n\nconst SelectMenuOption = ({\n  option,\n  isSelected,\n  menuItemsHaveEllipsis,\n  allowMultiple,\n  ...props\n}) => (\n  <Listbox.Option value={option} as={Fragment} disabled={option.isDisabled} {...props}>\n    {({ active }) => {\n      return (\n        <MenuItem\n          isSelected={isSelected}\n          isActive={active}\n          isDisabled={option.isDisabled}\n          hasEllipsis={menuItemsHaveEllipsis}\n          checkVariant={allowMultiple ? 'checkbox' : 'check'}\n        >\n          {option.content}\n        </MenuItem>\n      );\n    }}\n  </Listbox.Option>\n);\n\nconst SelectInfo = ({ label, hint }) => (\n  <>\n    {(label || hint) && (\n      <Box marginBottom=\"betweenFormControlAndLabel\">\n        {label && <Listbox.Label as={FieldLabel}>{label}</Listbox.Label>}\n        {hint && <FieldHint>{hint}</FieldHint>}\n      </Box>\n    )}\n  </>\n);\n\nconst SelectMenu = ({ children, floating, hasPortal, ...props }) => {\n  const Root = hasPortal ? FloatingPortal : Fragment;\n  return (\n    <Root>\n      <Listbox.Options static className=\"grn-context\" ref={floating} as={MenuCard} {...props}>\n        {children}\n      </Listbox.Options>\n    </Root>\n  );\n};\n\nconst getIsSelected = (isMultiple, option, selected) => {\n  if (!selected) return null;\n  if (isMultiple) {\n    return selected.some(selectedOption => selectedOption.content === option.content);\n  }\n\n  return option.value === selected.value;\n};\n\nexport const Select = ({\n  options,\n  value,\n  onChange,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  trigger: customTrigger,\n  triggerVariant = 'box',\n  label,\n  hint,\n  menuZIndex,\n  menuItemsHaveEllipsis = true,\n  allowMultiple,\n  hasPortal = true,\n}) => {\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n  const rootWidth = triggerVariant === 'text' ? 'fit-content' : undefined;\n\n  const optionMap = useMemo(() => {\n    return new Map(options.map(option => [option.value, option]));\n  }, [options]);\n\n  const selected = useMemo(\n    () => (allowMultiple ? value.map(value => optionMap.get(value)) : optionMap.get(value)),\n    [allowMultiple, optionMap, value],\n  );\n\n  return (\n    <Listbox\n      as={Root}\n      value={selected}\n      onChange={onChange}\n      width={rootWidth}\n      multiple={allowMultiple}\n    >\n      {({ open }) => (\n        <>\n          <SelectInfo label={label} hint={hint} />\n\n          <Box position=\"relative\" ref={reference}>\n            <SelectButton\n              customTrigger={customTrigger}\n              triggerVariant={triggerVariant}\n              isMultiple={allowMultiple}\n              selected={selected}\n            />\n\n            <SelectMenu\n              isOpen={open}\n              placement={menuPlacement}\n              maxHeight={menuMaxHeight}\n              zIndex={menuZIndex}\n              floating={floating}\n              style={floatingStyles}\n              hasPortal={hasPortal}\n            >\n              {options.map((option, index) => {\n                const isSelected = getIsSelected(allowMultiple, option, selected);\n                return (\n                  <SelectMenuOption\n                    isSelected={isSelected}\n                    key={index}\n                    option={option}\n                    hasEllipsis={menuItemsHaveEllipsis}\n                    allowMultiple={allowMultiple}\n                  />\n                );\n              })}\n            </SelectMenu>\n          </Box>\n        </>\n      )}\n    </Listbox>\n  );\n};\n\nSelect.Label = FieldLabel;\nSelect.Hint = FieldHint;\n\nSelect.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.oneOfType([PropTypes.string, PropTypes.array]).isRequired,\n  onChange: PropTypes.func.isRequired,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuZIndex: types.zIndex,\n  menuMaxHeight: types.dimension,\n  trigger: PropTypes.func,\n  triggerVariant: PropTypes.oneOf(['box', 'text']),\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  allowMultiple: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n};\n"]} */"));
42
42
 
43
43
  var Trigger = /*#__PURE__*/forwardRef(function (_ref, ref) {
44
44
  var children = _ref.children,
@@ -62,6 +62,8 @@ var Trigger = /*#__PURE__*/forwardRef(function (_ref, ref) {
62
62
  });
63
63
 
64
64
  var getTriggerContent = function getTriggerContent(isMultiple, selected) {
65
+ if (!selected) return null;
66
+
65
67
  if (isMultiple) {
66
68
  return selected.map(function (option) {
67
69
  return option.content;
@@ -95,7 +97,7 @@ var Root = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "production" ? {
95
97
  } : {
96
98
  name: "wowqs1",
97
99
  styles: "min-width:0px",
98
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/select.jsx"],"names":[],"mappings":"AA+EuB","file":"../../src/components/select.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { Box, Icon, Text, TextButton } from '.';\nimport { IconChevronDown } from '../icons';\nimport { Listbox } from '@headlessui/react';\nimport { getColor, getRadius } from '../utilities';\nimport { defaultProps, types } from '../types';\nimport {\n  componentVars,\n  FieldHint,\n  FieldLabel,\n  fieldVars,\n  MenuCard,\n  MenuItem,\n  styles,\n} from '../foundational';\nimport { useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\n\nconst TriggerButton = styled.button`\n  ${styles.buttonReset};\n  ${styles.transitions};\n\n  color: inherit;\n  display: flex;\n  align-items: center;\n  gap: 8px;\n  border: none;\n  background: ${getColor('fade1')};\n  padding: 0 ${fieldVars.xPadding};\n  min-height: ${componentVars.textBoxHeight};\n  border-radius: ${getRadius('s')};\n  width: 100%;\n  text-align: left;\n\n  &:hover {\n    background: ${getColor('fade2')};\n  }\n`;\n\nconst Trigger = forwardRef(({ children, triggerVariant, ...props }, ref) => (\n  <>\n    {triggerVariant === 'box' && (\n      <TriggerButton ref={ref} type=\"button\" {...props}>\n        <Text hasEllipsis>{children}</Text>\n        <Box marginLeft=\"auto\">\n          <Icon icon={<IconChevronDown />} />\n        </Box>\n      </TriggerButton>\n    )}\n    {triggerVariant === 'text' && (\n      <TextButton ref={ref} icon={<IconChevronDown />} iconPosition=\"right\" {...props}>\n        {children}\n      </TextButton>\n    )}\n  </>\n));\n\nconst getTriggerContent = (isMultiple, selected) => {\n  if (isMultiple) {\n    return selected.map(option => option.content).join(', ');\n  }\n  return selected.content;\n};\n\nconst SelectButton = ({ customTrigger, triggerVariant, isMultiple, selected }) => {\n  const trigger = getTriggerContent(isMultiple, selected);\n  return (\n    <Listbox.Button as={Fragment}>\n      {customTrigger ? (\n        customTrigger(selected)\n      ) : (\n        <Trigger triggerVariant={triggerVariant}>{trigger}</Trigger>\n      )}\n    </Listbox.Button>\n  );\n};\n\nconst Root = styled.div`\n  min-width: 0px;\n`;\n\nconst SelectMenuOption = ({ option, isSelected, menuItemsHaveEllipsis, ...props }) => (\n  <Listbox.Option value={option} as={Fragment} disabled={option.isDisabled} {...props}>\n    {({ active }) => {\n      return (\n        <MenuItem\n          isSelected={isSelected}\n          isActive={active}\n          isDisabled={option.isDisabled}\n          hasEllipsis={menuItemsHaveEllipsis}\n        >\n          {option.content}\n        </MenuItem>\n      );\n    }}\n  </Listbox.Option>\n);\n\nconst SelectInfo = ({ label, hint }) => (\n  <>\n    {(label || hint) && (\n      <Box marginBottom=\"betweenFormControlAndLabel\">\n        {label && <Listbox.Label as={FieldLabel}>{label}</Listbox.Label>}\n        {hint && <FieldHint>{hint}</FieldHint>}\n      </Box>\n    )}\n  </>\n);\n\nconst SelectMenu = ({ children, floating, hasPortal, ...props }) => {\n  const Root = hasPortal ? FloatingPortal : Fragment;\n  return (\n    <Root>\n      <Listbox.Options static className=\"grn-context\" ref={floating} as={MenuCard} {...props}>\n        {children}\n      </Listbox.Options>\n    </Root>\n  );\n};\n\nconst getIsSelected = (isMultiple, option, selected) => {\n  if (isMultiple) {\n    return selected.some(selectedOption => selectedOption.content === option.content);\n  }\n\n  return option.value === selected.value;\n};\n\nexport const Select = ({\n  options,\n  value,\n  onChange,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  trigger: customTrigger,\n  triggerVariant = 'box',\n  label,\n  hint,\n  menuZIndex,\n  menuItemsHaveEllipsis = true,\n  allowMultiple,\n  hasPortal = true,\n}) => {\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n  const rootWidth = triggerVariant === 'text' ? 'fit-content' : undefined;\n\n  const optionMap = useMemo(() => {\n    return new Map(options.map(option => [option.value, option]));\n  }, [options]);\n\n  const selected = useMemo(\n    () => (allowMultiple ? value.map(value => optionMap.get(value)) : optionMap.get(value)),\n    [allowMultiple, optionMap, value],\n  );\n\n  return (\n    <Listbox\n      as={Root}\n      value={selected}\n      onChange={onChange}\n      width={rootWidth}\n      multiple={allowMultiple}\n    >\n      {({ open }) => (\n        <>\n          <SelectInfo label={label} hint={hint} />\n\n          <Box position=\"relative\" ref={reference}>\n            <SelectButton\n              customTrigger={customTrigger}\n              triggerVariant={triggerVariant}\n              isMultiple={allowMultiple}\n              selected={selected}\n            />\n\n            <SelectMenu\n              isOpen={open}\n              placement={menuPlacement}\n              maxHeight={menuMaxHeight}\n              zIndex={menuZIndex}\n              floating={floating}\n              style={floatingStyles}\n              hasPortal={hasPortal}\n            >\n              {options.map((option, index) => {\n                const isSelected = getIsSelected(allowMultiple, option, selected);\n                return (\n                  <SelectMenuOption\n                    isSelected={isSelected}\n                    key={index}\n                    option={option}\n                    hasEllipsis={menuItemsHaveEllipsis}\n                  />\n                );\n              })}\n            </SelectMenu>\n          </Box>\n        </>\n      )}\n    </Listbox>\n  );\n};\n\nSelect.Label = FieldLabel;\nSelect.Hint = FieldHint;\n\nSelect.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.oneOfType([PropTypes.string, PropTypes.array]).isRequired,\n  onChange: PropTypes.func.isRequired,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuZIndex: types.zIndex,\n  menuMaxHeight: types.dimension,\n  trigger: PropTypes.func,\n  triggerVariant: PropTypes.oneOf(['box', 'text']),\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  allowMultiple: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n};\n"]} */",
100
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/select.jsx"],"names":[],"mappings":"AAgFuB","file":"../../src/components/select.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { Box, Icon, Text, TextButton } from '.';\nimport { IconChevronDown } from '../icons';\nimport { Listbox } from '@headlessui/react';\nimport { getColor, getRadius } from '../utilities';\nimport { defaultProps, types } from '../types';\nimport {\n  componentVars,\n  FieldHint,\n  FieldLabel,\n  fieldVars,\n  MenuCard,\n  MenuItem,\n  styles,\n} from '../foundational';\nimport { useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\n\nconst TriggerButton = styled.button`\n  ${styles.buttonReset};\n  ${styles.transitions};\n\n  color: inherit;\n  display: flex;\n  align-items: center;\n  gap: 8px;\n  border: none;\n  background: ${getColor('fade1')};\n  padding: 0 ${fieldVars.xPadding};\n  min-height: ${componentVars.textBoxHeight};\n  border-radius: ${getRadius('s')};\n  width: 100%;\n  text-align: left;\n\n  &:hover {\n    background: ${getColor('fade2')};\n  }\n`;\n\nconst Trigger = forwardRef(({ children, triggerVariant, ...props }, ref) => (\n  <>\n    {triggerVariant === 'box' && (\n      <TriggerButton ref={ref} type=\"button\" {...props}>\n        <Text hasEllipsis>{children}</Text>\n        <Box marginLeft=\"auto\">\n          <Icon icon={<IconChevronDown />} />\n        </Box>\n      </TriggerButton>\n    )}\n    {triggerVariant === 'text' && (\n      <TextButton ref={ref} icon={<IconChevronDown />} iconPosition=\"right\" {...props}>\n        {children}\n      </TextButton>\n    )}\n  </>\n));\n\nconst getTriggerContent = (isMultiple, selected) => {\n  if (!selected) return null;\n  if (isMultiple) {\n    return selected.map(option => option.content).join(', ');\n  }\n  return selected.content;\n};\n\nconst SelectButton = ({ customTrigger, triggerVariant, isMultiple, selected }) => {\n  const trigger = getTriggerContent(isMultiple, selected);\n  return (\n    <Listbox.Button as={Fragment}>\n      {customTrigger ? (\n        customTrigger(selected)\n      ) : (\n        <Trigger triggerVariant={triggerVariant}>{trigger}</Trigger>\n      )}\n    </Listbox.Button>\n  );\n};\n\nconst Root = styled.div`\n  min-width: 0px;\n`;\n\nconst SelectMenuOption = ({\n  option,\n  isSelected,\n  menuItemsHaveEllipsis,\n  allowMultiple,\n  ...props\n}) => (\n  <Listbox.Option value={option} as={Fragment} disabled={option.isDisabled} {...props}>\n    {({ active }) => {\n      return (\n        <MenuItem\n          isSelected={isSelected}\n          isActive={active}\n          isDisabled={option.isDisabled}\n          hasEllipsis={menuItemsHaveEllipsis}\n          checkVariant={allowMultiple ? 'checkbox' : 'check'}\n        >\n          {option.content}\n        </MenuItem>\n      );\n    }}\n  </Listbox.Option>\n);\n\nconst SelectInfo = ({ label, hint }) => (\n  <>\n    {(label || hint) && (\n      <Box marginBottom=\"betweenFormControlAndLabel\">\n        {label && <Listbox.Label as={FieldLabel}>{label}</Listbox.Label>}\n        {hint && <FieldHint>{hint}</FieldHint>}\n      </Box>\n    )}\n  </>\n);\n\nconst SelectMenu = ({ children, floating, hasPortal, ...props }) => {\n  const Root = hasPortal ? FloatingPortal : Fragment;\n  return (\n    <Root>\n      <Listbox.Options static className=\"grn-context\" ref={floating} as={MenuCard} {...props}>\n        {children}\n      </Listbox.Options>\n    </Root>\n  );\n};\n\nconst getIsSelected = (isMultiple, option, selected) => {\n  if (!selected) return null;\n  if (isMultiple) {\n    return selected.some(selectedOption => selectedOption.content === option.content);\n  }\n\n  return option.value === selected.value;\n};\n\nexport const Select = ({\n  options,\n  value,\n  onChange,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  trigger: customTrigger,\n  triggerVariant = 'box',\n  label,\n  hint,\n  menuZIndex,\n  menuItemsHaveEllipsis = true,\n  allowMultiple,\n  hasPortal = true,\n}) => {\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n  const rootWidth = triggerVariant === 'text' ? 'fit-content' : undefined;\n\n  const optionMap = useMemo(() => {\n    return new Map(options.map(option => [option.value, option]));\n  }, [options]);\n\n  const selected = useMemo(\n    () => (allowMultiple ? value.map(value => optionMap.get(value)) : optionMap.get(value)),\n    [allowMultiple, optionMap, value],\n  );\n\n  return (\n    <Listbox\n      as={Root}\n      value={selected}\n      onChange={onChange}\n      width={rootWidth}\n      multiple={allowMultiple}\n    >\n      {({ open }) => (\n        <>\n          <SelectInfo label={label} hint={hint} />\n\n          <Box position=\"relative\" ref={reference}>\n            <SelectButton\n              customTrigger={customTrigger}\n              triggerVariant={triggerVariant}\n              isMultiple={allowMultiple}\n              selected={selected}\n            />\n\n            <SelectMenu\n              isOpen={open}\n              placement={menuPlacement}\n              maxHeight={menuMaxHeight}\n              zIndex={menuZIndex}\n              floating={floating}\n              style={floatingStyles}\n              hasPortal={hasPortal}\n            >\n              {options.map((option, index) => {\n                const isSelected = getIsSelected(allowMultiple, option, selected);\n                return (\n                  <SelectMenuOption\n                    isSelected={isSelected}\n                    key={index}\n                    option={option}\n                    hasEllipsis={menuItemsHaveEllipsis}\n                    allowMultiple={allowMultiple}\n                  />\n                );\n              })}\n            </SelectMenu>\n          </Box>\n        </>\n      )}\n    </Listbox>\n  );\n};\n\nSelect.Label = FieldLabel;\nSelect.Hint = FieldHint;\n\nSelect.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.oneOfType([PropTypes.string, PropTypes.array]).isRequired,\n  onChange: PropTypes.func.isRequired,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuZIndex: types.zIndex,\n  menuMaxHeight: types.dimension,\n  trigger: PropTypes.func,\n  triggerVariant: PropTypes.oneOf(['box', 'text']),\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  allowMultiple: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n};\n"]} */",
99
101
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
100
102
  });
101
103
 
@@ -103,6 +105,7 @@ var SelectMenuOption = function SelectMenuOption(_ref3) {
103
105
  var option = _ref3.option,
104
106
  isSelected = _ref3.isSelected,
105
107
  menuItemsHaveEllipsis = _ref3.menuItemsHaveEllipsis,
108
+ allowMultiple = _ref3.allowMultiple,
106
109
  props = _objectWithoutProperties(_ref3, _excluded2);
107
110
 
108
111
  return ___EmotionJSX(Listbox.Option, _extends({
@@ -115,7 +118,8 @@ var SelectMenuOption = function SelectMenuOption(_ref3) {
115
118
  isSelected: isSelected,
116
119
  isActive: active,
117
120
  isDisabled: option.isDisabled,
118
- hasEllipsis: menuItemsHaveEllipsis
121
+ hasEllipsis: menuItemsHaveEllipsis,
122
+ checkVariant: allowMultiple ? 'checkbox' : 'check'
119
123
  }, option.content);
120
124
  });
121
125
  };
@@ -146,6 +150,8 @@ var SelectMenu = function SelectMenu(_ref6) {
146
150
  };
147
151
 
148
152
  var getIsSelected = function getIsSelected(isMultiple, option, selected) {
153
+ if (!selected) return null;
154
+
149
155
  if (isMultiple) {
150
156
  return selected.some(function (selectedOption) {
151
157
  return selectedOption.content === option.content;
@@ -228,7 +234,8 @@ export var Select = function Select(_ref7) {
228
234
  isSelected: isSelected,
229
235
  key: index,
230
236
  option: option,
231
- hasEllipsis: menuItemsHaveEllipsis
237
+ hasEllipsis: menuItemsHaveEllipsis,
238
+ allowMultiple: allowMultiple
232
239
  });
233
240
  }))));
234
241
  });
@@ -3,8 +3,8 @@ import "core-js/modules/es.array.index-of.js";
3
3
  import "core-js/modules/es.symbol.js";
4
4
  import "core-js/modules/es.object.define-property.js";
5
5
  var _excluded = ["children", "zIndex", "placement", "maxHeight", "isOpen"],
6
- _excluded2 = ["children", "isDisabled", "hasIcon", "isSelected", "color"],
7
- _excluded3 = ["children", "isSelected", "isActive", "isDisabled", "icon", "variant", "hasDivider", "hasEllipsis"];
6
+ _excluded2 = ["children", "isDisabled", "hasIcon", "isSelected", "color", "columns"],
7
+ _excluded3 = ["children", "isSelected", "isActive", "isDisabled", "icon", "variant", "hasDivider", "hasEllipsis", "checkVariant"];
8
8
 
9
9
  function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
10
10
 
@@ -16,9 +16,10 @@ function _objectWithoutPropertiesLoose(source, excluded) { if (source == null) r
16
16
 
17
17
  import "core-js/modules/es.object.assign.js";
18
18
  import React, { forwardRef } from 'react';
19
+ import PropTypes from 'prop-types';
19
20
  import { css } from '@emotion/react';
20
21
  import { getColor, getDimension, getRadius, getShadow, getTransition } from '../utilities';
21
- import { Box, Icon, Text } from '../components';
22
+ import { Box, Checkbox, Icon, Text } from '../components';
22
23
  import { IconCheck } from '../icons';
23
24
  import { componentVars } from './styles';
24
25
  import { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';
@@ -92,7 +93,7 @@ export var MenuCard = /*#__PURE__*/forwardRef(function (_ref3, ref) {
92
93
 
93
94
  return ___EmotionJSX("ul", _extends({
94
95
  ref: ref,
95
- css: /*#__PURE__*/css("padding:", cardPaddingPx, " 0;margin:0;background:", getColor('floatingBackground'), ";z-index:", zIndex, ";box-shadow:", getShadow('m'), ";border-radius:", getRadius('s'), ";overflow:auto;max-height:", maxHeight, ";", !isOpen && "opacity: 0; pointer-events: none;", ";transition-property:opacity;transition-duration:", getTransition(isOpen ? 'xxFast' : 'xFast'), ";&:focus{outline:none;}" + (process.env.NODE_ENV === "production" ? "" : ";label:MenuCard;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9mb3VuZGF0aW9uYWwvbWVudS5qc3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBZ0VjIiwiZmlsZSI6Ii4uLy4uL3NyYy9mb3VuZGF0aW9uYWwvbWVudS5qc3giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QsIHsgZm9yd2FyZFJlZiB9IGZyb20gJ3JlYWN0JztcbmltcG9ydCB7IGNzcyB9IGZyb20gJ0BlbW90aW9uL3JlYWN0JztcbmltcG9ydCB7IGdldENvbG9yLCBnZXREaW1lbnNpb24sIGdldFJhZGl1cywgZ2V0U2hhZG93LCBnZXRUcmFuc2l0aW9uIH0gZnJvbSAnLi4vdXRpbGl0aWVzJztcbmltcG9ydCB7IEJveCwgSWNvbiwgVGV4dCB9IGZyb20gJy4uL2NvbXBvbmVudHMnO1xuaW1wb3J0IHsgSWNvbkNoZWNrIH0gZnJvbSAnLi4vaWNvbnMnO1xuaW1wb3J0IHsgY29tcG9uZW50VmFycyB9IGZyb20gJy4vc3R5bGVzJztcbmltcG9ydCB7IGF1dG9VcGRhdGUsIGZsaXAsIHNpemUsIHVzZUZsb2F0aW5nIH0gZnJvbSAnQGZsb2F0aW5nLXVpL3JlYWN0LWRvbS1pbnRlcmFjdGlvbnMnO1xuXG5jb25zdCBjYXJkUGFkZGluZyA9IDg7XG5jb25zdCBjYXJkUGFkZGluZ1B4ID0gYCR7Y2FyZFBhZGRpbmd9cHhgO1xuY29uc3QgaXRlbVlQYWRkaW5nID0gMTA7XG5jb25zdCBpdGVtWFBhZGRpbmcgPSAxMjtcblxuZXhwb3J0IGNvbnN0IHBsYWNlbWVudHNNYXAgPSB7XG4gIHRvcDogJ3RvcCcsXG4gIHRvcFN0YXJ0OiAndG9wLXN0YXJ0JyxcbiAgdG9wRW5kOiAndG9wLWVuZCcsXG4gIGJvdHRvbTogJ2JvdHRvbScsXG4gIGJvdHRvbVN0YXJ0OiAnYm90dG9tLXN0YXJ0JyxcbiAgYm90dG9tRW5kOiAnYm90dG9tLWVuZCcsXG4gIHJpZ2h0OiAncmlnaHQnLFxuICByaWdodFN0YXJ0OiAncmlnaHQtc3RhcnQnLFxuICByaWdodEVuZDogJ3JpZ2h0LWVuZCcsXG4gIGxlZnQ6ICdsZWZ0JyxcbiAgbGVmdFN0YXJ0OiAnbGVmdC1zdGFydCcsXG4gIGxlZnRFbmQ6ICdsZWZ0LWVuZCcsXG59O1xuXG5leHBvcnQgY29uc3QgdXNlTWVudVBvc2l0aW9uID0gKHsgbWVudVdpZHRoLCBtZW51UGxhY2VtZW50IH0pID0+IHtcbiAgY29uc3QgeyB4LCB5LCByZWZlcmVuY2UsIGZsb2F0aW5nLCBzdHJhdGVneSB9ID0gdXNlRmxvYXRpbmcoe1xuICAgIHBsYWNlbWVudDogcGxhY2VtZW50c01hcFttZW51UGxhY2VtZW50XSxcbiAgICB3aGlsZUVsZW1lbnRzTW91bnRlZDogYXV0b1VwZGF0ZSxcbiAgICBtaWRkbGV3YXJlOiBbXG4gICAgICBmbGlwKHsgcGFkZGluZzogMjQsIGZsaXBBbGlnbm1lbnQ6IGZhbHNlIH0pLFxuICAgICAgc2l6ZSh7XG4gICAgICAgIGFwcGx5KHsgcmVjdHMsIGVsZW1lbnRzIH0pIHtcbiAgICAgICAgICBjb25zdCB3aWR0aCA9IGdldERpbWVuc2lvbihtZW51V2lkdGgpIHx8IGAke3JlY3RzLnJlZmVyZW5jZS53aWR0aH1weGA7XG4gICAgICAgICAgT2JqZWN0LmFzc2lnbihlbGVtZW50cy5mbG9hdGluZy5zdHlsZSwge1xuICAgICAgICAgICAgd2lkdGg6IHdpZHRoLFxuICAgICAgICAgIH0pO1xuICAgICAgICB9LFxuICAgICAgICBwYWRkaW5nOiAyNCxcbiAgICAgIH0pLFxuICAgIF0sXG4gIH0pO1xuXG4gIGNvbnN0IGZsb2F0aW5nU3R5bGVzID0ge1xuICAgIHBvc2l0aW9uOiBzdHJhdGVneSxcbiAgICB0b3A6IHkgPz8gMCxcbiAgICBsZWZ0OiB4ID8/IDAsXG4gICAgekluZGV4OiAxMTAwLFxuICB9O1xuXG4gIHJldHVybiB7XG4gICAgcmVmZXJlbmNlLFxuICAgIGZsb2F0aW5nLFxuICAgIGZsb2F0aW5nU3R5bGVzLFxuICB9O1xufTtcblxuZXhwb3J0IGNvbnN0IE1lbnVDYXJkID0gZm9yd2FyZFJlZihcbiAgKHsgY2hpbGRyZW4sIHpJbmRleCA9IDEsIHBsYWNlbWVudCwgbWF4SGVpZ2h0LCBpc09wZW4sIC4uLnByb3BzIH0sIHJlZikgPT4gKFxuICAgIDx1bFxuICAgICAgcmVmPXtyZWZ9XG4gICAgICBjc3M9e2Nzc2BcbiAgICAgICAgcGFkZGluZzogJHtjYXJkUGFkZGluZ1B4fSAwO1xuICAgICAgICBtYXJnaW46IDA7XG4gICAgICAgIGJhY2tncm91bmQ6ICR7Z2V0Q29sb3IoJ2Zsb2F0aW5nQmFja2dyb3VuZCcpfTtcbiAgICAgICAgei1pbmRleDogJHt6SW5kZXh9O1xuICAgICAgICBib3gtc2hhZG93OiAke2dldFNoYWRvdygnbScpfTtcbiAgICAgICAgYm9yZGVyLXJhZGl1czogJHtnZXRSYWRpdXMoJ3MnKX07XG4gICAgICAgIG92ZXJmbG93OiBhdXRvO1xuICAgICAgICBtYXgtaGVpZ2h0OiAke21heEhlaWdodH07XG4gICAgICAgICR7IWlzT3BlbiAmJiBgb3BhY2l0eTogMDsgcG9pbnRlci1ldmVudHM6IG5vbmU7YH07XG4gICAgICAgIHRyYW5zaXRpb24tcHJvcGVydHk6IG9wYWNpdHk7XG4gICAgICAgIHRyYW5zaXRpb24tZHVyYXRpb246ICR7Z2V0VHJhbnNpdGlvbihpc09wZW4gPyAneHhGYXN0JyA6ICd4RmFzdCcpfTtcblxuICAgICAgICAmOmZvY3VzIHtcbiAgICAgICAgICBvdXRsaW5lOiBub25lO1xuICAgICAgICB9XG4gICAgICBgfVxuICAgICAgey4uLnByb3BzfVxuICAgID5cbiAgICAgIHtjaGlsZHJlbn1cbiAgICA8L3VsPlxuICApLFxuKTtcblxuY29uc3QgdmFyaWFudFN0eWxlcyA9IHtcbiAgbmV1dHJhbDoge1xuICAgIGJhY2tncm91bmRDb2xvckFjdGl2ZTogZ2V0Q29sb3IoJ2ZhZGUxJyksXG4gICAgaWNvbkNvbG9yQWN0aXZlOiBnZXRDb2xvcignY29udGVudCcpLFxuICB9LFxuICBkYW5nZXI6IHtcbiAgICBjb2xvcjogZ2V0Q29sb3IoJ2RhbmdlcicpLFxuICAgIGJhY2tncm91bmRDb2xvckFjdGl2ZTogY29tcG9uZW50VmFycy5jbGVhckJ1dHRvbkRhbmdlckJhY2tncm91bmRIb3ZlcixcbiAgfSxcbn07XG5cbmNvbnN0IE1lbnVJdGVtUm9vdCA9IGZvcndhcmRSZWYoXG4gICh7IGNoaWxkcmVuLCBpc0Rpc2FibGVkLCBoYXNJY29uLCBpc1NlbGVjdGVkLCBjb2xvciwgLi4ucHJvcHMgfSwgcmVmKSA9PiAoXG4gICAgPGxpXG4gICAgICByZWY9e3JlZn1cbiAgICAgIGNzcz17Y3NzYFxuICAgICAgICBkaXNwbGF5OiBncmlkO1xuICAgICAgICBncmlkLWF1dG8tZmxvdzogY29sdW1uO1xuICAgICAgICBncmlkLXRlbXBsYXRlLWNvbHVtbnM6ICR7aGFzSWNvbiA/ICdhdXRvIDFmcicgOiAnMWZyIGF1dG8nfTtcbiAgICAgICAgZ2FwOiA4cHg7XG4gICAgICAgICR7aXNEaXNhYmxlZCAmJiBgY29sb3I6ICR7Z2V0Q29sb3IoJ2Rpc2FibGVkQ29udGVudCcpfWB9O1xuICAgICAgICAkeyFpc0Rpc2FibGVkICYmIGBjdXJzb3I6IHBvaW50ZXI7YH07XG4gICAgICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gICAgICAgIGp1c3RpZnktY29udGVudDogc3RhcnQ7XG4gICAgICAgIGxpc3Qtc3R5bGU6IG5vbmU7XG4gICAgICAgIHBhZGRpbmc6ICR7aXRlbVlQYWRkaW5nfXB4ICR7aXRlbVhQYWRkaW5nfXB4O1xuICAgICAgICBtYXJnaW46IDAgJHtjYXJkUGFkZGluZ31weDtcbiAgICAgICAgbWluLWhlaWdodDogJHtjb21wb25lbnRWYXJzLnRleHRCb3hIZWlnaHR9O1xuICAgICAgICBib3JkZXItcmFkaXVzOiAke2dldFJhZGl1cygncycpfTtcbiAgICAgICAgYXBwZWFyYW5jZTogbm9uZTtcbiAgICAgICAgY29sb3I6ICR7Y29sb3J9O1xuXG4gICAgICAgIC5JY29uIHtcbiAgICAgICAgICB0cmFuc2l0aW9uOiB1bnNldDtcbiAgICAgICAgfVxuICAgICAgYH1cbiAgICAgIHsuLi5wcm9wc31cbiAgICA+XG4gICAgICB7Y2hpbGRyZW59XG4gICAgPC9saT5cbiAgKSxcbik7XG5cbmV4cG9ydCBjb25zdCBNZW51R3JvdXBUaXRsZSA9ICh7IGNoaWxkcmVuLCBoYXNEaXZpZGVyIH0pID0+IChcbiAgPEJveFxuICAgIHBhZGRpbmdYPXtgJHtjYXJkUGFkZGluZyArIGl0ZW1YUGFkZGluZ31weGB9XG4gICAgcGFkZGluZ0JvdHRvbT17Y2FyZFBhZGRpbmdQeH1cbiAgICBtYXJnaW5Ub3A9e2NhcmRQYWRkaW5nUHh9XG4gICAgcGFkZGluZ1RvcD17aGFzRGl2aWRlciA/IGAke2NhcmRQYWRkaW5nICogMn1weGAgOiB1bmRlZmluZWR9XG4gICAgYm9yZGVyU2lkZT17aGFzRGl2aWRlciA/ICd0b3AnIDogdW5kZWZpbmVkfVxuICA+XG4gICAgPFRleHQgdmFyaWFudD1cImNhcHNcIj57Y2hpbGRyZW59PC9UZXh0PlxuICA8L0JveD5cbik7XG5cbmNvbnN0IE1lbnVJdGVtRGl2aWRlciA9ICgpID0+IChcbiAgPEJveCBoZWlnaHQ9e2NvbXBvbmVudFZhcnMuZGl2aWRlclN0cm9rZVNpemV9IG1hcmdpblk9ezF9IGJhY2tncm91bmRDb2xvcj1cImJvcmRlclwiIC8+XG4pO1xuXG5leHBvcnQgY29uc3QgTWVudUl0ZW0gPSBmb3J3YXJkUmVmKFxuICAoXG4gICAge1xuICAgICAgY2hpbGRyZW4sXG4gICAgICBpc1NlbGVjdGVkLFxuICAgICAgaXNBY3RpdmUsXG4gICAgICBpc0Rpc2FibGVkLFxuICAgICAgaWNvbixcbiAgICAgIHZhcmlhbnQgPSAnbmV1dHJhbCcsXG4gICAgICBoYXNEaXZpZGVyLFxuICAgICAgaGFzRWxsaXBzaXMsXG4gICAgICAuLi5wcm9wc1xuICAgIH0sXG4gICAgcmVmLFxuICApID0+IHtcbiAgICByZXR1cm4gKFxuICAgICAgPD5cbiAgICAgICAge2hhc0RpdmlkZXIgJiYgPE1lbnVJdGVtRGl2aWRlciAvPn1cbiAgICAgICAgPE1lbnVJdGVtUm9vdFxuICAgICAgICAgIHJlZj17cmVmfVxuICAgICAgICAgIGlzU2VsZWN0ZWQ9e2lzU2VsZWN0ZWR9XG4gICAgICAgICAgaXNEaXNhYmxlZD17aXNEaXNhYmxlZH1cbiAgICAgICAgICBjb2xvcj17dmFyaWFudFN0eWxlc1t2YXJpYW50XS5jb2xvcn1cbiAgICAgICAgICBoYXNJY29uPXtCb29sZWFuKGljb24pfVxuICAgICAgICAgIHN0eWxlPXt7XG4gICAgICAgICAgICBiYWNrZ3JvdW5kOiBpc0FjdGl2ZSAmJiB2YXJpYW50U3R5bGVzW3ZhcmlhbnRdLmJhY2tncm91bmRDb2xvckFjdGl2ZSxcbiAgICAgICAgICAgIFsnLS1ncm4tY29sb3ItaWNvbiddOiBpc0FjdGl2ZSAmJiB2YXJpYW50U3R5bGVzW3ZhcmlhbnRdLmljb25Db2xvckFjdGl2ZSxcbiAgICAgICAgICB9fVxuICAgICAgICAgIHsuLi5wcm9wc31cbiAgICAgICAgPlxuICAgICAgICAgIHtpY29uICYmIDxJY29uIGljb249e2ljb259IGhhc0V2ZW5Cb3VuZGFyeSBjb2xvcj17dmFyaWFudFN0eWxlc1t2YXJpYW50XS5jb2xvcn0gLz59XG4gICAgICAgICAgPEJveCB3aWR0aD1cIjEwMCVcIiBtaW5XaWR0aD1cIjBweFwiIHRhZz1cInNwYW5cIj5cbiAgICAgICAgICAgIDxUZXh0IGhhc0VsbGlwc2lzPXtoYXNFbGxpcHNpc30+e2NoaWxkcmVufTwvVGV4dD5cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgICA8Qm94IG9wYWNpdHk9e2lzU2VsZWN0ZWQgPyAnMScgOiAnMCd9PlxuICAgICAgICAgICAgPEljb24gaWNvbj17PEljb25DaGVjayAvPn0gLz5cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgPC9NZW51SXRlbVJvb3Q+XG4gICAgICA8Lz5cbiAgICApO1xuICB9LFxuKTtcbiJdfQ== */")
96
+ css: /*#__PURE__*/css("padding:", cardPaddingPx, " 0;margin:0;background:", getColor('floatingBackground'), ";z-index:", zIndex, ";box-shadow:", getShadow('m'), ";border-radius:", getRadius('s'), ";overflow:auto;max-height:", maxHeight, ";", !isOpen && "opacity: 0; pointer-events: none;", ";transition-property:opacity;transition-duration:", getTransition(isOpen ? 'xxFast' : 'xFast'), ";&:focus{outline:none;}" + (process.env.NODE_ENV === "production" ? "" : ";label:MenuCard;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AAiEc","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, { forwardRef } from 'react';\nimport PropTypes from 'prop-types';\nimport { css } from '@emotion/react';\nimport { getColor, getDimension, getRadius, getShadow, getTransition } from '../utilities';\nimport { Box, Checkbox, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { componentVars } from './styles';\nimport { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemYPadding = 10;\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 } = 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    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nexport const MenuCard = forwardRef(\n  ({ children, zIndex = 1, placement, maxHeight, isOpen, ...props }, ref) => (\n    <ul\n      ref={ref}\n      css={css`\n        padding: ${cardPaddingPx} 0;\n        margin: 0;\n        background: ${getColor('floatingBackground')};\n        z-index: ${zIndex};\n        box-shadow: ${getShadow('m')};\n        border-radius: ${getRadius('s')};\n        overflow: auto;\n        max-height: ${maxHeight};\n        ${!isOpen && `opacity: 0; pointer-events: none;`};\n        transition-property: opacity;\n        transition-duration: ${getTransition(isOpen ? 'xxFast' : 'xFast')};\n\n        &:focus {\n          outline: none;\n        }\n      `}\n      {...props}\n    >\n      {children}\n    </ul>\n  ),\n);\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('fade1'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('danger'),\n    backgroundColorActive: componentVars.clearButtonDangerBackgroundHover,\n  },\n};\n\nconst MenuItemRoot = forwardRef(\n  ({ children, isDisabled, hasIcon, isSelected, color, columns, ...props }, ref) => (\n    <li\n      ref={ref}\n      css={css`\n        display: grid;\n        grid-auto-flow: column;\n        gap: 8px;\n        ${isDisabled && `color: ${getColor('disabledContent')}`};\n        ${!isDisabled && `cursor: pointer;`};\n        align-items: center;\n        justify-content: start;\n        list-style: none;\n        padding: ${itemYPadding}px ${itemXPadding}px;\n        margin: 0 ${cardPadding}px;\n        min-height: ${componentVars.textBoxHeight};\n        border-radius: ${getRadius('s')};\n        appearance: none;\n        color: ${color};\n\n        .Icon {\n          transition: unset;\n        }\n      `}\n      {...props}\n    >\n      {children}\n    </li>\n  ),\n);\n\nexport const MenuGroupTitle = ({ children, hasDivider }) => (\n  <Box\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingBottom={cardPaddingPx}\n    marginTop={cardPaddingPx}\n    paddingTop={hasDivider ? `${cardPadding * 2}px` : undefined}\n    borderSide={hasDivider ? 'top' : undefined}\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nconst MenuItemDivider = () => (\n  <Box height={componentVars.dividerStrokeSize} marginY={1} backgroundColor=\"border\" />\n);\n\nexport const MenuItem = forwardRef(\n  (\n    {\n      children,\n      isSelected,\n      isActive,\n      isDisabled,\n      icon,\n      variant = 'neutral',\n      hasDivider,\n      hasEllipsis,\n      checkVariant = 'check',\n      ...props\n    },\n    ref,\n  ) => {\n    const hasCheck = checkVariant === 'check';\n    const hasCheckbox = checkVariant === 'checkbox';\n\n    const getColumns = () => {\n      if (Boolean(icon) || hasCheckbox) return 'auto 1fr';\n      if (hasCheck) return '1fr auto';\n    };\n\n    return (\n      <>\n        {hasDivider && <MenuItemDivider />}\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            ['--grn-color-icon']: isActive && variantStyles[variant].iconColorActive,\n            gridTemplateColumns: getColumns(),\n          }}\n          {...props}\n        >\n          {hasCheckbox && <Checkbox isChecked={isSelected} disabled={isDisabled} />}\n          {icon && <Icon icon={icon} hasEvenBoundary color={variantStyles[variant].color} />}\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 />} />\n            </Box>\n          )}\n        </MenuItemRoot>\n      </>\n    );\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  hasDivider: PropTypes.bool,\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'checkbox', 'none']),\n};\n"]} */")
96
97
  }, props), children);
97
98
  });
98
99
  var variantStyles = {
@@ -111,11 +112,12 @@ var MenuItemRoot = /*#__PURE__*/forwardRef(function (_ref4, ref) {
111
112
  hasIcon = _ref4.hasIcon,
112
113
  isSelected = _ref4.isSelected,
113
114
  color = _ref4.color,
115
+ columns = _ref4.columns,
114
116
  props = _objectWithoutProperties(_ref4, _excluded2);
115
117
 
116
118
  return ___EmotionJSX("li", _extends({
117
119
  ref: ref,
118
- css: /*#__PURE__*/css("display:grid;grid-auto-flow:column;grid-template-columns:", hasIcon ? 'auto 1fr' : '1fr auto', ";gap:8px;", isDisabled && "color: ".concat(getColor('disabledContent')), ";", !isDisabled && "cursor: pointer;", ";align-items:center;justify-content:start;list-style:none;padding:", itemYPadding, "px ", itemXPadding, "px;margin:0 ", cardPadding, "px;min-height:", componentVars.textBoxHeight, ";border-radius:", getRadius('s'), ";appearance:none;color:", color, ";.Icon{transition:unset;}" + (process.env.NODE_ENV === "production" ? "" : ";label:MenuItemRoot;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9mb3VuZGF0aW9uYWwvbWVudS5qc3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBdUdjIiwiZmlsZSI6Ii4uLy4uL3NyYy9mb3VuZGF0aW9uYWwvbWVudS5qc3giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgUmVhY3QsIHsgZm9yd2FyZFJlZiB9IGZyb20gJ3JlYWN0JztcbmltcG9ydCB7IGNzcyB9IGZyb20gJ0BlbW90aW9uL3JlYWN0JztcbmltcG9ydCB7IGdldENvbG9yLCBnZXREaW1lbnNpb24sIGdldFJhZGl1cywgZ2V0U2hhZG93LCBnZXRUcmFuc2l0aW9uIH0gZnJvbSAnLi4vdXRpbGl0aWVzJztcbmltcG9ydCB7IEJveCwgSWNvbiwgVGV4dCB9IGZyb20gJy4uL2NvbXBvbmVudHMnO1xuaW1wb3J0IHsgSWNvbkNoZWNrIH0gZnJvbSAnLi4vaWNvbnMnO1xuaW1wb3J0IHsgY29tcG9uZW50VmFycyB9IGZyb20gJy4vc3R5bGVzJztcbmltcG9ydCB7IGF1dG9VcGRhdGUsIGZsaXAsIHNpemUsIHVzZUZsb2F0aW5nIH0gZnJvbSAnQGZsb2F0aW5nLXVpL3JlYWN0LWRvbS1pbnRlcmFjdGlvbnMnO1xuXG5jb25zdCBjYXJkUGFkZGluZyA9IDg7XG5jb25zdCBjYXJkUGFkZGluZ1B4ID0gYCR7Y2FyZFBhZGRpbmd9cHhgO1xuY29uc3QgaXRlbVlQYWRkaW5nID0gMTA7XG5jb25zdCBpdGVtWFBhZGRpbmcgPSAxMjtcblxuZXhwb3J0IGNvbnN0IHBsYWNlbWVudHNNYXAgPSB7XG4gIHRvcDogJ3RvcCcsXG4gIHRvcFN0YXJ0OiAndG9wLXN0YXJ0JyxcbiAgdG9wRW5kOiAndG9wLWVuZCcsXG4gIGJvdHRvbTogJ2JvdHRvbScsXG4gIGJvdHRvbVN0YXJ0OiAnYm90dG9tLXN0YXJ0JyxcbiAgYm90dG9tRW5kOiAnYm90dG9tLWVuZCcsXG4gIHJpZ2h0OiAncmlnaHQnLFxuICByaWdodFN0YXJ0OiAncmlnaHQtc3RhcnQnLFxuICByaWdodEVuZDogJ3JpZ2h0LWVuZCcsXG4gIGxlZnQ6ICdsZWZ0JyxcbiAgbGVmdFN0YXJ0OiAnbGVmdC1zdGFydCcsXG4gIGxlZnRFbmQ6ICdsZWZ0LWVuZCcsXG59O1xuXG5leHBvcnQgY29uc3QgdXNlTWVudVBvc2l0aW9uID0gKHsgbWVudVdpZHRoLCBtZW51UGxhY2VtZW50IH0pID0+IHtcbiAgY29uc3QgeyB4LCB5LCByZWZlcmVuY2UsIGZsb2F0aW5nLCBzdHJhdGVneSB9ID0gdXNlRmxvYXRpbmcoe1xuICAgIHBsYWNlbWVudDogcGxhY2VtZW50c01hcFttZW51UGxhY2VtZW50XSxcbiAgICB3aGlsZUVsZW1lbnRzTW91bnRlZDogYXV0b1VwZGF0ZSxcbiAgICBtaWRkbGV3YXJlOiBbXG4gICAgICBmbGlwKHsgcGFkZGluZzogMjQsIGZsaXBBbGlnbm1lbnQ6IGZhbHNlIH0pLFxuICAgICAgc2l6ZSh7XG4gICAgICAgIGFwcGx5KHsgcmVjdHMsIGVsZW1lbnRzIH0pIHtcbiAgICAgICAgICBjb25zdCB3aWR0aCA9IGdldERpbWVuc2lvbihtZW51V2lkdGgpIHx8IGAke3JlY3RzLnJlZmVyZW5jZS53aWR0aH1weGA7XG4gICAgICAgICAgT2JqZWN0LmFzc2lnbihlbGVtZW50cy5mbG9hdGluZy5zdHlsZSwge1xuICAgICAgICAgICAgd2lkdGg6IHdpZHRoLFxuICAgICAgICAgIH0pO1xuICAgICAgICB9LFxuICAgICAgICBwYWRkaW5nOiAyNCxcbiAgICAgIH0pLFxuICAgIF0sXG4gIH0pO1xuXG4gIGNvbnN0IGZsb2F0aW5nU3R5bGVzID0ge1xuICAgIHBvc2l0aW9uOiBzdHJhdGVneSxcbiAgICB0b3A6IHkgPz8gMCxcbiAgICBsZWZ0OiB4ID8/IDAsXG4gICAgekluZGV4OiAxMTAwLFxuICB9O1xuXG4gIHJldHVybiB7XG4gICAgcmVmZXJlbmNlLFxuICAgIGZsb2F0aW5nLFxuICAgIGZsb2F0aW5nU3R5bGVzLFxuICB9O1xufTtcblxuZXhwb3J0IGNvbnN0IE1lbnVDYXJkID0gZm9yd2FyZFJlZihcbiAgKHsgY2hpbGRyZW4sIHpJbmRleCA9IDEsIHBsYWNlbWVudCwgbWF4SGVpZ2h0LCBpc09wZW4sIC4uLnByb3BzIH0sIHJlZikgPT4gKFxuICAgIDx1bFxuICAgICAgcmVmPXtyZWZ9XG4gICAgICBjc3M9e2Nzc2BcbiAgICAgICAgcGFkZGluZzogJHtjYXJkUGFkZGluZ1B4fSAwO1xuICAgICAgICBtYXJnaW46IDA7XG4gICAgICAgIGJhY2tncm91bmQ6ICR7Z2V0Q29sb3IoJ2Zsb2F0aW5nQmFja2dyb3VuZCcpfTtcbiAgICAgICAgei1pbmRleDogJHt6SW5kZXh9O1xuICAgICAgICBib3gtc2hhZG93OiAke2dldFNoYWRvdygnbScpfTtcbiAgICAgICAgYm9yZGVyLXJhZGl1czogJHtnZXRSYWRpdXMoJ3MnKX07XG4gICAgICAgIG92ZXJmbG93OiBhdXRvO1xuICAgICAgICBtYXgtaGVpZ2h0OiAke21heEhlaWdodH07XG4gICAgICAgICR7IWlzT3BlbiAmJiBgb3BhY2l0eTogMDsgcG9pbnRlci1ldmVudHM6IG5vbmU7YH07XG4gICAgICAgIHRyYW5zaXRpb24tcHJvcGVydHk6IG9wYWNpdHk7XG4gICAgICAgIHRyYW5zaXRpb24tZHVyYXRpb246ICR7Z2V0VHJhbnNpdGlvbihpc09wZW4gPyAneHhGYXN0JyA6ICd4RmFzdCcpfTtcblxuICAgICAgICAmOmZvY3VzIHtcbiAgICAgICAgICBvdXRsaW5lOiBub25lO1xuICAgICAgICB9XG4gICAgICBgfVxuICAgICAgey4uLnByb3BzfVxuICAgID5cbiAgICAgIHtjaGlsZHJlbn1cbiAgICA8L3VsPlxuICApLFxuKTtcblxuY29uc3QgdmFyaWFudFN0eWxlcyA9IHtcbiAgbmV1dHJhbDoge1xuICAgIGJhY2tncm91bmRDb2xvckFjdGl2ZTogZ2V0Q29sb3IoJ2ZhZGUxJyksXG4gICAgaWNvbkNvbG9yQWN0aXZlOiBnZXRDb2xvcignY29udGVudCcpLFxuICB9LFxuICBkYW5nZXI6IHtcbiAgICBjb2xvcjogZ2V0Q29sb3IoJ2RhbmdlcicpLFxuICAgIGJhY2tncm91bmRDb2xvckFjdGl2ZTogY29tcG9uZW50VmFycy5jbGVhckJ1dHRvbkRhbmdlckJhY2tncm91bmRIb3ZlcixcbiAgfSxcbn07XG5cbmNvbnN0IE1lbnVJdGVtUm9vdCA9IGZvcndhcmRSZWYoXG4gICh7IGNoaWxkcmVuLCBpc0Rpc2FibGVkLCBoYXNJY29uLCBpc1NlbGVjdGVkLCBjb2xvciwgLi4ucHJvcHMgfSwgcmVmKSA9PiAoXG4gICAgPGxpXG4gICAgICByZWY9e3JlZn1cbiAgICAgIGNzcz17Y3NzYFxuICAgICAgICBkaXNwbGF5OiBncmlkO1xuICAgICAgICBncmlkLWF1dG8tZmxvdzogY29sdW1uO1xuICAgICAgICBncmlkLXRlbXBsYXRlLWNvbHVtbnM6ICR7aGFzSWNvbiA/ICdhdXRvIDFmcicgOiAnMWZyIGF1dG8nfTtcbiAgICAgICAgZ2FwOiA4cHg7XG4gICAgICAgICR7aXNEaXNhYmxlZCAmJiBgY29sb3I6ICR7Z2V0Q29sb3IoJ2Rpc2FibGVkQ29udGVudCcpfWB9O1xuICAgICAgICAkeyFpc0Rpc2FibGVkICYmIGBjdXJzb3I6IHBvaW50ZXI7YH07XG4gICAgICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gICAgICAgIGp1c3RpZnktY29udGVudDogc3RhcnQ7XG4gICAgICAgIGxpc3Qtc3R5bGU6IG5vbmU7XG4gICAgICAgIHBhZGRpbmc6ICR7aXRlbVlQYWRkaW5nfXB4ICR7aXRlbVhQYWRkaW5nfXB4O1xuICAgICAgICBtYXJnaW46IDAgJHtjYXJkUGFkZGluZ31weDtcbiAgICAgICAgbWluLWhlaWdodDogJHtjb21wb25lbnRWYXJzLnRleHRCb3hIZWlnaHR9O1xuICAgICAgICBib3JkZXItcmFkaXVzOiAke2dldFJhZGl1cygncycpfTtcbiAgICAgICAgYXBwZWFyYW5jZTogbm9uZTtcbiAgICAgICAgY29sb3I6ICR7Y29sb3J9O1xuXG4gICAgICAgIC5JY29uIHtcbiAgICAgICAgICB0cmFuc2l0aW9uOiB1bnNldDtcbiAgICAgICAgfVxuICAgICAgYH1cbiAgICAgIHsuLi5wcm9wc31cbiAgICA+XG4gICAgICB7Y2hpbGRyZW59XG4gICAgPC9saT5cbiAgKSxcbik7XG5cbmV4cG9ydCBjb25zdCBNZW51R3JvdXBUaXRsZSA9ICh7IGNoaWxkcmVuLCBoYXNEaXZpZGVyIH0pID0+IChcbiAgPEJveFxuICAgIHBhZGRpbmdYPXtgJHtjYXJkUGFkZGluZyArIGl0ZW1YUGFkZGluZ31weGB9XG4gICAgcGFkZGluZ0JvdHRvbT17Y2FyZFBhZGRpbmdQeH1cbiAgICBtYXJnaW5Ub3A9e2NhcmRQYWRkaW5nUHh9XG4gICAgcGFkZGluZ1RvcD17aGFzRGl2aWRlciA/IGAke2NhcmRQYWRkaW5nICogMn1weGAgOiB1bmRlZmluZWR9XG4gICAgYm9yZGVyU2lkZT17aGFzRGl2aWRlciA/ICd0b3AnIDogdW5kZWZpbmVkfVxuICA+XG4gICAgPFRleHQgdmFyaWFudD1cImNhcHNcIj57Y2hpbGRyZW59PC9UZXh0PlxuICA8L0JveD5cbik7XG5cbmNvbnN0IE1lbnVJdGVtRGl2aWRlciA9ICgpID0+IChcbiAgPEJveCBoZWlnaHQ9e2NvbXBvbmVudFZhcnMuZGl2aWRlclN0cm9rZVNpemV9IG1hcmdpblk9ezF9IGJhY2tncm91bmRDb2xvcj1cImJvcmRlclwiIC8+XG4pO1xuXG5leHBvcnQgY29uc3QgTWVudUl0ZW0gPSBmb3J3YXJkUmVmKFxuICAoXG4gICAge1xuICAgICAgY2hpbGRyZW4sXG4gICAgICBpc1NlbGVjdGVkLFxuICAgICAgaXNBY3RpdmUsXG4gICAgICBpc0Rpc2FibGVkLFxuICAgICAgaWNvbixcbiAgICAgIHZhcmlhbnQgPSAnbmV1dHJhbCcsXG4gICAgICBoYXNEaXZpZGVyLFxuICAgICAgaGFzRWxsaXBzaXMsXG4gICAgICAuLi5wcm9wc1xuICAgIH0sXG4gICAgcmVmLFxuICApID0+IHtcbiAgICByZXR1cm4gKFxuICAgICAgPD5cbiAgICAgICAge2hhc0RpdmlkZXIgJiYgPE1lbnVJdGVtRGl2aWRlciAvPn1cbiAgICAgICAgPE1lbnVJdGVtUm9vdFxuICAgICAgICAgIHJlZj17cmVmfVxuICAgICAgICAgIGlzU2VsZWN0ZWQ9e2lzU2VsZWN0ZWR9XG4gICAgICAgICAgaXNEaXNhYmxlZD17aXNEaXNhYmxlZH1cbiAgICAgICAgICBjb2xvcj17dmFyaWFudFN0eWxlc1t2YXJpYW50XS5jb2xvcn1cbiAgICAgICAgICBoYXNJY29uPXtCb29sZWFuKGljb24pfVxuICAgICAgICAgIHN0eWxlPXt7XG4gICAgICAgICAgICBiYWNrZ3JvdW5kOiBpc0FjdGl2ZSAmJiB2YXJpYW50U3R5bGVzW3ZhcmlhbnRdLmJhY2tncm91bmRDb2xvckFjdGl2ZSxcbiAgICAgICAgICAgIFsnLS1ncm4tY29sb3ItaWNvbiddOiBpc0FjdGl2ZSAmJiB2YXJpYW50U3R5bGVzW3ZhcmlhbnRdLmljb25Db2xvckFjdGl2ZSxcbiAgICAgICAgICB9fVxuICAgICAgICAgIHsuLi5wcm9wc31cbiAgICAgICAgPlxuICAgICAgICAgIHtpY29uICYmIDxJY29uIGljb249e2ljb259IGhhc0V2ZW5Cb3VuZGFyeSBjb2xvcj17dmFyaWFudFN0eWxlc1t2YXJpYW50XS5jb2xvcn0gLz59XG4gICAgICAgICAgPEJveCB3aWR0aD1cIjEwMCVcIiBtaW5XaWR0aD1cIjBweFwiIHRhZz1cInNwYW5cIj5cbiAgICAgICAgICAgIDxUZXh0IGhhc0VsbGlwc2lzPXtoYXNFbGxpcHNpc30+e2NoaWxkcmVufTwvVGV4dD5cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgICA8Qm94IG9wYWNpdHk9e2lzU2VsZWN0ZWQgPyAnMScgOiAnMCd9PlxuICAgICAgICAgICAgPEljb24gaWNvbj17PEljb25DaGVjayAvPn0gLz5cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgPC9NZW51SXRlbVJvb3Q+XG4gICAgICA8Lz5cbiAgICApO1xuICB9LFxuKTtcbiJdfQ== */")
120
+ css: /*#__PURE__*/css("display:grid;grid-auto-flow:column;gap:8px;", isDisabled && "color: ".concat(getColor('disabledContent')), ";", !isDisabled && "cursor: pointer;", ";align-items:center;justify-content:start;list-style:none;padding:", itemYPadding, "px ", itemXPadding, "px;margin:0 ", cardPadding, "px;min-height:", componentVars.textBoxHeight, ";border-radius:", getRadius('s'), ";appearance:none;color:", color, ";.Icon{transition:unset;}" + (process.env.NODE_ENV === "production" ? "" : ";label:MenuItemRoot;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AAwGc","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, { forwardRef } from 'react';\nimport PropTypes from 'prop-types';\nimport { css } from '@emotion/react';\nimport { getColor, getDimension, getRadius, getShadow, getTransition } from '../utilities';\nimport { Box, Checkbox, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { componentVars } from './styles';\nimport { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemYPadding = 10;\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 } = 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    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nexport const MenuCard = forwardRef(\n  ({ children, zIndex = 1, placement, maxHeight, isOpen, ...props }, ref) => (\n    <ul\n      ref={ref}\n      css={css`\n        padding: ${cardPaddingPx} 0;\n        margin: 0;\n        background: ${getColor('floatingBackground')};\n        z-index: ${zIndex};\n        box-shadow: ${getShadow('m')};\n        border-radius: ${getRadius('s')};\n        overflow: auto;\n        max-height: ${maxHeight};\n        ${!isOpen && `opacity: 0; pointer-events: none;`};\n        transition-property: opacity;\n        transition-duration: ${getTransition(isOpen ? 'xxFast' : 'xFast')};\n\n        &:focus {\n          outline: none;\n        }\n      `}\n      {...props}\n    >\n      {children}\n    </ul>\n  ),\n);\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('fade1'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('danger'),\n    backgroundColorActive: componentVars.clearButtonDangerBackgroundHover,\n  },\n};\n\nconst MenuItemRoot = forwardRef(\n  ({ children, isDisabled, hasIcon, isSelected, color, columns, ...props }, ref) => (\n    <li\n      ref={ref}\n      css={css`\n        display: grid;\n        grid-auto-flow: column;\n        gap: 8px;\n        ${isDisabled && `color: ${getColor('disabledContent')}`};\n        ${!isDisabled && `cursor: pointer;`};\n        align-items: center;\n        justify-content: start;\n        list-style: none;\n        padding: ${itemYPadding}px ${itemXPadding}px;\n        margin: 0 ${cardPadding}px;\n        min-height: ${componentVars.textBoxHeight};\n        border-radius: ${getRadius('s')};\n        appearance: none;\n        color: ${color};\n\n        .Icon {\n          transition: unset;\n        }\n      `}\n      {...props}\n    >\n      {children}\n    </li>\n  ),\n);\n\nexport const MenuGroupTitle = ({ children, hasDivider }) => (\n  <Box\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingBottom={cardPaddingPx}\n    marginTop={cardPaddingPx}\n    paddingTop={hasDivider ? `${cardPadding * 2}px` : undefined}\n    borderSide={hasDivider ? 'top' : undefined}\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nconst MenuItemDivider = () => (\n  <Box height={componentVars.dividerStrokeSize} marginY={1} backgroundColor=\"border\" />\n);\n\nexport const MenuItem = forwardRef(\n  (\n    {\n      children,\n      isSelected,\n      isActive,\n      isDisabled,\n      icon,\n      variant = 'neutral',\n      hasDivider,\n      hasEllipsis,\n      checkVariant = 'check',\n      ...props\n    },\n    ref,\n  ) => {\n    const hasCheck = checkVariant === 'check';\n    const hasCheckbox = checkVariant === 'checkbox';\n\n    const getColumns = () => {\n      if (Boolean(icon) || hasCheckbox) return 'auto 1fr';\n      if (hasCheck) return '1fr auto';\n    };\n\n    return (\n      <>\n        {hasDivider && <MenuItemDivider />}\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            ['--grn-color-icon']: isActive && variantStyles[variant].iconColorActive,\n            gridTemplateColumns: getColumns(),\n          }}\n          {...props}\n        >\n          {hasCheckbox && <Checkbox isChecked={isSelected} disabled={isDisabled} />}\n          {icon && <Icon icon={icon} hasEvenBoundary color={variantStyles[variant].color} />}\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 />} />\n            </Box>\n          )}\n        </MenuItemRoot>\n      </>\n    );\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  hasDivider: PropTypes.bool,\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'checkbox', 'none']),\n};\n"]} */")
119
121
  }, props), children);
120
122
  });
121
123
  export var MenuGroupTitle = function MenuGroupTitle(_ref5) {
@@ -141,6 +143,8 @@ var MenuItemDivider = function MenuItemDivider() {
141
143
  };
142
144
 
143
145
  export var MenuItem = /*#__PURE__*/forwardRef(function (_ref6, ref) {
146
+ var _ref7;
147
+
144
148
  var children = _ref6.children,
145
149
  isSelected = _ref6.isSelected,
146
150
  isActive = _ref6.isActive,
@@ -150,18 +154,31 @@ export var MenuItem = /*#__PURE__*/forwardRef(function (_ref6, ref) {
150
154
  variant = _ref6$variant === void 0 ? 'neutral' : _ref6$variant,
151
155
  hasDivider = _ref6.hasDivider,
152
156
  hasEllipsis = _ref6.hasEllipsis,
157
+ _ref6$checkVariant = _ref6.checkVariant,
158
+ checkVariant = _ref6$checkVariant === void 0 ? 'check' : _ref6$checkVariant,
153
159
  props = _objectWithoutProperties(_ref6, _excluded3);
154
160
 
161
+ var hasCheck = checkVariant === 'check';
162
+ var hasCheckbox = checkVariant === 'checkbox';
163
+
164
+ var getColumns = function getColumns() {
165
+ if (Boolean(icon) || hasCheckbox) return 'auto 1fr';
166
+ if (hasCheck) return '1fr auto';
167
+ };
168
+
155
169
  return ___EmotionJSX(React.Fragment, null, hasDivider && ___EmotionJSX(MenuItemDivider, null), ___EmotionJSX(MenuItemRoot, _extends({
156
170
  ref: ref,
157
171
  isSelected: isSelected,
158
172
  isDisabled: isDisabled,
159
173
  color: variantStyles[variant].color,
160
174
  hasIcon: Boolean(icon),
161
- style: _defineProperty({
175
+ style: (_ref7 = {
162
176
  background: isActive && variantStyles[variant].backgroundColorActive
163
- }, '--grn-color-icon', isActive && variantStyles[variant].iconColorActive)
164
- }, props), icon && ___EmotionJSX(Icon, {
177
+ }, _defineProperty(_ref7, '--grn-color-icon', isActive && variantStyles[variant].iconColorActive), _defineProperty(_ref7, "gridTemplateColumns", getColumns()), _ref7)
178
+ }, props), hasCheckbox && ___EmotionJSX(Checkbox, {
179
+ isChecked: isSelected,
180
+ disabled: isDisabled
181
+ }), icon && ___EmotionJSX(Icon, {
165
182
  icon: icon,
166
183
  hasEvenBoundary: true,
167
184
  color: variantStyles[variant].color
@@ -171,9 +188,19 @@ export var MenuItem = /*#__PURE__*/forwardRef(function (_ref6, ref) {
171
188
  tag: "span"
172
189
  }, ___EmotionJSX(Text, {
173
190
  hasEllipsis: hasEllipsis
174
- }, children)), ___EmotionJSX(Box, {
191
+ }, children)), hasCheck && ___EmotionJSX(Box, {
175
192
  opacity: isSelected ? '1' : '0'
176
193
  }, ___EmotionJSX(Icon, {
177
194
  icon: ___EmotionJSX(IconCheck, null)
178
195
  }))));
179
- });
196
+ });
197
+ MenuItem.propTypes = {
198
+ isSelected: PropTypes.bool,
199
+ isActive: PropTypes.bool,
200
+ isDisabled: PropTypes.bool,
201
+ icon: PropTypes.node,
202
+ variant: PropTypes.oneOf(['neutral', 'danger']),
203
+ hasDivider: PropTypes.bool,
204
+ hasEllipsis: PropTypes.bool,
205
+ checkVariant: PropTypes.oneOf(['check', 'checkbox', 'none'])
206
+ };
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@flodesk/grain",
3
- "version": "7.7.0",
3
+ "version": "7.8.0",
4
4
  "description": "Flodesk design system",
5
5
  "module": "es/index.js",
6
6
  "author": "Flodesk",