@flodesk/grain 7.2.0 → 7.3.1

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.
@@ -3,10 +3,15 @@ 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.assign.js";
5
5
  import _styled from "@emotion/styled/base";
6
- var _excluded = ["children", "triggerVariant"];
7
- import "core-js/modules/es.array.find.js";
8
- import "core-js/modules/es.object.to-string.js";
6
+ var _excluded = ["children", "triggerVariant"],
7
+ _excluded2 = ["option", "isSelected", "menuItemsHaveEllipsis"],
8
+ _excluded3 = ["children", "floating"];
9
9
  import "core-js/modules/es.array.map.js";
10
+ import "core-js/modules/es.object.to-string.js";
11
+ import "core-js/modules/es.array.iterator.js";
12
+ import "core-js/modules/es.map.js";
13
+ import "core-js/modules/es.string.iterator.js";
14
+ import "core-js/modules/web.dom-collections.iterator.js";
10
15
 
11
16
  function _extends() { _extends = Object.assign ? Object.assign.bind() : function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); }
12
17
 
@@ -33,7 +38,7 @@ var TriggerButton = /*#__PURE__*/_styled("button", process.env.NODE_ENV === "pro
33
38
  } : {
34
39
  target: "e17qd7kh1",
35
40
  label: "TriggerButton"
36
- })(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,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb21wb25lbnRzL3NlbGVjdC5qc3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBb0JtQyIsImZpbGUiOiIuLi8uLi9zcmMvY29tcG9uZW50cy9zZWxlY3QuanN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFByb3BUeXBlcyBmcm9tICdwcm9wLXR5cGVzJztcbmltcG9ydCBSZWFjdCwgeyBmb3J3YXJkUmVmLCBGcmFnbWVudCwgdXNlTWVtbyB9IGZyb20gJ3JlYWN0JztcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJztcbmltcG9ydCB7IEJveCwgSWNvbiwgVGV4dCwgVGV4dEJ1dHRvbiB9IGZyb20gJy4nO1xuaW1wb3J0IHsgSWNvbkNoZXZyb25Eb3duIH0gZnJvbSAnLi4vaWNvbnMnO1xuaW1wb3J0IHsgTGlzdGJveCB9IGZyb20gJ0BoZWFkbGVzc3VpL3JlYWN0JztcbmltcG9ydCB7IGdldENvbG9yLCBnZXRSYWRpdXMgfSBmcm9tICcuLi91dGlsaXRpZXMnO1xuaW1wb3J0IHsgZGVmYXVsdFByb3BzLCB0eXBlcyB9IGZyb20gJy4uL3R5cGVzJztcbmltcG9ydCB7XG4gIGNvbXBvbmVudFZhcnMsXG4gIEZpZWxkSGludCxcbiAgRmllbGRMYWJlbCxcbiAgZmllbGRWYXJzLFxuICBNZW51Q2FyZCxcbiAgTWVudUl0ZW0sXG4gIHN0eWxlcyxcbn0gZnJvbSAnLi4vZm91bmRhdGlvbmFsJztcbmltcG9ydCB7IHVzZU1lbnVQb3NpdGlvbiB9IGZyb20gJy4uL2ZvdW5kYXRpb25hbC9tZW51JztcbmltcG9ydCB7IEZsb2F0aW5nUG9ydGFsIH0gZnJvbSAnQGZsb2F0aW5nLXVpL3JlYWN0LWRvbS1pbnRlcmFjdGlvbnMnO1xuXG5jb25zdCBUcmlnZ2VyQnV0dG9uID0gc3R5bGVkLmJ1dHRvbmBcbiAgJHtzdHlsZXMuYnV0dG9uUmVzZXR9O1xuICAke3N0eWxlcy50cmFuc2l0aW9uc307XG5cbiAgY29sb3I6IGluaGVyaXQ7XG4gIGRpc3BsYXk6IGZsZXg7XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIGdhcDogOHB4O1xuICBib3JkZXI6IG5vbmU7XG4gIGJhY2tncm91bmQ6ICR7Z2V0Q29sb3IoJ2ZhZGUxJyl9O1xuICBwYWRkaW5nOiAwICR7ZmllbGRWYXJzLnhQYWRkaW5nfTtcbiAgbWluLWhlaWdodDogJHtjb21wb25lbnRWYXJzLnRleHRCb3hIZWlnaHR9O1xuICBib3JkZXItcmFkaXVzOiAke2dldFJhZGl1cygncycpfTtcbiAgd2lkdGg6IDEwMCU7XG4gIHRleHQtYWxpZ246IGxlZnQ7XG5cbiAgJjpob3ZlciB7XG4gICAgYmFja2dyb3VuZDogJHtnZXRDb2xvcignZmFkZTInKX07XG4gIH1cbmA7XG5cbmNvbnN0IFRyaWdnZXIgPSBmb3J3YXJkUmVmKCh7IGNoaWxkcmVuLCB0cmlnZ2VyVmFyaWFudCwgLi4ucHJvcHMgfSwgcmVmKSA9PiAoXG4gIDw+XG4gICAge3RyaWdnZXJWYXJpYW50ID09PSAnYm94JyAmJiAoXG4gICAgICA8VHJpZ2dlckJ1dHRvbiByZWY9e3JlZn0gdHlwZT1cImJ1dHRvblwiIHsuLi5wcm9wc30+XG4gICAgICAgIDxUZXh0IGhhc0VsbGlwc2lzPntjaGlsZHJlbn08L1RleHQ+XG4gICAgICAgIDxCb3ggbWFyZ2luTGVmdD1cImF1dG9cIj5cbiAgICAgICAgICA8SWNvbiBpY29uPXs8SWNvbkNoZXZyb25Eb3duIC8+fSAvPlxuICAgICAgICA8L0JveD5cbiAgICAgIDwvVHJpZ2dlckJ1dHRvbj5cbiAgICApfVxuICAgIHt0cmlnZ2VyVmFyaWFudCA9PT0gJ3RleHQnICYmIChcbiAgICAgIDxUZXh0QnV0dG9uIHJlZj17cmVmfSBpY29uPXs8SWNvbkNoZXZyb25Eb3duIC8+fSBpY29uUG9zaXRpb249XCJyaWdodFwiIHsuLi5wcm9wc30+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvVGV4dEJ1dHRvbj5cbiAgICApfVxuICA8Lz5cbikpO1xuXG5jb25zdCBSb290ID0gc3R5bGVkLmRpdmBcbiAgbWluLXdpZHRoOiAwcHg7XG5gO1xuXG5leHBvcnQgY29uc3QgU2VsZWN0ID0gKHtcbiAgb3B0aW9ucyxcbiAgdmFsdWUsXG4gIG9uQ2hhbmdlLFxuICBtZW51UGxhY2VtZW50ID0gZGVmYXVsdFByb3BzLm1lbnVQbGFjZW1lbnQsXG4gIG1lbnVXaWR0aCxcbiAgbWVudU1heEhlaWdodCA9IGRlZmF1bHRQcm9wcy5tZW51TWF4SGVpZ2h0LFxuICB0cmlnZ2VyOiBjdXN0b21UcmlnZ2VyLFxuICB0cmlnZ2VyVmFyaWFudCA9ICdib3gnLFxuICBsYWJlbCxcbiAgaGludCxcbiAgbWVudVpJbmRleCxcbiAgbWVudUl0ZW1zSGF2ZUVsbGlwc2lzID0gdHJ1ZSxcbn0pID0+IHtcbiAgY29uc3QgZmllbGRNYXJnaW5Ub3AgPSBsYWJlbCB8fCBoaW50ID8gJ2JldHdlZW5Gb3JtQ29udHJvbEFuZExhYmVsJyA6IHVuZGVmaW5lZDtcbiAgY29uc3Qgc2VsZWN0ZWRPcHRpb24gPSB1c2VNZW1vKFxuICAgICgpID0+IG9wdGlvbnMuZmluZChvcHRpb24gPT4gb3B0aW9uLnZhbHVlID09PSB2YWx1ZSksXG4gICAgW29wdGlvbnMsIHZhbHVlXSxcbiAgKTtcblxuICBjb25zdCB7IHJlZmVyZW5jZSwgZmxvYXRpbmcsIGZsb2F0aW5nU3R5bGVzIH0gPSB1c2VNZW51UG9zaXRpb24oeyBtZW51V2lkdGgsIG1lbnVQbGFjZW1lbnQgfSk7XG5cbiAgY29uc3Qgcm9vdFdpZHRoID0gdHJpZ2dlclZhcmlhbnQgPT09ICd0ZXh0JyA/ICdmaXQtY29udGVudCcgOiB1bmRlZmluZWQ7XG5cbiAgcmV0dXJuIChcbiAgICA8TGlzdGJveCBhcz17Um9vdH0gdmFsdWU9e3ZhbHVlfSBvbkNoYW5nZT17b25DaGFuZ2V9IHdpZHRoPXtyb290V2lkdGh9PlxuICAgICAgeyh7IG9wZW4gfSkgPT4gKFxuICAgICAgICA8PlxuICAgICAgICAgIHtsYWJlbCAmJiA8TGlzdGJveC5MYWJlbCBhcz17RmllbGRMYWJlbH0+e2xhYmVsfTwvTGlzdGJveC5MYWJlbD59XG4gICAgICAgICAge2hpbnQgJiYgPEZpZWxkSGludD57aGludH08L0ZpZWxkSGludD59XG5cbiAgICAgICAgICA8Qm94IHBvc2l0aW9uPVwicmVsYXRpdmVcIiBtYXJnaW5Ub3A9e2ZpZWxkTWFyZ2luVG9wfSByZWY9e3JlZmVyZW5jZX0+XG4gICAgICAgICAgICA8TGlzdGJveC5CdXR0b24gYXM9e0ZyYWdtZW50fT5cbiAgICAgICAgICAgICAge2N1c3RvbVRyaWdnZXIgPyAoXG4gICAgICAgICAgICAgICAgY3VzdG9tVHJpZ2dlcihzZWxlY3RlZE9wdGlvbilcbiAgICAgICAgICAgICAgKSA6IChcbiAgICAgICAgICAgICAgICA8VHJpZ2dlciB0cmlnZ2VyVmFyaWFudD17dHJpZ2dlclZhcmlhbnR9PlxuICAgICAgICAgICAgICAgICAge3NlbGVjdGVkT3B0aW9uID8gc2VsZWN0ZWRPcHRpb24uY29udGVudCA6ICcnfVxuICAgICAgICAgICAgICAgIDwvVHJpZ2dlcj5cbiAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgIDwvTGlzdGJveC5CdXR0b24+XG5cbiAgICAgICAgICAgIDxGbG9hdGluZ1BvcnRhbD5cbiAgICAgICAgICAgICAgPExpc3Rib3guT3B0aW9uc1xuICAgICAgICAgICAgICAgIHN0YXRpY1xuICAgICAgICAgICAgICAgIGlzT3Blbj17b3Blbn1cbiAgICAgICAgICAgICAgICBjbGFzc05hbWU9XCJncm4tY29udGV4dFwiXG4gICAgICAgICAgICAgICAgcmVmPXtmbG9hdGluZ31cbiAgICAgICAgICAgICAgICBwbGFjZW1lbnQ9e21lbnVQbGFjZW1lbnR9XG4gICAgICAgICAgICAgICAgYXM9e01lbnVDYXJkfVxuICAgICAgICAgICAgICAgIHpJbmRleD17bWVudVpJbmRleH1cbiAgICAgICAgICAgICAgICBtYXhIZWlnaHQ9e21lbnVNYXhIZWlnaHR9XG4gICAgICAgICAgICAgICAgc3R5bGU9e2Zsb2F0aW5nU3R5bGVzfVxuICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAge29wdGlvbnMubWFwKChvcHRpb24sIGluZGV4KSA9PiAoXG4gICAgICAgICAgICAgICAgICA8TGlzdGJveC5PcHRpb25cbiAgICAgICAgICAgICAgICAgICAga2V5PXtpbmRleH1cbiAgICAgICAgICAgICAgICAgICAgdmFsdWU9e29wdGlvbn1cbiAgICAgICAgICAgICAgICAgICAgYXM9e0ZyYWdtZW50fVxuICAgICAgICAgICAgICAgICAgICBkaXNhYmxlZD17b3B0aW9uLmlzRGlzYWJsZWR9XG4gICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgIHsoeyBhY3RpdmUgfSkgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgIHJldHVybiAoXG4gICAgICAgICAgICAgICAgICAgICAgICA8TWVudUl0ZW1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgaXNTZWxlY3RlZD17b3B0aW9uLnZhbHVlID09PSB2YWx1ZX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgaXNBY3RpdmU9e2FjdGl2ZX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgaXNEaXNhYmxlZD17b3B0aW9uLmlzRGlzYWJsZWR9XG4gICAgICAgICAgICAgICAgICAgICAgICAgIGhhc0VsbGlwc2lzPXttZW51SXRlbXNIYXZlRWxsaXBzaXN9XG4gICAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgICAgIHtvcHRpb24uY29udGVudH1cbiAgICAgICAgICAgICAgICAgICAgICAgIDwvTWVudUl0ZW0+XG4gICAgICAgICAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgIDwvTGlzdGJveC5PcHRpb24+XG4gICAgICAgICAgICAgICAgKSl9XG4gICAgICAgICAgICAgIDwvTGlzdGJveC5PcHRpb25zPlxuICAgICAgICAgICAgPC9GbG9hdGluZ1BvcnRhbD5cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgPC8+XG4gICAgICApfVxuICAgIDwvTGlzdGJveD5cbiAgKTtcbn07XG5cblNlbGVjdC5MYWJlbCA9IEZpZWxkTGFiZWw7XG5TZWxlY3QuSGludCA9IEZpZWxkSGludDtcblxuU2VsZWN0LnByb3BUeXBlcyA9IHtcbiAgb3B0aW9uczogUHJvcFR5cGVzLmFycmF5LFxuICB2YWx1ZTogUHJvcFR5cGVzLnN0cmluZy5pc1JlcXVpcmVkLFxuICBvbkNoYW5nZTogUHJvcFR5cGVzLmZ1bmMuaXNSZXF1aXJlZCxcbiAgbWVudVBsYWNlbWVudDogdHlwZXMubWVudVBsYWNlbWVudCxcbiAgbWVudVdpZHRoOiB0eXBlcy5kaW1lbnNpb24sXG4gIG1lbnVaSW5kZXg6IHR5cGVzLnpJbmRleCxcbiAgbWVudU1heEhlaWdodDogdHlwZXMuZGltZW5zaW9uLFxuICB0cmlnZ2VyOiBQcm9wVHlwZXMuZnVuYyxcbiAgdHJpZ2dlclZhcmlhbnQ6IFByb3BUeXBlcy5vbmVPZihbJ2JveCcsICd0ZXh0J10pLFxuICBsYWJlbDogdHlwZXMubGFiZWwsXG4gIGhpbnQ6IHR5cGVzLmhpbnQsXG4gIG1lbnVJdGVtc0hhdmVFbGxpcHNpczogUHJvcFR5cGVzLmJvb2wsXG59O1xuIl19 */"));
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, ...props }) => (\n  <FloatingPortal>\n    <Listbox.Options static className=\"grn-context\" ref={floating} as={MenuCard} {...props}>\n      {children}\n    </Listbox.Options>\n  </FloatingPortal>\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}) => {\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            >\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};\n"]} */"));
37
42
 
38
43
  var Trigger = /*#__PURE__*/forwardRef(function (_ref, ref) {
39
44
  var children = _ref.children,
@@ -56,6 +61,29 @@ var Trigger = /*#__PURE__*/forwardRef(function (_ref, ref) {
56
61
  }, props), children));
57
62
  });
58
63
 
64
+ var getTriggerContent = function getTriggerContent(isMultiple, selected) {
65
+ if (isMultiple) {
66
+ return selected.map(function (option) {
67
+ return option.content;
68
+ }).join(', ');
69
+ }
70
+
71
+ return selected.content;
72
+ };
73
+
74
+ var SelectButton = function SelectButton(_ref2) {
75
+ var customTrigger = _ref2.customTrigger,
76
+ triggerVariant = _ref2.triggerVariant,
77
+ isMultiple = _ref2.isMultiple,
78
+ selected = _ref2.selected;
79
+ var trigger = getTriggerContent(isMultiple, selected);
80
+ return ___EmotionJSX(Listbox.Button, {
81
+ as: Fragment
82
+ }, customTrigger ? customTrigger(selected) : ___EmotionJSX(Trigger, {
83
+ triggerVariant: triggerVariant
84
+ }, trigger));
85
+ };
86
+
59
87
  var Root = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "production" ? {
60
88
  target: "e17qd7kh0"
61
89
  } : {
@@ -67,33 +95,82 @@ var Root = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "production" ? {
67
95
  } : {
68
96
  name: "wowqs1",
69
97
  styles: "min-width:0px",
70
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb21wb25lbnRzL3NlbGVjdC5qc3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMkR1QiIsImZpbGUiOiIuLi8uLi9zcmMvY29tcG9uZW50cy9zZWxlY3QuanN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFByb3BUeXBlcyBmcm9tICdwcm9wLXR5cGVzJztcbmltcG9ydCBSZWFjdCwgeyBmb3J3YXJkUmVmLCBGcmFnbWVudCwgdXNlTWVtbyB9IGZyb20gJ3JlYWN0JztcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJztcbmltcG9ydCB7IEJveCwgSWNvbiwgVGV4dCwgVGV4dEJ1dHRvbiB9IGZyb20gJy4nO1xuaW1wb3J0IHsgSWNvbkNoZXZyb25Eb3duIH0gZnJvbSAnLi4vaWNvbnMnO1xuaW1wb3J0IHsgTGlzdGJveCB9IGZyb20gJ0BoZWFkbGVzc3VpL3JlYWN0JztcbmltcG9ydCB7IGdldENvbG9yLCBnZXRSYWRpdXMgfSBmcm9tICcuLi91dGlsaXRpZXMnO1xuaW1wb3J0IHsgZGVmYXVsdFByb3BzLCB0eXBlcyB9IGZyb20gJy4uL3R5cGVzJztcbmltcG9ydCB7XG4gIGNvbXBvbmVudFZhcnMsXG4gIEZpZWxkSGludCxcbiAgRmllbGRMYWJlbCxcbiAgZmllbGRWYXJzLFxuICBNZW51Q2FyZCxcbiAgTWVudUl0ZW0sXG4gIHN0eWxlcyxcbn0gZnJvbSAnLi4vZm91bmRhdGlvbmFsJztcbmltcG9ydCB7IHVzZU1lbnVQb3NpdGlvbiB9IGZyb20gJy4uL2ZvdW5kYXRpb25hbC9tZW51JztcbmltcG9ydCB7IEZsb2F0aW5nUG9ydGFsIH0gZnJvbSAnQGZsb2F0aW5nLXVpL3JlYWN0LWRvbS1pbnRlcmFjdGlvbnMnO1xuXG5jb25zdCBUcmlnZ2VyQnV0dG9uID0gc3R5bGVkLmJ1dHRvbmBcbiAgJHtzdHlsZXMuYnV0dG9uUmVzZXR9O1xuICAke3N0eWxlcy50cmFuc2l0aW9uc307XG5cbiAgY29sb3I6IGluaGVyaXQ7XG4gIGRpc3BsYXk6IGZsZXg7XG4gIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gIGdhcDogOHB4O1xuICBib3JkZXI6IG5vbmU7XG4gIGJhY2tncm91bmQ6ICR7Z2V0Q29sb3IoJ2ZhZGUxJyl9O1xuICBwYWRkaW5nOiAwICR7ZmllbGRWYXJzLnhQYWRkaW5nfTtcbiAgbWluLWhlaWdodDogJHtjb21wb25lbnRWYXJzLnRleHRCb3hIZWlnaHR9O1xuICBib3JkZXItcmFkaXVzOiAke2dldFJhZGl1cygncycpfTtcbiAgd2lkdGg6IDEwMCU7XG4gIHRleHQtYWxpZ246IGxlZnQ7XG5cbiAgJjpob3ZlciB7XG4gICAgYmFja2dyb3VuZDogJHtnZXRDb2xvcignZmFkZTInKX07XG4gIH1cbmA7XG5cbmNvbnN0IFRyaWdnZXIgPSBmb3J3YXJkUmVmKCh7IGNoaWxkcmVuLCB0cmlnZ2VyVmFyaWFudCwgLi4ucHJvcHMgfSwgcmVmKSA9PiAoXG4gIDw+XG4gICAge3RyaWdnZXJWYXJpYW50ID09PSAnYm94JyAmJiAoXG4gICAgICA8VHJpZ2dlckJ1dHRvbiByZWY9e3JlZn0gdHlwZT1cImJ1dHRvblwiIHsuLi5wcm9wc30+XG4gICAgICAgIDxUZXh0IGhhc0VsbGlwc2lzPntjaGlsZHJlbn08L1RleHQ+XG4gICAgICAgIDxCb3ggbWFyZ2luTGVmdD1cImF1dG9cIj5cbiAgICAgICAgICA8SWNvbiBpY29uPXs8SWNvbkNoZXZyb25Eb3duIC8+fSAvPlxuICAgICAgICA8L0JveD5cbiAgICAgIDwvVHJpZ2dlckJ1dHRvbj5cbiAgICApfVxuICAgIHt0cmlnZ2VyVmFyaWFudCA9PT0gJ3RleHQnICYmIChcbiAgICAgIDxUZXh0QnV0dG9uIHJlZj17cmVmfSBpY29uPXs8SWNvbkNoZXZyb25Eb3duIC8+fSBpY29uUG9zaXRpb249XCJyaWdodFwiIHsuLi5wcm9wc30+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvVGV4dEJ1dHRvbj5cbiAgICApfVxuICA8Lz5cbikpO1xuXG5jb25zdCBSb290ID0gc3R5bGVkLmRpdmBcbiAgbWluLXdpZHRoOiAwcHg7XG5gO1xuXG5leHBvcnQgY29uc3QgU2VsZWN0ID0gKHtcbiAgb3B0aW9ucyxcbiAgdmFsdWUsXG4gIG9uQ2hhbmdlLFxuICBtZW51UGxhY2VtZW50ID0gZGVmYXVsdFByb3BzLm1lbnVQbGFjZW1lbnQsXG4gIG1lbnVXaWR0aCxcbiAgbWVudU1heEhlaWdodCA9IGRlZmF1bHRQcm9wcy5tZW51TWF4SGVpZ2h0LFxuICB0cmlnZ2VyOiBjdXN0b21UcmlnZ2VyLFxuICB0cmlnZ2VyVmFyaWFudCA9ICdib3gnLFxuICBsYWJlbCxcbiAgaGludCxcbiAgbWVudVpJbmRleCxcbiAgbWVudUl0ZW1zSGF2ZUVsbGlwc2lzID0gdHJ1ZSxcbn0pID0+IHtcbiAgY29uc3QgZmllbGRNYXJnaW5Ub3AgPSBsYWJlbCB8fCBoaW50ID8gJ2JldHdlZW5Gb3JtQ29udHJvbEFuZExhYmVsJyA6IHVuZGVmaW5lZDtcbiAgY29uc3Qgc2VsZWN0ZWRPcHRpb24gPSB1c2VNZW1vKFxuICAgICgpID0+IG9wdGlvbnMuZmluZChvcHRpb24gPT4gb3B0aW9uLnZhbHVlID09PSB2YWx1ZSksXG4gICAgW29wdGlvbnMsIHZhbHVlXSxcbiAgKTtcblxuICBjb25zdCB7IHJlZmVyZW5jZSwgZmxvYXRpbmcsIGZsb2F0aW5nU3R5bGVzIH0gPSB1c2VNZW51UG9zaXRpb24oeyBtZW51V2lkdGgsIG1lbnVQbGFjZW1lbnQgfSk7XG5cbiAgY29uc3Qgcm9vdFdpZHRoID0gdHJpZ2dlclZhcmlhbnQgPT09ICd0ZXh0JyA/ICdmaXQtY29udGVudCcgOiB1bmRlZmluZWQ7XG5cbiAgcmV0dXJuIChcbiAgICA8TGlzdGJveCBhcz17Um9vdH0gdmFsdWU9e3ZhbHVlfSBvbkNoYW5nZT17b25DaGFuZ2V9IHdpZHRoPXtyb290V2lkdGh9PlxuICAgICAgeyh7IG9wZW4gfSkgPT4gKFxuICAgICAgICA8PlxuICAgICAgICAgIHtsYWJlbCAmJiA8TGlzdGJveC5MYWJlbCBhcz17RmllbGRMYWJlbH0+e2xhYmVsfTwvTGlzdGJveC5MYWJlbD59XG4gICAgICAgICAge2hpbnQgJiYgPEZpZWxkSGludD57aGludH08L0ZpZWxkSGludD59XG5cbiAgICAgICAgICA8Qm94IHBvc2l0aW9uPVwicmVsYXRpdmVcIiBtYXJnaW5Ub3A9e2ZpZWxkTWFyZ2luVG9wfSByZWY9e3JlZmVyZW5jZX0+XG4gICAgICAgICAgICA8TGlzdGJveC5CdXR0b24gYXM9e0ZyYWdtZW50fT5cbiAgICAgICAgICAgICAge2N1c3RvbVRyaWdnZXIgPyAoXG4gICAgICAgICAgICAgICAgY3VzdG9tVHJpZ2dlcihzZWxlY3RlZE9wdGlvbilcbiAgICAgICAgICAgICAgKSA6IChcbiAgICAgICAgICAgICAgICA8VHJpZ2dlciB0cmlnZ2VyVmFyaWFudD17dHJpZ2dlclZhcmlhbnR9PlxuICAgICAgICAgICAgICAgICAge3NlbGVjdGVkT3B0aW9uID8gc2VsZWN0ZWRPcHRpb24uY29udGVudCA6ICcnfVxuICAgICAgICAgICAgICAgIDwvVHJpZ2dlcj5cbiAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgIDwvTGlzdGJveC5CdXR0b24+XG5cbiAgICAgICAgICAgIDxGbG9hdGluZ1BvcnRhbD5cbiAgICAgICAgICAgICAgPExpc3Rib3guT3B0aW9uc1xuICAgICAgICAgICAgICAgIHN0YXRpY1xuICAgICAgICAgICAgICAgIGlzT3Blbj17b3Blbn1cbiAgICAgICAgICAgICAgICBjbGFzc05hbWU9XCJncm4tY29udGV4dFwiXG4gICAgICAgICAgICAgICAgcmVmPXtmbG9hdGluZ31cbiAgICAgICAgICAgICAgICBwbGFjZW1lbnQ9e21lbnVQbGFjZW1lbnR9XG4gICAgICAgICAgICAgICAgYXM9e01lbnVDYXJkfVxuICAgICAgICAgICAgICAgIHpJbmRleD17bWVudVpJbmRleH1cbiAgICAgICAgICAgICAgICBtYXhIZWlnaHQ9e21lbnVNYXhIZWlnaHR9XG4gICAgICAgICAgICAgICAgc3R5bGU9e2Zsb2F0aW5nU3R5bGVzfVxuICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAge29wdGlvbnMubWFwKChvcHRpb24sIGluZGV4KSA9PiAoXG4gICAgICAgICAgICAgICAgICA8TGlzdGJveC5PcHRpb25cbiAgICAgICAgICAgICAgICAgICAga2V5PXtpbmRleH1cbiAgICAgICAgICAgICAgICAgICAgdmFsdWU9e29wdGlvbn1cbiAgICAgICAgICAgICAgICAgICAgYXM9e0ZyYWdtZW50fVxuICAgICAgICAgICAgICAgICAgICBkaXNhYmxlZD17b3B0aW9uLmlzRGlzYWJsZWR9XG4gICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgIHsoeyBhY3RpdmUgfSkgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgIHJldHVybiAoXG4gICAgICAgICAgICAgICAgICAgICAgICA8TWVudUl0ZW1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgaXNTZWxlY3RlZD17b3B0aW9uLnZhbHVlID09PSB2YWx1ZX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgaXNBY3RpdmU9e2FjdGl2ZX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgaXNEaXNhYmxlZD17b3B0aW9uLmlzRGlzYWJsZWR9XG4gICAgICAgICAgICAgICAgICAgICAgICAgIGhhc0VsbGlwc2lzPXttZW51SXRlbXNIYXZlRWxsaXBzaXN9XG4gICAgICAgICAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICAgICAgICAgIHtvcHRpb24uY29udGVudH1cbiAgICAgICAgICAgICAgICAgICAgICAgIDwvTWVudUl0ZW0+XG4gICAgICAgICAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgIDwvTGlzdGJveC5PcHRpb24+XG4gICAgICAgICAgICAgICAgKSl9XG4gICAgICAgICAgICAgIDwvTGlzdGJveC5PcHRpb25zPlxuICAgICAgICAgICAgPC9GbG9hdGluZ1BvcnRhbD5cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgPC8+XG4gICAgICApfVxuICAgIDwvTGlzdGJveD5cbiAgKTtcbn07XG5cblNlbGVjdC5MYWJlbCA9IEZpZWxkTGFiZWw7XG5TZWxlY3QuSGludCA9IEZpZWxkSGludDtcblxuU2VsZWN0LnByb3BUeXBlcyA9IHtcbiAgb3B0aW9uczogUHJvcFR5cGVzLmFycmF5LFxuICB2YWx1ZTogUHJvcFR5cGVzLnN0cmluZy5pc1JlcXVpcmVkLFxuICBvbkNoYW5nZTogUHJvcFR5cGVzLmZ1bmMuaXNSZXF1aXJlZCxcbiAgbWVudVBsYWNlbWVudDogdHlwZXMubWVudVBsYWNlbWVudCxcbiAgbWVudVdpZHRoOiB0eXBlcy5kaW1lbnNpb24sXG4gIG1lbnVaSW5kZXg6IHR5cGVzLnpJbmRleCxcbiAgbWVudU1heEhlaWdodDogdHlwZXMuZGltZW5zaW9uLFxuICB0cmlnZ2VyOiBQcm9wVHlwZXMuZnVuYyxcbiAgdHJpZ2dlclZhcmlhbnQ6IFByb3BUeXBlcy5vbmVPZihbJ2JveCcsICd0ZXh0J10pLFxuICBsYWJlbDogdHlwZXMubGFiZWwsXG4gIGhpbnQ6IHR5cGVzLmhpbnQsXG4gIG1lbnVJdGVtc0hhdmVFbGxpcHNpczogUHJvcFR5cGVzLmJvb2wsXG59O1xuIl19 */",
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, ...props }) => (\n  <FloatingPortal>\n    <Listbox.Options static className=\"grn-context\" ref={floating} as={MenuCard} {...props}>\n      {children}\n    </Listbox.Options>\n  </FloatingPortal>\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}) => {\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            >\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};\n"]} */",
71
99
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
72
100
  });
73
101
 
74
- export var Select = function Select(_ref2) {
75
- var options = _ref2.options,
76
- value = _ref2.value,
77
- onChange = _ref2.onChange,
78
- _ref2$menuPlacement = _ref2.menuPlacement,
79
- menuPlacement = _ref2$menuPlacement === void 0 ? defaultProps.menuPlacement : _ref2$menuPlacement,
80
- menuWidth = _ref2.menuWidth,
81
- _ref2$menuMaxHeight = _ref2.menuMaxHeight,
82
- menuMaxHeight = _ref2$menuMaxHeight === void 0 ? defaultProps.menuMaxHeight : _ref2$menuMaxHeight,
83
- customTrigger = _ref2.trigger,
84
- _ref2$triggerVariant = _ref2.triggerVariant,
85
- triggerVariant = _ref2$triggerVariant === void 0 ? 'box' : _ref2$triggerVariant,
86
- label = _ref2.label,
87
- hint = _ref2.hint,
88
- menuZIndex = _ref2.menuZIndex,
89
- _ref2$menuItemsHaveEl = _ref2.menuItemsHaveEllipsis,
90
- menuItemsHaveEllipsis = _ref2$menuItemsHaveEl === void 0 ? true : _ref2$menuItemsHaveEl;
91
- var fieldMarginTop = label || hint ? 'betweenFormControlAndLabel' : undefined;
92
- var selectedOption = useMemo(function () {
93
- return options.find(function (option) {
94
- return option.value === value;
102
+ var SelectMenuOption = function SelectMenuOption(_ref3) {
103
+ var option = _ref3.option,
104
+ isSelected = _ref3.isSelected,
105
+ menuItemsHaveEllipsis = _ref3.menuItemsHaveEllipsis,
106
+ props = _objectWithoutProperties(_ref3, _excluded2);
107
+
108
+ return ___EmotionJSX(Listbox.Option, _extends({
109
+ value: option,
110
+ as: Fragment,
111
+ disabled: option.isDisabled
112
+ }, props), function (_ref4) {
113
+ var active = _ref4.active;
114
+ return ___EmotionJSX(MenuItem, {
115
+ isSelected: isSelected,
116
+ isActive: active,
117
+ isDisabled: option.isDisabled,
118
+ hasEllipsis: menuItemsHaveEllipsis
119
+ }, option.content);
120
+ });
121
+ };
122
+
123
+ var SelectInfo = function SelectInfo(_ref5) {
124
+ var label = _ref5.label,
125
+ hint = _ref5.hint;
126
+ return ___EmotionJSX(React.Fragment, null, (label || hint) && ___EmotionJSX(Box, {
127
+ marginBottom: "betweenFormControlAndLabel"
128
+ }, label && ___EmotionJSX(Listbox.Label, {
129
+ as: FieldLabel
130
+ }, label), hint && ___EmotionJSX(FieldHint, null, hint)));
131
+ };
132
+
133
+ var SelectMenu = function SelectMenu(_ref6) {
134
+ var children = _ref6.children,
135
+ floating = _ref6.floating,
136
+ props = _objectWithoutProperties(_ref6, _excluded3);
137
+
138
+ return ___EmotionJSX(FloatingPortal, null, ___EmotionJSX(Listbox.Options, _extends({
139
+ static: true,
140
+ className: "grn-context",
141
+ ref: floating,
142
+ as: MenuCard
143
+ }, props), children));
144
+ };
145
+
146
+ var getIsSelected = function getIsSelected(isMultiple, option, selected) {
147
+ if (isMultiple) {
148
+ return selected.some(function (selectedOption) {
149
+ return selectedOption.content === option.content;
95
150
  });
96
- }, [options, value]);
151
+ }
152
+
153
+ return option.value === selected.value;
154
+ };
155
+
156
+ export var Select = function Select(_ref7) {
157
+ var options = _ref7.options,
158
+ value = _ref7.value,
159
+ onChange = _ref7.onChange,
160
+ _ref7$menuPlacement = _ref7.menuPlacement,
161
+ menuPlacement = _ref7$menuPlacement === void 0 ? defaultProps.menuPlacement : _ref7$menuPlacement,
162
+ menuWidth = _ref7.menuWidth,
163
+ _ref7$menuMaxHeight = _ref7.menuMaxHeight,
164
+ menuMaxHeight = _ref7$menuMaxHeight === void 0 ? defaultProps.menuMaxHeight : _ref7$menuMaxHeight,
165
+ customTrigger = _ref7.trigger,
166
+ _ref7$triggerVariant = _ref7.triggerVariant,
167
+ triggerVariant = _ref7$triggerVariant === void 0 ? 'box' : _ref7$triggerVariant,
168
+ label = _ref7.label,
169
+ hint = _ref7.hint,
170
+ menuZIndex = _ref7.menuZIndex,
171
+ _ref7$menuItemsHaveEl = _ref7.menuItemsHaveEllipsis,
172
+ menuItemsHaveEllipsis = _ref7$menuItemsHaveEl === void 0 ? true : _ref7$menuItemsHaveEl,
173
+ allowMultiple = _ref7.allowMultiple;
97
174
 
98
175
  var _useMenuPosition = useMenuPosition({
99
176
  menuWidth: menuWidth,
@@ -104,56 +181,58 @@ export var Select = function Select(_ref2) {
104
181
  floatingStyles = _useMenuPosition.floatingStyles;
105
182
 
106
183
  var rootWidth = triggerVariant === 'text' ? 'fit-content' : undefined;
184
+ var optionMap = useMemo(function () {
185
+ return new Map(options.map(function (option) {
186
+ return [option.value, option];
187
+ }));
188
+ }, [options]);
189
+ var selected = useMemo(function () {
190
+ return allowMultiple ? value.map(function (value) {
191
+ return optionMap.get(value);
192
+ }) : optionMap.get(value);
193
+ }, [allowMultiple, optionMap, value]);
107
194
  return ___EmotionJSX(Listbox, {
108
195
  as: Root,
109
- value: value,
196
+ value: selected,
110
197
  onChange: onChange,
111
- width: rootWidth
112
- }, function (_ref3) {
113
- var open = _ref3.open;
114
- return ___EmotionJSX(React.Fragment, null, label && ___EmotionJSX(Listbox.Label, {
115
- as: FieldLabel
116
- }, label), hint && ___EmotionJSX(FieldHint, null, hint), ___EmotionJSX(Box, {
198
+ width: rootWidth,
199
+ multiple: allowMultiple
200
+ }, function (_ref8) {
201
+ var open = _ref8.open;
202
+ return ___EmotionJSX(React.Fragment, null, ___EmotionJSX(SelectInfo, {
203
+ label: label,
204
+ hint: hint
205
+ }), ___EmotionJSX(Box, {
117
206
  position: "relative",
118
- marginTop: fieldMarginTop,
119
207
  ref: reference
120
- }, ___EmotionJSX(Listbox.Button, {
121
- as: Fragment
122
- }, customTrigger ? customTrigger(selectedOption) : ___EmotionJSX(Trigger, {
123
- triggerVariant: triggerVariant
124
- }, selectedOption ? selectedOption.content : '')), ___EmotionJSX(FloatingPortal, null, ___EmotionJSX(Listbox.Options, {
125
- static: true,
208
+ }, ___EmotionJSX(SelectButton, {
209
+ customTrigger: customTrigger,
210
+ triggerVariant: triggerVariant,
211
+ isMultiple: allowMultiple,
212
+ selected: selected
213
+ }), ___EmotionJSX(SelectMenu, {
126
214
  isOpen: open,
127
- className: "grn-context",
128
- ref: floating,
129
215
  placement: menuPlacement,
130
- as: MenuCard,
131
- zIndex: menuZIndex,
132
216
  maxHeight: menuMaxHeight,
217
+ zIndex: menuZIndex,
218
+ floating: floating,
133
219
  style: floatingStyles
134
220
  }, options.map(function (option, index) {
135
- return ___EmotionJSX(Listbox.Option, {
221
+ var isSelected = getIsSelected(allowMultiple, option, selected);
222
+ return ___EmotionJSX(SelectMenuOption, {
223
+ isSelected: isSelected,
136
224
  key: index,
137
- value: option,
138
- as: Fragment,
139
- disabled: option.isDisabled
140
- }, function (_ref4) {
141
- var active = _ref4.active;
142
- return ___EmotionJSX(MenuItem, {
143
- isSelected: option.value === value,
144
- isActive: active,
145
- isDisabled: option.isDisabled,
146
- hasEllipsis: menuItemsHaveEllipsis
147
- }, option.content);
225
+ option: option,
226
+ hasEllipsis: menuItemsHaveEllipsis
148
227
  });
149
- })))));
228
+ }))));
150
229
  });
151
230
  };
152
231
  Select.Label = FieldLabel;
153
232
  Select.Hint = FieldHint;
154
233
  Select.propTypes = {
155
234
  options: PropTypes.array,
156
- value: PropTypes.string.isRequired,
235
+ value: PropTypes.oneOfType([PropTypes.string, PropTypes.array]).isRequired,
157
236
  onChange: PropTypes.func.isRequired,
158
237
  menuPlacement: types.menuPlacement,
159
238
  menuWidth: types.dimension,
@@ -163,5 +242,6 @@ Select.propTypes = {
163
242
  triggerVariant: PropTypes.oneOf(['box', 'text']),
164
243
  label: types.label,
165
244
  hint: types.hint,
166
- menuItemsHaveEllipsis: PropTypes.bool
245
+ menuItemsHaveEllipsis: PropTypes.bool,
246
+ allowMultiple: PropTypes.bool
167
247
  };
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@flodesk/grain",
3
- "version": "7.2.0",
3
+ "version": "7.3.1",
4
4
  "description": "Flodesk design system",
5
5
  "module": "es/index.js",
6
6
  "author": "Flodesk",
@@ -12,7 +12,8 @@
12
12
  "start": "next start",
13
13
  "lint": "eslint --ext .js --ext .ts --ext .jsx --ext .mdx .",
14
14
  "semantic-release": "semantic-release",
15
- "build:es": "del es/* && cross-env NODE_ENV=production babel --no-babelrc --config-file ./babel.config.es.js --out-dir es --copy-files src"
15
+ "build:es": "del es/* && cross-env NODE_ENV=production babel --no-babelrc --config-file ./babel.config.es.js --out-dir es --copy-files src",
16
+ "cc": "node create-component.js"
16
17
  },
17
18
  "publishConfig": {
18
19
  "access": "public"