@flodesk/grain 11.47.2 → 11.48.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.
@@ -207,9 +207,7 @@ export const Autocomplete = _ref4 => {
207
207
  const showGroupTitle = getShowGroupTitle(index, option, filteredOptions);
208
208
  return ___EmotionJSX(Fragment, {
209
209
  key: index
210
- }, showGroupTitle && ___EmotionJSX(MenuGroupTitle, {
211
- hasDivider: index > 0
212
- }, option.groupTitle), ___EmotionJSX(Combobox.Option, {
210
+ }, showGroupTitle && ___EmotionJSX(MenuGroupTitle, null, option.groupTitle), ___EmotionJSX(Combobox.Option, {
213
211
  value: option,
214
212
  as: Fragment,
215
213
  disabled: option.isDisabled
@@ -1,7 +1,7 @@
1
1
  import _styled from "@emotion/styled/base";
2
2
  const _excluded = ["onClick"],
3
3
  _excluded2 = ["children", "hasEllipsis", "paddingLeft"],
4
- _excluded3 = ["options", "value", "onChange", "isCreatable", "onCreate", "menuPlacement", "menuWidth", "menuMaxHeight", "menuZIndex", "placeholder", "label", "hint", "menuItemsHaveEllipsis", "hasPortal", "hasError", "errorMessage", "isDisabled", "searchField", "backgroundColor", "onClear", "hasSpinner", "size"];
4
+ _excluded3 = ["options", "value", "onChange", "isCreatable", "onCreate", "menuPlacement", "menuWidth", "menuMaxHeight", "menuZIndex", "placeholder", "label", "hint", "menuItemsHaveEllipsis", "hasPortal", "hasError", "errorMessage", "isDisabled", "searchField", "backgroundColor", "onClear", "hasSpinner", "size", "renderGroupTitle"];
5
5
  import "core-js/modules/web.dom-collections.iterator.js";
6
6
 
7
7
  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); }
@@ -71,7 +71,7 @@ const Root = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "production" ?
71
71
  open
72
72
  } = _ref8;
73
73
  return open ? 'auto' : 'none';
74
- }, ";transition:opacity ", transitionSpeed, ";&:focus-visible,&:placeholder-shown{opacity:1;pointer-events:auto;transition:opacity ", transitionSpeed, ";}&:disabled:placeholder-shown{background:transparent;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/autocomplete2.jsx"],"names":[],"mappings":"AAwBuB","file":"../../src/components/autocomplete2.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useState } from 'react';\nimport { Icon, Box, Arrange, Spinner } from '.';\nimport { IconChevronDown, IconPlus, IconCross } from '../icons';\nimport { Combobox } from '@headlessui/react';\nimport { defaultProps, types } from '../types';\nimport {\n  InputField,\n  MenuCard,\n  MenuCardTransition,\n  MenuGroupTitle,\n  FieldLabel,\n  FieldHint,\n  styles,\n  FieldClearButton,\n} from '../foundational';\nimport { MenuItemDivider, useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\nimport { Global } from '@emotion/react';\nimport { getColor, getIconSize, getRadius, getSpace, getTransition } from '../utilities';\nimport styled from '@emotion/styled';\n\nconst transitionSpeed = getTransition('xxFast');\n\nconst Root = styled.div`\n  ${styles.transitions};\n  ${({ hasError, isDisabled }) =>\n    !hasError &&\n    !isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color);\n      &:hover { --borderColor: var(--grn-field-border-color-hover) };\n      &:focus-within { --borderColor: var(--grn-field-border-color-focus) };\n    `};\n\n  ${({ hasError }) => hasError && `--borderColor: ${getColor('danger')};`};\n\n  ${({ backgroundColor, isDisabled }) =>\n    !backgroundColor &&\n    !isDisabled &&\n    `\n      --backgroundColor: transparent;\n    `};\n\n  ${({ backgroundColor }) =>\n    backgroundColor &&\n    `\n      --backgroundColor: ${getColor(backgroundColor)};\n    `};\n\n  ${({ isDisabled }) =>\n    isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color-disabled);\n      --backgroundColor: var(--grn-field-background-disabled);\n\n      * {\n        pointer-events: none;\n      }\n    `};\n\n  position: relative;\n  margin-top: ${({ marginTop }) => getSpace(marginTop)};\n  border-radius: ${getRadius('s')};\n  box-shadow: inset 0 0 0 var(--grn-border-size) var(--borderColor);\n  background-color: var(--backgroundColor);\n\n  .autocompleteInput {\n    line-height: 20px;\n    border: none;\n    opacity: ${({ open }) => (open ? 1 : 0)};\n    pointer-events: ${({ open }) => (open ? 'auto' : 'none')};\n    transition: opacity ${transitionSpeed};\n\n    &:focus-visible,\n    &:placeholder-shown {\n      opacity: 1;\n      pointer-events: auto;\n      transition: opacity ${transitionSpeed};\n    }\n    &:disabled:placeholder-shown {\n      background: transparent;\n    }\n  }\n`;\n\nconst EmptyState = () => (\n  <Box paddingY=\"s\" paddingX=\"m\" color=\"content2\">\n    No results\n  </Box>\n);\n\nconst rightSectionGap = 'var(--grn-space-m)';\n\nconst RightSection = ({ children, right }) => (\n  <Arrange\n    right={right}\n    gap={rightSectionGap}\n    position=\"absolute\"\n    top=\"0px\"\n    bottom=\"0px\"\n    margin=\"auto\"\n    height=\"fit-content\"\n  >\n    {children}\n  </Arrange>\n);\n\nconst ClearButton = ({ onClick, ...props }) => {\n  const handleClear = e => {\n    e.stopPropagation();\n    onClick();\n  };\n  return <FieldClearButton onClick={e => handleClear(e)} {...props} />;\n};\n\nconst getFilteredOptions = (query, options, searchField) => {\n  if (query === '') return options;\n\n  return options.filter(option => option[searchField].toLowerCase().includes(query.toLowerCase()));\n};\n\nconst getShowGroupTitle = (index, option, filteredOptions) => {\n  const firstOptionHasTitle = index === 0 && option.groupTitle;\n  const titleChanged = index > 0 && option.groupTitle !== filteredOptions[index - 1].groupTitle;\n\n  return firstOptionHasTitle || titleChanged;\n};\n\nconst menuItemStyles = `\n  .autocompleteMenuItem {\n    display: flex;\n    align-items: center;\n    list-style: none;\n    min-height: var(--grn-textbox-height-m);\n    border-radius: ${getRadius('s')};\n    appearance: none;\n    padding-block: 0;\n    padding: 4px 12px;\n\n    --checkUri: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%2337312f73\"%3E%3C/path%3E%3C/svg%3E');\n    --checkUriActive: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%23241a17\"%3E%3C/path%3E%3C/svg%3E');\n\n    &[data-headlessui-state='active'],\n    &[data-headlessui-state='active selected'] {\n      background-color: ${getColor('fade1')};\n    }\n\n    &[data-headlessui-state='selected'],\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUri);\n      background-repeat: no-repeat;\n      background-position: center right 12px;\n      background-size: auto ${getIconSize('m')};\n    }\n\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUriActive);\n    }\n\n    &[data-headlessui-state='disabled'] {\n      color: ${getColor('disabledContent')};\n      cursor: default;\n    }\n\n    ul & {\n      cursor: pointer;\n      padding-right: 40px;\n    }\n  }\n\n  .autocompleteMenuItemText {\n    flex-grow: 1;\n\n    &.hasEllipsis {\n      white-space: nowrap;\n      overflow: hidden;\n      text-overflow: ellipsis;\n    }\n  }\n`;\n\nconst Item = forwardRef(({ children, hasEllipsis, paddingLeft = '12px', ...props }, ref) => {\n  return (\n    <li\n      className=\"autocompleteMenuItem\"\n      ref={ref}\n      style={{ paddingLeft: getSpace(paddingLeft) }}\n      {...props}\n    >\n      <span\n        className={\n          hasEllipsis ? 'autocompleteMenuItemText hasEllipsis' : 'autocompleteMenuItemText'\n        }\n      >\n        {children}\n      </span>\n    </li>\n  );\n});\n\nconst Trigger = styled.button`\n  text-align: left;\n  height: ${({ size }) => `var(--grn-textbox-height-${size})`};\n  background-color: transparent;\n  cursor: text;\n  appearance: none;\n  display: block;\n  font: inherit;\n  color: inherit;\n  width: 100%;\n  outline: none;\n  border: none;\n  padding: 0;\n  opacity: ${({ open }) => (open ? 0 : 1)};\n  pointer-events: ${({ open }) => (open ? 'none' : 'auto')};\n  position: absolute;\n  top: 0;\n  left: 0;\n  transition: opacity ${transitionSpeed};\n\n  .autocompleteMenuItem {\n    padding-right: ${({ paddingRight }) => paddingRight};\n  }\n\n  .autocompleteInput:focus-visible + & {\n    opacity: 0;\n    pointer-events: none;\n  }\n`;\n\nconst TriggerText = styled.span`\n  line-height: 20px;\n  display: block;\n  padding-block: 0;\n  padding-right: ${({ paddingRight }) => paddingRight};\n  padding-left: ${({ paddingLeft }) => paddingLeft};\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`;\n\nexport const Autocomplete2 = ({\n  options,\n  value,\n  onChange,\n  isCreatable,\n  onCreate,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  menuZIndex,\n  placeholder,\n  label,\n  hint,\n  menuItemsHaveEllipsis = true,\n  hasPortal = true,\n  hasError,\n  errorMessage,\n  isDisabled,\n  searchField = 'content',\n  backgroundColor,\n  onClear,\n  hasSpinner,\n  size = 'm',\n  ...props\n}) => {\n  const [query, setQuery] = useState('');\n\n  const handleCreatableChange = option => {\n    if (onCreate) {\n      const isCreatableOption = !options.find(item => item.value === option.value);\n      if (isCreatableOption) {\n        onCreate(option);\n        return;\n      }\n    }\n    onChange(option);\n  };\n\n  const handleChange = option => {\n    if (isCreatable) {\n      handleCreatableChange(option);\n    } else {\n      onChange(option);\n    }\n  };\n\n  const filteredOptions = getFilteredOptions(query, options, searchField);\n\n  const isShowCreateOption = isCreatable && query.length > 0;\n\n  const noResults = !Boolean(filteredOptions.length) && !isShowCreateOption;\n\n  const fieldMarginTop = label || hint ? 'var(--grn-form-control-label-gap)' : undefined;\n\n  const selectedOption = useMemo(\n    () => options.find(option => option.value === value),\n    [options, value],\n  );\n\n  const newOption = useMemo(() => ({ value: query, content: query }), [query]);\n\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n\n  const OptionsRoot = hasPortal ? FloatingPortal : Fragment;\n\n  const hasClearButton = onClear && Boolean(selectedOption);\n\n  const inputPaddingX = `var(--grn-field-paddingX-${size})`;\n\n  const chevronWidth = `${IconChevronDown.viewBoxWidth / 2}px`;\n  const textGap = 'var(--grn-space-s2)';\n  const spinnerSpace = `calc(var(--grn-unit) * 3) + ${rightSectionGap}`;\n  const iconCrossWidth = `${IconCross.viewBoxWidth / 2}px`;\n  const clearButtonSpace = `${iconCrossWidth} + ${rightSectionGap}`;\n\n  const paddingRightValues = [textGap];\n  if (hasSpinner) paddingRightValues.push(spinnerSpace);\n  if (hasClearButton) paddingRightValues.push(clearButtonSpace);\n  paddingRightValues.push(chevronWidth, inputPaddingX);\n\n  const inputPaddingRight = `calc(${paddingRightValues.join(' + ')})`;\n\n  return (\n    <>\n      <Global styles={menuItemStyles} />\n      <Combobox\n        disabled={isDisabled}\n        as=\"div\"\n        value={selectedOption || ''}\n        onChange={handleChange}\n        {...props}\n      >\n        {({ open }) => {\n          const hasItem = selectedOption?.item;\n          return (\n            <>\n              {label && <Combobox.Label as={FieldLabel}>{label}</Combobox.Label>}\n              {hint && <FieldHint>{hint}</FieldHint>}\n\n              <Root\n                marginTop={fieldMarginTop}\n                ref={reference}\n                open={open}\n                hasError={hasError}\n                backgroundColor={backgroundColor}\n                isDisabled={isDisabled}\n              >\n                <Combobox.Button as={Box} position=\"relative\">\n                  <Combobox.Input\n                    as={InputField}\n                    autoComplete=\"off\"\n                    onChange={event => setQuery(event.target.value)}\n                    placeholder={placeholder}\n                    displayValue={option => option && option.content}\n                    paddingRight={inputPaddingRight}\n                    hasError={hasError}\n                    className=\"autocompleteInput\"\n                    size={size}\n                  />\n\n                  <Trigger open={open} tabIndex={-1} paddingRight={inputPaddingRight} size={size}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight} paddingLeft={inputPaddingX}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <RightSection right={inputPaddingX}>\n                    {hasSpinner && <Spinner />}\n                    {hasClearButton && <ClearButton onClick={onClear} trimSides />}\n                    <Icon icon={<IconChevronDown />} />\n                  </RightSection>\n                </Combobox.Button>\n\n                <OptionsRoot>\n                  <MenuCardTransition afterLeave={() => setQuery('')}>\n                    <Combobox.Options\n                      static\n                      ref={floating}\n                      placement={menuPlacement}\n                      maxHeight={menuMaxHeight}\n                      as={MenuCard}\n                      zIndex={menuZIndex}\n                      style={floatingStyles}\n                      isOpen={open}\n                    >\n                      {filteredOptions.map((option, index) => {\n                        const showGroupTitle = getShowGroupTitle(index, option, filteredOptions);\n\n                        return (\n                          <Fragment key={index}>\n                            {showGroupTitle && (\n                              <MenuGroupTitle hasDivider={index > 0}>\n                                {option.groupTitle}\n                              </MenuGroupTitle>\n                            )}\n\n                            <Combobox.Option\n                              value={option}\n                              as={Fragment}\n                              disabled={option.isDisabled}\n                            >\n                              {({ active }) => {\n                                const isDisabled = option.isDisabled;\n                                const isSelected = option.value === value;\n                                const isActive = !option.isDisabled && active;\n                                const hasEllipsis = menuItemsHaveEllipsis;\n                                if (option.item) {\n                                  return option.item({\n                                    isActive,\n                                    isSelected,\n                                    isDisabled,\n                                    hasEllipsis,\n                                  });\n                                }\n                                return <Item hasEllipsis={hasEllipsis}>{option.content}</Item>;\n                              }}\n                            </Combobox.Option>\n                          </Fragment>\n                        );\n                      })}\n                      {isShowCreateOption && (\n                        <>\n                          {filteredOptions.length > 0 && <MenuItemDivider />}\n                          <Combobox.Option value={newOption} as={Fragment}>\n                            {({ active }) => (\n                              <Item isActive={active} hasEllipsis={menuItemsHaveEllipsis}>\n                                <Arrange gap=\"s\" tag=\"span\">\n                                  <Icon icon={<IconPlus />} color={active ? 'content' : 'icon'} />\n                                  Create {`\"` + query + `\"`}\n                                </Arrange>\n                              </Item>\n                            )}\n                          </Combobox.Option>\n                        </>\n                      )}\n                      {noResults && <EmptyState />}\n                    </Combobox.Options>\n                  </MenuCardTransition>\n                </OptionsRoot>\n              </Root>\n              {errorMessage && (\n                <Box marginTop=\"var(--grn-form-control-label-gap)\" color=\"danger\">\n                  {errorMessage}\n                </Box>\n              )}\n            </>\n          );\n        }}\n      </Combobox>\n    </>\n  );\n};\n\nAutocomplete2.Label = FieldLabel;\nAutocomplete2.Hint = FieldHint;\nAutocomplete2.Item = Item;\n\nAutocomplete2.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.string,\n  onChange: PropTypes.func.isRequired,\n  isCreatable: PropTypes.bool,\n  onCreate: PropTypes.func,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuMaxHeight: types.dimension,\n  menuZIndex: types.zIndex,\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n  placeholder: PropTypes.string,\n  hasError: PropTypes.bool,\n  errorMessage: PropTypes.string,\n  isDisabled: PropTypes.bool,\n  backgroundColor: types.color,\n  onClear: PropTypes.func,\n  hasSpinner: PropTypes.bool,\n  size: PropTypes.oneOf(['m', 'l']),\n};\n"]} */"));
74
+ }, ";transition:opacity ", transitionSpeed, ";&:focus-visible,&:placeholder-shown{opacity:1;pointer-events:auto;transition:opacity ", transitionSpeed, ";}&:disabled:placeholder-shown{background:transparent;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/autocomplete2.jsx"],"names":[],"mappings":"AAwBuB","file":"../../src/components/autocomplete2.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useState } from 'react';\nimport { Icon, Box, Arrange, Spinner } from '.';\nimport { IconChevronDown, IconPlus, IconCross } from '../icons';\nimport { Combobox } from '@headlessui/react';\nimport { defaultProps, types } from '../types';\nimport {\n  InputField,\n  MenuCard,\n  MenuCardTransition,\n  MenuGroupTitle,\n  FieldLabel,\n  FieldHint,\n  styles,\n  FieldClearButton,\n} from '../foundational';\nimport { MenuItemDivider, useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\nimport { Global } from '@emotion/react';\nimport { getColor, getIconSize, getRadius, getSpace, getTransition } from '../utilities';\nimport styled from '@emotion/styled';\n\nconst transitionSpeed = getTransition('xxFast');\n\nconst Root = styled.div`\n  ${styles.transitions};\n  ${({ hasError, isDisabled }) =>\n    !hasError &&\n    !isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color);\n      &:hover { --borderColor: var(--grn-field-border-color-hover) };\n      &:focus-within { --borderColor: var(--grn-field-border-color-focus) };\n    `};\n\n  ${({ hasError }) => hasError && `--borderColor: ${getColor('danger')};`};\n\n  ${({ backgroundColor, isDisabled }) =>\n    !backgroundColor &&\n    !isDisabled &&\n    `\n      --backgroundColor: transparent;\n    `};\n\n  ${({ backgroundColor }) =>\n    backgroundColor &&\n    `\n      --backgroundColor: ${getColor(backgroundColor)};\n    `};\n\n  ${({ isDisabled }) =>\n    isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color-disabled);\n      --backgroundColor: var(--grn-field-background-disabled);\n\n      * {\n        pointer-events: none;\n      }\n    `};\n\n  position: relative;\n  margin-top: ${({ marginTop }) => getSpace(marginTop)};\n  border-radius: ${getRadius('s')};\n  box-shadow: inset 0 0 0 var(--grn-border-size) var(--borderColor);\n  background-color: var(--backgroundColor);\n\n  .autocompleteInput {\n    line-height: 20px;\n    border: none;\n    opacity: ${({ open }) => (open ? 1 : 0)};\n    pointer-events: ${({ open }) => (open ? 'auto' : 'none')};\n    transition: opacity ${transitionSpeed};\n\n    &:focus-visible,\n    &:placeholder-shown {\n      opacity: 1;\n      pointer-events: auto;\n      transition: opacity ${transitionSpeed};\n    }\n    &:disabled:placeholder-shown {\n      background: transparent;\n    }\n  }\n`;\n\nconst EmptyState = () => (\n  <Box paddingY=\"s\" paddingX=\"m\" color=\"content2\">\n    No results\n  </Box>\n);\n\nconst rightSectionGap = 'var(--grn-space-m)';\n\nconst RightSection = ({ children, right }) => (\n  <Arrange\n    right={right}\n    gap={rightSectionGap}\n    position=\"absolute\"\n    top=\"0px\"\n    bottom=\"0px\"\n    margin=\"auto\"\n    height=\"fit-content\"\n  >\n    {children}\n  </Arrange>\n);\n\nconst ClearButton = ({ onClick, ...props }) => {\n  const handleClear = e => {\n    e.stopPropagation();\n    onClick();\n  };\n  return <FieldClearButton onClick={e => handleClear(e)} {...props} />;\n};\n\nconst getFilteredOptions = (query, options, searchField) => {\n  if (query === '') return options;\n\n  return options.filter(option => option[searchField].toLowerCase().includes(query.toLowerCase()));\n};\n\nconst getShowGroupTitle = (index, option, filteredOptions) => {\n  const firstOptionHasTitle = index === 0 && option.groupTitle;\n  const titleChanged = index > 0 && option.groupTitle !== filteredOptions[index - 1].groupTitle;\n\n  return firstOptionHasTitle || titleChanged;\n};\n\nconst menuItemStyles = `\n  .autocompleteMenuItem {\n    display: flex;\n    align-items: center;\n    list-style: none;\n    min-height: var(--grn-textbox-height-m);\n    border-radius: ${getRadius('s')};\n    appearance: none;\n    padding-block: 0;\n    padding: 4px 12px;\n\n    --checkUri: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%2337312f73\"%3E%3C/path%3E%3C/svg%3E');\n    --checkUriActive: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%23241a17\"%3E%3C/path%3E%3C/svg%3E');\n\n    &[data-headlessui-state='active'],\n    &[data-headlessui-state='active selected'] {\n      background-color: ${getColor('fade1')};\n    }\n\n    &[data-headlessui-state='selected'],\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUri);\n      background-repeat: no-repeat;\n      background-position: center right 12px;\n      background-size: auto ${getIconSize('m')};\n    }\n\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUriActive);\n    }\n\n    &[data-headlessui-state='disabled'] {\n      color: ${getColor('disabledContent')};\n      cursor: default;\n    }\n\n    ul & {\n      cursor: pointer;\n      padding-right: 40px;\n    }\n  }\n\n  .autocompleteMenuItemText {\n    flex-grow: 1;\n\n    &.hasEllipsis {\n      white-space: nowrap;\n      overflow: hidden;\n      text-overflow: ellipsis;\n    }\n  }\n`;\n\nconst Item = forwardRef(({ children, hasEllipsis, paddingLeft = '12px', ...props }, ref) => {\n  return (\n    <li\n      className=\"autocompleteMenuItem\"\n      ref={ref}\n      style={{ paddingLeft: getSpace(paddingLeft) }}\n      {...props}\n    >\n      <span\n        className={\n          hasEllipsis ? 'autocompleteMenuItemText hasEllipsis' : 'autocompleteMenuItemText'\n        }\n      >\n        {children}\n      </span>\n    </li>\n  );\n});\n\nconst Trigger = styled.button`\n  text-align: left;\n  height: ${({ size }) => `var(--grn-textbox-height-${size})`};\n  background-color: transparent;\n  cursor: text;\n  appearance: none;\n  display: block;\n  font: inherit;\n  color: inherit;\n  width: 100%;\n  outline: none;\n  border: none;\n  padding: 0;\n  opacity: ${({ open }) => (open ? 0 : 1)};\n  pointer-events: ${({ open }) => (open ? 'none' : 'auto')};\n  position: absolute;\n  top: 0;\n  left: 0;\n  transition: opacity ${transitionSpeed};\n\n  .autocompleteMenuItem {\n    padding-right: ${({ paddingRight }) => paddingRight};\n  }\n\n  .autocompleteInput:focus-visible + & {\n    opacity: 0;\n    pointer-events: none;\n  }\n`;\n\nconst TriggerText = styled.span`\n  line-height: 20px;\n  display: block;\n  padding-block: 0;\n  padding-right: ${({ paddingRight }) => paddingRight};\n  padding-left: ${({ paddingLeft }) => paddingLeft};\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`;\n\nexport const Autocomplete2 = ({\n  options,\n  value,\n  onChange,\n  isCreatable,\n  onCreate,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  menuZIndex,\n  placeholder,\n  label,\n  hint,\n  menuItemsHaveEllipsis = true,\n  hasPortal = true,\n  hasError,\n  errorMessage,\n  isDisabled,\n  searchField = 'content',\n  backgroundColor,\n  onClear,\n  hasSpinner,\n  size = 'm',\n  renderGroupTitle,\n  ...props\n}) => {\n  const [query, setQuery] = useState('');\n\n  const handleCreatableChange = option => {\n    if (onCreate) {\n      const isCreatableOption = !options.find(item => item.value === option.value);\n      if (isCreatableOption) {\n        onCreate(option);\n        return;\n      }\n    }\n    onChange(option);\n  };\n\n  const handleChange = option => {\n    if (isCreatable) {\n      handleCreatableChange(option);\n    } else {\n      onChange(option);\n    }\n  };\n\n  const filteredOptions = getFilteredOptions(query, options, searchField);\n\n  const isShowCreateOption = isCreatable && query.length > 0;\n\n  const noResults = !Boolean(filteredOptions.length) && !isShowCreateOption;\n\n  const fieldMarginTop = label || hint ? 'var(--grn-form-control-label-gap)' : undefined;\n\n  const selectedOption = useMemo(\n    () => options.find(option => option.value === value),\n    [options, value],\n  );\n\n  const newOption = useMemo(() => ({ value: query, content: query }), [query]);\n\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n\n  const OptionsRoot = hasPortal ? FloatingPortal : Fragment;\n\n  const hasClearButton = onClear && Boolean(selectedOption);\n\n  const inputPaddingX = `var(--grn-field-paddingX-${size})`;\n\n  const chevronWidth = `${IconChevronDown.viewBoxWidth / 2}px`;\n  const textGap = 'var(--grn-space-s2)';\n  const spinnerSpace = `calc(var(--grn-unit) * 3) + ${rightSectionGap}`;\n  const iconCrossWidth = `${IconCross.viewBoxWidth / 2}px`;\n  const clearButtonSpace = `${iconCrossWidth} + ${rightSectionGap}`;\n\n  const paddingRightValues = [textGap];\n  if (hasSpinner) paddingRightValues.push(spinnerSpace);\n  if (hasClearButton) paddingRightValues.push(clearButtonSpace);\n  paddingRightValues.push(chevronWidth, inputPaddingX);\n\n  const inputPaddingRight = `calc(${paddingRightValues.join(' + ')})`;\n\n  return (\n    <>\n      <Global styles={menuItemStyles} />\n      <Combobox\n        disabled={isDisabled}\n        as=\"div\"\n        value={selectedOption || ''}\n        onChange={handleChange}\n        {...props}\n      >\n        {({ open }) => {\n          const hasItem = selectedOption?.item;\n          return (\n            <>\n              {label && <Combobox.Label as={FieldLabel}>{label}</Combobox.Label>}\n              {hint && <FieldHint>{hint}</FieldHint>}\n\n              <Root\n                marginTop={fieldMarginTop}\n                ref={reference}\n                open={open}\n                hasError={hasError}\n                backgroundColor={backgroundColor}\n                isDisabled={isDisabled}\n              >\n                <Combobox.Button as={Box} position=\"relative\">\n                  <Combobox.Input\n                    as={InputField}\n                    autoComplete=\"off\"\n                    onChange={event => setQuery(event.target.value)}\n                    placeholder={placeholder}\n                    displayValue={option => option && option.content}\n                    paddingRight={inputPaddingRight}\n                    hasError={hasError}\n                    className=\"autocompleteInput\"\n                    size={size}\n                  />\n\n                  <Trigger open={open} tabIndex={-1} paddingRight={inputPaddingRight} size={size}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight} paddingLeft={inputPaddingX}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <RightSection right={inputPaddingX}>\n                    {hasSpinner && <Spinner />}\n                    {hasClearButton && <ClearButton onClick={onClear} trimSides />}\n                    <Icon icon={<IconChevronDown />} />\n                  </RightSection>\n                </Combobox.Button>\n\n                <OptionsRoot>\n                  <MenuCardTransition afterLeave={() => setQuery('')}>\n                    <Combobox.Options\n                      static\n                      ref={floating}\n                      placement={menuPlacement}\n                      maxHeight={menuMaxHeight}\n                      as={MenuCard}\n                      zIndex={menuZIndex}\n                      style={floatingStyles}\n                      isOpen={open}\n                    >\n                      {filteredOptions.map((option, index) => {\n                        const showGroupTitle = getShowGroupTitle(index, option, filteredOptions);\n\n                        return (\n                          <Fragment key={index}>\n                            {showGroupTitle &&\n                              (renderGroupTitle ? (\n                                renderGroupTitle(option.groupTitle)\n                              ) : (\n                                <MenuGroupTitle>{option.groupTitle}</MenuGroupTitle>\n                              ))}\n\n                            <Combobox.Option\n                              value={option}\n                              as={Fragment}\n                              disabled={option.isDisabled}\n                            >\n                              {({ active }) => {\n                                const isDisabled = option.isDisabled;\n                                const isSelected = option.value === value;\n                                const isActive = !option.isDisabled && active;\n                                const hasEllipsis = menuItemsHaveEllipsis;\n                                if (option.item) {\n                                  return option.item({\n                                    isActive,\n                                    isSelected,\n                                    isDisabled,\n                                    hasEllipsis,\n                                  });\n                                }\n                                return <Item hasEllipsis={hasEllipsis}>{option.content}</Item>;\n                              }}\n                            </Combobox.Option>\n                          </Fragment>\n                        );\n                      })}\n                      {isShowCreateOption && (\n                        <>\n                          {filteredOptions.length > 0 && <MenuItemDivider />}\n                          <Combobox.Option value={newOption} as={Fragment}>\n                            {({ active }) => (\n                              <Item isActive={active} hasEllipsis={menuItemsHaveEllipsis}>\n                                <Arrange gap=\"s\" tag=\"span\">\n                                  <Icon icon={<IconPlus />} color={active ? 'content' : 'icon'} />\n                                  Create {`\"` + query + `\"`}\n                                </Arrange>\n                              </Item>\n                            )}\n                          </Combobox.Option>\n                        </>\n                      )}\n                      {noResults && <EmptyState />}\n                    </Combobox.Options>\n                  </MenuCardTransition>\n                </OptionsRoot>\n              </Root>\n              {errorMessage && (\n                <Box marginTop=\"var(--grn-form-control-label-gap)\" color=\"danger\">\n                  {errorMessage}\n                </Box>\n              )}\n            </>\n          );\n        }}\n      </Combobox>\n    </>\n  );\n};\n\nAutocomplete2.Label = FieldLabel;\nAutocomplete2.Hint = FieldHint;\nAutocomplete2.Item = Item;\n\nAutocomplete2.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.string,\n  onChange: PropTypes.func.isRequired,\n  isCreatable: PropTypes.bool,\n  onCreate: PropTypes.func,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuMaxHeight: types.dimension,\n  menuZIndex: types.zIndex,\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n  placeholder: PropTypes.string,\n  hasError: PropTypes.bool,\n  errorMessage: PropTypes.string,\n  isDisabled: PropTypes.bool,\n  backgroundColor: types.color,\n  onClear: PropTypes.func,\n  hasSpinner: PropTypes.bool,\n  size: PropTypes.oneOf(['m', 'l']),\n};\n"]} */"));
75
75
 
76
76
  const EmptyState = () => ___EmotionJSX(Box, {
77
77
  paddingY: "s",
@@ -169,7 +169,7 @@ const Trigger = /*#__PURE__*/_styled("button", process.env.NODE_ENV === "product
169
169
  paddingRight
170
170
  } = _ref15;
171
171
  return paddingRight;
172
- }, ";}.autocompleteInput:focus-visible+&{opacity:0;pointer-events:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/autocomplete2.jsx"],"names":[],"mappings":"AAyM6B","file":"../../src/components/autocomplete2.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useState } from 'react';\nimport { Icon, Box, Arrange, Spinner } from '.';\nimport { IconChevronDown, IconPlus, IconCross } from '../icons';\nimport { Combobox } from '@headlessui/react';\nimport { defaultProps, types } from '../types';\nimport {\n  InputField,\n  MenuCard,\n  MenuCardTransition,\n  MenuGroupTitle,\n  FieldLabel,\n  FieldHint,\n  styles,\n  FieldClearButton,\n} from '../foundational';\nimport { MenuItemDivider, useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\nimport { Global } from '@emotion/react';\nimport { getColor, getIconSize, getRadius, getSpace, getTransition } from '../utilities';\nimport styled from '@emotion/styled';\n\nconst transitionSpeed = getTransition('xxFast');\n\nconst Root = styled.div`\n  ${styles.transitions};\n  ${({ hasError, isDisabled }) =>\n    !hasError &&\n    !isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color);\n      &:hover { --borderColor: var(--grn-field-border-color-hover) };\n      &:focus-within { --borderColor: var(--grn-field-border-color-focus) };\n    `};\n\n  ${({ hasError }) => hasError && `--borderColor: ${getColor('danger')};`};\n\n  ${({ backgroundColor, isDisabled }) =>\n    !backgroundColor &&\n    !isDisabled &&\n    `\n      --backgroundColor: transparent;\n    `};\n\n  ${({ backgroundColor }) =>\n    backgroundColor &&\n    `\n      --backgroundColor: ${getColor(backgroundColor)};\n    `};\n\n  ${({ isDisabled }) =>\n    isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color-disabled);\n      --backgroundColor: var(--grn-field-background-disabled);\n\n      * {\n        pointer-events: none;\n      }\n    `};\n\n  position: relative;\n  margin-top: ${({ marginTop }) => getSpace(marginTop)};\n  border-radius: ${getRadius('s')};\n  box-shadow: inset 0 0 0 var(--grn-border-size) var(--borderColor);\n  background-color: var(--backgroundColor);\n\n  .autocompleteInput {\n    line-height: 20px;\n    border: none;\n    opacity: ${({ open }) => (open ? 1 : 0)};\n    pointer-events: ${({ open }) => (open ? 'auto' : 'none')};\n    transition: opacity ${transitionSpeed};\n\n    &:focus-visible,\n    &:placeholder-shown {\n      opacity: 1;\n      pointer-events: auto;\n      transition: opacity ${transitionSpeed};\n    }\n    &:disabled:placeholder-shown {\n      background: transparent;\n    }\n  }\n`;\n\nconst EmptyState = () => (\n  <Box paddingY=\"s\" paddingX=\"m\" color=\"content2\">\n    No results\n  </Box>\n);\n\nconst rightSectionGap = 'var(--grn-space-m)';\n\nconst RightSection = ({ children, right }) => (\n  <Arrange\n    right={right}\n    gap={rightSectionGap}\n    position=\"absolute\"\n    top=\"0px\"\n    bottom=\"0px\"\n    margin=\"auto\"\n    height=\"fit-content\"\n  >\n    {children}\n  </Arrange>\n);\n\nconst ClearButton = ({ onClick, ...props }) => {\n  const handleClear = e => {\n    e.stopPropagation();\n    onClick();\n  };\n  return <FieldClearButton onClick={e => handleClear(e)} {...props} />;\n};\n\nconst getFilteredOptions = (query, options, searchField) => {\n  if (query === '') return options;\n\n  return options.filter(option => option[searchField].toLowerCase().includes(query.toLowerCase()));\n};\n\nconst getShowGroupTitle = (index, option, filteredOptions) => {\n  const firstOptionHasTitle = index === 0 && option.groupTitle;\n  const titleChanged = index > 0 && option.groupTitle !== filteredOptions[index - 1].groupTitle;\n\n  return firstOptionHasTitle || titleChanged;\n};\n\nconst menuItemStyles = `\n  .autocompleteMenuItem {\n    display: flex;\n    align-items: center;\n    list-style: none;\n    min-height: var(--grn-textbox-height-m);\n    border-radius: ${getRadius('s')};\n    appearance: none;\n    padding-block: 0;\n    padding: 4px 12px;\n\n    --checkUri: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%2337312f73\"%3E%3C/path%3E%3C/svg%3E');\n    --checkUriActive: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%23241a17\"%3E%3C/path%3E%3C/svg%3E');\n\n    &[data-headlessui-state='active'],\n    &[data-headlessui-state='active selected'] {\n      background-color: ${getColor('fade1')};\n    }\n\n    &[data-headlessui-state='selected'],\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUri);\n      background-repeat: no-repeat;\n      background-position: center right 12px;\n      background-size: auto ${getIconSize('m')};\n    }\n\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUriActive);\n    }\n\n    &[data-headlessui-state='disabled'] {\n      color: ${getColor('disabledContent')};\n      cursor: default;\n    }\n\n    ul & {\n      cursor: pointer;\n      padding-right: 40px;\n    }\n  }\n\n  .autocompleteMenuItemText {\n    flex-grow: 1;\n\n    &.hasEllipsis {\n      white-space: nowrap;\n      overflow: hidden;\n      text-overflow: ellipsis;\n    }\n  }\n`;\n\nconst Item = forwardRef(({ children, hasEllipsis, paddingLeft = '12px', ...props }, ref) => {\n  return (\n    <li\n      className=\"autocompleteMenuItem\"\n      ref={ref}\n      style={{ paddingLeft: getSpace(paddingLeft) }}\n      {...props}\n    >\n      <span\n        className={\n          hasEllipsis ? 'autocompleteMenuItemText hasEllipsis' : 'autocompleteMenuItemText'\n        }\n      >\n        {children}\n      </span>\n    </li>\n  );\n});\n\nconst Trigger = styled.button`\n  text-align: left;\n  height: ${({ size }) => `var(--grn-textbox-height-${size})`};\n  background-color: transparent;\n  cursor: text;\n  appearance: none;\n  display: block;\n  font: inherit;\n  color: inherit;\n  width: 100%;\n  outline: none;\n  border: none;\n  padding: 0;\n  opacity: ${({ open }) => (open ? 0 : 1)};\n  pointer-events: ${({ open }) => (open ? 'none' : 'auto')};\n  position: absolute;\n  top: 0;\n  left: 0;\n  transition: opacity ${transitionSpeed};\n\n  .autocompleteMenuItem {\n    padding-right: ${({ paddingRight }) => paddingRight};\n  }\n\n  .autocompleteInput:focus-visible + & {\n    opacity: 0;\n    pointer-events: none;\n  }\n`;\n\nconst TriggerText = styled.span`\n  line-height: 20px;\n  display: block;\n  padding-block: 0;\n  padding-right: ${({ paddingRight }) => paddingRight};\n  padding-left: ${({ paddingLeft }) => paddingLeft};\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`;\n\nexport const Autocomplete2 = ({\n  options,\n  value,\n  onChange,\n  isCreatable,\n  onCreate,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  menuZIndex,\n  placeholder,\n  label,\n  hint,\n  menuItemsHaveEllipsis = true,\n  hasPortal = true,\n  hasError,\n  errorMessage,\n  isDisabled,\n  searchField = 'content',\n  backgroundColor,\n  onClear,\n  hasSpinner,\n  size = 'm',\n  ...props\n}) => {\n  const [query, setQuery] = useState('');\n\n  const handleCreatableChange = option => {\n    if (onCreate) {\n      const isCreatableOption = !options.find(item => item.value === option.value);\n      if (isCreatableOption) {\n        onCreate(option);\n        return;\n      }\n    }\n    onChange(option);\n  };\n\n  const handleChange = option => {\n    if (isCreatable) {\n      handleCreatableChange(option);\n    } else {\n      onChange(option);\n    }\n  };\n\n  const filteredOptions = getFilteredOptions(query, options, searchField);\n\n  const isShowCreateOption = isCreatable && query.length > 0;\n\n  const noResults = !Boolean(filteredOptions.length) && !isShowCreateOption;\n\n  const fieldMarginTop = label || hint ? 'var(--grn-form-control-label-gap)' : undefined;\n\n  const selectedOption = useMemo(\n    () => options.find(option => option.value === value),\n    [options, value],\n  );\n\n  const newOption = useMemo(() => ({ value: query, content: query }), [query]);\n\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n\n  const OptionsRoot = hasPortal ? FloatingPortal : Fragment;\n\n  const hasClearButton = onClear && Boolean(selectedOption);\n\n  const inputPaddingX = `var(--grn-field-paddingX-${size})`;\n\n  const chevronWidth = `${IconChevronDown.viewBoxWidth / 2}px`;\n  const textGap = 'var(--grn-space-s2)';\n  const spinnerSpace = `calc(var(--grn-unit) * 3) + ${rightSectionGap}`;\n  const iconCrossWidth = `${IconCross.viewBoxWidth / 2}px`;\n  const clearButtonSpace = `${iconCrossWidth} + ${rightSectionGap}`;\n\n  const paddingRightValues = [textGap];\n  if (hasSpinner) paddingRightValues.push(spinnerSpace);\n  if (hasClearButton) paddingRightValues.push(clearButtonSpace);\n  paddingRightValues.push(chevronWidth, inputPaddingX);\n\n  const inputPaddingRight = `calc(${paddingRightValues.join(' + ')})`;\n\n  return (\n    <>\n      <Global styles={menuItemStyles} />\n      <Combobox\n        disabled={isDisabled}\n        as=\"div\"\n        value={selectedOption || ''}\n        onChange={handleChange}\n        {...props}\n      >\n        {({ open }) => {\n          const hasItem = selectedOption?.item;\n          return (\n            <>\n              {label && <Combobox.Label as={FieldLabel}>{label}</Combobox.Label>}\n              {hint && <FieldHint>{hint}</FieldHint>}\n\n              <Root\n                marginTop={fieldMarginTop}\n                ref={reference}\n                open={open}\n                hasError={hasError}\n                backgroundColor={backgroundColor}\n                isDisabled={isDisabled}\n              >\n                <Combobox.Button as={Box} position=\"relative\">\n                  <Combobox.Input\n                    as={InputField}\n                    autoComplete=\"off\"\n                    onChange={event => setQuery(event.target.value)}\n                    placeholder={placeholder}\n                    displayValue={option => option && option.content}\n                    paddingRight={inputPaddingRight}\n                    hasError={hasError}\n                    className=\"autocompleteInput\"\n                    size={size}\n                  />\n\n                  <Trigger open={open} tabIndex={-1} paddingRight={inputPaddingRight} size={size}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight} paddingLeft={inputPaddingX}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <RightSection right={inputPaddingX}>\n                    {hasSpinner && <Spinner />}\n                    {hasClearButton && <ClearButton onClick={onClear} trimSides />}\n                    <Icon icon={<IconChevronDown />} />\n                  </RightSection>\n                </Combobox.Button>\n\n                <OptionsRoot>\n                  <MenuCardTransition afterLeave={() => setQuery('')}>\n                    <Combobox.Options\n                      static\n                      ref={floating}\n                      placement={menuPlacement}\n                      maxHeight={menuMaxHeight}\n                      as={MenuCard}\n                      zIndex={menuZIndex}\n                      style={floatingStyles}\n                      isOpen={open}\n                    >\n                      {filteredOptions.map((option, index) => {\n                        const showGroupTitle = getShowGroupTitle(index, option, filteredOptions);\n\n                        return (\n                          <Fragment key={index}>\n                            {showGroupTitle && (\n                              <MenuGroupTitle hasDivider={index > 0}>\n                                {option.groupTitle}\n                              </MenuGroupTitle>\n                            )}\n\n                            <Combobox.Option\n                              value={option}\n                              as={Fragment}\n                              disabled={option.isDisabled}\n                            >\n                              {({ active }) => {\n                                const isDisabled = option.isDisabled;\n                                const isSelected = option.value === value;\n                                const isActive = !option.isDisabled && active;\n                                const hasEllipsis = menuItemsHaveEllipsis;\n                                if (option.item) {\n                                  return option.item({\n                                    isActive,\n                                    isSelected,\n                                    isDisabled,\n                                    hasEllipsis,\n                                  });\n                                }\n                                return <Item hasEllipsis={hasEllipsis}>{option.content}</Item>;\n                              }}\n                            </Combobox.Option>\n                          </Fragment>\n                        );\n                      })}\n                      {isShowCreateOption && (\n                        <>\n                          {filteredOptions.length > 0 && <MenuItemDivider />}\n                          <Combobox.Option value={newOption} as={Fragment}>\n                            {({ active }) => (\n                              <Item isActive={active} hasEllipsis={menuItemsHaveEllipsis}>\n                                <Arrange gap=\"s\" tag=\"span\">\n                                  <Icon icon={<IconPlus />} color={active ? 'content' : 'icon'} />\n                                  Create {`\"` + query + `\"`}\n                                </Arrange>\n                              </Item>\n                            )}\n                          </Combobox.Option>\n                        </>\n                      )}\n                      {noResults && <EmptyState />}\n                    </Combobox.Options>\n                  </MenuCardTransition>\n                </OptionsRoot>\n              </Root>\n              {errorMessage && (\n                <Box marginTop=\"var(--grn-form-control-label-gap)\" color=\"danger\">\n                  {errorMessage}\n                </Box>\n              )}\n            </>\n          );\n        }}\n      </Combobox>\n    </>\n  );\n};\n\nAutocomplete2.Label = FieldLabel;\nAutocomplete2.Hint = FieldHint;\nAutocomplete2.Item = Item;\n\nAutocomplete2.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.string,\n  onChange: PropTypes.func.isRequired,\n  isCreatable: PropTypes.bool,\n  onCreate: PropTypes.func,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuMaxHeight: types.dimension,\n  menuZIndex: types.zIndex,\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n  placeholder: PropTypes.string,\n  hasError: PropTypes.bool,\n  errorMessage: PropTypes.string,\n  isDisabled: PropTypes.bool,\n  backgroundColor: types.color,\n  onClear: PropTypes.func,\n  hasSpinner: PropTypes.bool,\n  size: PropTypes.oneOf(['m', 'l']),\n};\n"]} */"));
172
+ }, ";}.autocompleteInput:focus-visible+&{opacity:0;pointer-events:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/autocomplete2.jsx"],"names":[],"mappings":"AAyM6B","file":"../../src/components/autocomplete2.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useState } from 'react';\nimport { Icon, Box, Arrange, Spinner } from '.';\nimport { IconChevronDown, IconPlus, IconCross } from '../icons';\nimport { Combobox } from '@headlessui/react';\nimport { defaultProps, types } from '../types';\nimport {\n  InputField,\n  MenuCard,\n  MenuCardTransition,\n  MenuGroupTitle,\n  FieldLabel,\n  FieldHint,\n  styles,\n  FieldClearButton,\n} from '../foundational';\nimport { MenuItemDivider, useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\nimport { Global } from '@emotion/react';\nimport { getColor, getIconSize, getRadius, getSpace, getTransition } from '../utilities';\nimport styled from '@emotion/styled';\n\nconst transitionSpeed = getTransition('xxFast');\n\nconst Root = styled.div`\n  ${styles.transitions};\n  ${({ hasError, isDisabled }) =>\n    !hasError &&\n    !isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color);\n      &:hover { --borderColor: var(--grn-field-border-color-hover) };\n      &:focus-within { --borderColor: var(--grn-field-border-color-focus) };\n    `};\n\n  ${({ hasError }) => hasError && `--borderColor: ${getColor('danger')};`};\n\n  ${({ backgroundColor, isDisabled }) =>\n    !backgroundColor &&\n    !isDisabled &&\n    `\n      --backgroundColor: transparent;\n    `};\n\n  ${({ backgroundColor }) =>\n    backgroundColor &&\n    `\n      --backgroundColor: ${getColor(backgroundColor)};\n    `};\n\n  ${({ isDisabled }) =>\n    isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color-disabled);\n      --backgroundColor: var(--grn-field-background-disabled);\n\n      * {\n        pointer-events: none;\n      }\n    `};\n\n  position: relative;\n  margin-top: ${({ marginTop }) => getSpace(marginTop)};\n  border-radius: ${getRadius('s')};\n  box-shadow: inset 0 0 0 var(--grn-border-size) var(--borderColor);\n  background-color: var(--backgroundColor);\n\n  .autocompleteInput {\n    line-height: 20px;\n    border: none;\n    opacity: ${({ open }) => (open ? 1 : 0)};\n    pointer-events: ${({ open }) => (open ? 'auto' : 'none')};\n    transition: opacity ${transitionSpeed};\n\n    &:focus-visible,\n    &:placeholder-shown {\n      opacity: 1;\n      pointer-events: auto;\n      transition: opacity ${transitionSpeed};\n    }\n    &:disabled:placeholder-shown {\n      background: transparent;\n    }\n  }\n`;\n\nconst EmptyState = () => (\n  <Box paddingY=\"s\" paddingX=\"m\" color=\"content2\">\n    No results\n  </Box>\n);\n\nconst rightSectionGap = 'var(--grn-space-m)';\n\nconst RightSection = ({ children, right }) => (\n  <Arrange\n    right={right}\n    gap={rightSectionGap}\n    position=\"absolute\"\n    top=\"0px\"\n    bottom=\"0px\"\n    margin=\"auto\"\n    height=\"fit-content\"\n  >\n    {children}\n  </Arrange>\n);\n\nconst ClearButton = ({ onClick, ...props }) => {\n  const handleClear = e => {\n    e.stopPropagation();\n    onClick();\n  };\n  return <FieldClearButton onClick={e => handleClear(e)} {...props} />;\n};\n\nconst getFilteredOptions = (query, options, searchField) => {\n  if (query === '') return options;\n\n  return options.filter(option => option[searchField].toLowerCase().includes(query.toLowerCase()));\n};\n\nconst getShowGroupTitle = (index, option, filteredOptions) => {\n  const firstOptionHasTitle = index === 0 && option.groupTitle;\n  const titleChanged = index > 0 && option.groupTitle !== filteredOptions[index - 1].groupTitle;\n\n  return firstOptionHasTitle || titleChanged;\n};\n\nconst menuItemStyles = `\n  .autocompleteMenuItem {\n    display: flex;\n    align-items: center;\n    list-style: none;\n    min-height: var(--grn-textbox-height-m);\n    border-radius: ${getRadius('s')};\n    appearance: none;\n    padding-block: 0;\n    padding: 4px 12px;\n\n    --checkUri: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%2337312f73\"%3E%3C/path%3E%3C/svg%3E');\n    --checkUriActive: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%23241a17\"%3E%3C/path%3E%3C/svg%3E');\n\n    &[data-headlessui-state='active'],\n    &[data-headlessui-state='active selected'] {\n      background-color: ${getColor('fade1')};\n    }\n\n    &[data-headlessui-state='selected'],\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUri);\n      background-repeat: no-repeat;\n      background-position: center right 12px;\n      background-size: auto ${getIconSize('m')};\n    }\n\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUriActive);\n    }\n\n    &[data-headlessui-state='disabled'] {\n      color: ${getColor('disabledContent')};\n      cursor: default;\n    }\n\n    ul & {\n      cursor: pointer;\n      padding-right: 40px;\n    }\n  }\n\n  .autocompleteMenuItemText {\n    flex-grow: 1;\n\n    &.hasEllipsis {\n      white-space: nowrap;\n      overflow: hidden;\n      text-overflow: ellipsis;\n    }\n  }\n`;\n\nconst Item = forwardRef(({ children, hasEllipsis, paddingLeft = '12px', ...props }, ref) => {\n  return (\n    <li\n      className=\"autocompleteMenuItem\"\n      ref={ref}\n      style={{ paddingLeft: getSpace(paddingLeft) }}\n      {...props}\n    >\n      <span\n        className={\n          hasEllipsis ? 'autocompleteMenuItemText hasEllipsis' : 'autocompleteMenuItemText'\n        }\n      >\n        {children}\n      </span>\n    </li>\n  );\n});\n\nconst Trigger = styled.button`\n  text-align: left;\n  height: ${({ size }) => `var(--grn-textbox-height-${size})`};\n  background-color: transparent;\n  cursor: text;\n  appearance: none;\n  display: block;\n  font: inherit;\n  color: inherit;\n  width: 100%;\n  outline: none;\n  border: none;\n  padding: 0;\n  opacity: ${({ open }) => (open ? 0 : 1)};\n  pointer-events: ${({ open }) => (open ? 'none' : 'auto')};\n  position: absolute;\n  top: 0;\n  left: 0;\n  transition: opacity ${transitionSpeed};\n\n  .autocompleteMenuItem {\n    padding-right: ${({ paddingRight }) => paddingRight};\n  }\n\n  .autocompleteInput:focus-visible + & {\n    opacity: 0;\n    pointer-events: none;\n  }\n`;\n\nconst TriggerText = styled.span`\n  line-height: 20px;\n  display: block;\n  padding-block: 0;\n  padding-right: ${({ paddingRight }) => paddingRight};\n  padding-left: ${({ paddingLeft }) => paddingLeft};\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`;\n\nexport const Autocomplete2 = ({\n  options,\n  value,\n  onChange,\n  isCreatable,\n  onCreate,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  menuZIndex,\n  placeholder,\n  label,\n  hint,\n  menuItemsHaveEllipsis = true,\n  hasPortal = true,\n  hasError,\n  errorMessage,\n  isDisabled,\n  searchField = 'content',\n  backgroundColor,\n  onClear,\n  hasSpinner,\n  size = 'm',\n  renderGroupTitle,\n  ...props\n}) => {\n  const [query, setQuery] = useState('');\n\n  const handleCreatableChange = option => {\n    if (onCreate) {\n      const isCreatableOption = !options.find(item => item.value === option.value);\n      if (isCreatableOption) {\n        onCreate(option);\n        return;\n      }\n    }\n    onChange(option);\n  };\n\n  const handleChange = option => {\n    if (isCreatable) {\n      handleCreatableChange(option);\n    } else {\n      onChange(option);\n    }\n  };\n\n  const filteredOptions = getFilteredOptions(query, options, searchField);\n\n  const isShowCreateOption = isCreatable && query.length > 0;\n\n  const noResults = !Boolean(filteredOptions.length) && !isShowCreateOption;\n\n  const fieldMarginTop = label || hint ? 'var(--grn-form-control-label-gap)' : undefined;\n\n  const selectedOption = useMemo(\n    () => options.find(option => option.value === value),\n    [options, value],\n  );\n\n  const newOption = useMemo(() => ({ value: query, content: query }), [query]);\n\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n\n  const OptionsRoot = hasPortal ? FloatingPortal : Fragment;\n\n  const hasClearButton = onClear && Boolean(selectedOption);\n\n  const inputPaddingX = `var(--grn-field-paddingX-${size})`;\n\n  const chevronWidth = `${IconChevronDown.viewBoxWidth / 2}px`;\n  const textGap = 'var(--grn-space-s2)';\n  const spinnerSpace = `calc(var(--grn-unit) * 3) + ${rightSectionGap}`;\n  const iconCrossWidth = `${IconCross.viewBoxWidth / 2}px`;\n  const clearButtonSpace = `${iconCrossWidth} + ${rightSectionGap}`;\n\n  const paddingRightValues = [textGap];\n  if (hasSpinner) paddingRightValues.push(spinnerSpace);\n  if (hasClearButton) paddingRightValues.push(clearButtonSpace);\n  paddingRightValues.push(chevronWidth, inputPaddingX);\n\n  const inputPaddingRight = `calc(${paddingRightValues.join(' + ')})`;\n\n  return (\n    <>\n      <Global styles={menuItemStyles} />\n      <Combobox\n        disabled={isDisabled}\n        as=\"div\"\n        value={selectedOption || ''}\n        onChange={handleChange}\n        {...props}\n      >\n        {({ open }) => {\n          const hasItem = selectedOption?.item;\n          return (\n            <>\n              {label && <Combobox.Label as={FieldLabel}>{label}</Combobox.Label>}\n              {hint && <FieldHint>{hint}</FieldHint>}\n\n              <Root\n                marginTop={fieldMarginTop}\n                ref={reference}\n                open={open}\n                hasError={hasError}\n                backgroundColor={backgroundColor}\n                isDisabled={isDisabled}\n              >\n                <Combobox.Button as={Box} position=\"relative\">\n                  <Combobox.Input\n                    as={InputField}\n                    autoComplete=\"off\"\n                    onChange={event => setQuery(event.target.value)}\n                    placeholder={placeholder}\n                    displayValue={option => option && option.content}\n                    paddingRight={inputPaddingRight}\n                    hasError={hasError}\n                    className=\"autocompleteInput\"\n                    size={size}\n                  />\n\n                  <Trigger open={open} tabIndex={-1} paddingRight={inputPaddingRight} size={size}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight} paddingLeft={inputPaddingX}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <RightSection right={inputPaddingX}>\n                    {hasSpinner && <Spinner />}\n                    {hasClearButton && <ClearButton onClick={onClear} trimSides />}\n                    <Icon icon={<IconChevronDown />} />\n                  </RightSection>\n                </Combobox.Button>\n\n                <OptionsRoot>\n                  <MenuCardTransition afterLeave={() => setQuery('')}>\n                    <Combobox.Options\n                      static\n                      ref={floating}\n                      placement={menuPlacement}\n                      maxHeight={menuMaxHeight}\n                      as={MenuCard}\n                      zIndex={menuZIndex}\n                      style={floatingStyles}\n                      isOpen={open}\n                    >\n                      {filteredOptions.map((option, index) => {\n                        const showGroupTitle = getShowGroupTitle(index, option, filteredOptions);\n\n                        return (\n                          <Fragment key={index}>\n                            {showGroupTitle &&\n                              (renderGroupTitle ? (\n                                renderGroupTitle(option.groupTitle)\n                              ) : (\n                                <MenuGroupTitle>{option.groupTitle}</MenuGroupTitle>\n                              ))}\n\n                            <Combobox.Option\n                              value={option}\n                              as={Fragment}\n                              disabled={option.isDisabled}\n                            >\n                              {({ active }) => {\n                                const isDisabled = option.isDisabled;\n                                const isSelected = option.value === value;\n                                const isActive = !option.isDisabled && active;\n                                const hasEllipsis = menuItemsHaveEllipsis;\n                                if (option.item) {\n                                  return option.item({\n                                    isActive,\n                                    isSelected,\n                                    isDisabled,\n                                    hasEllipsis,\n                                  });\n                                }\n                                return <Item hasEllipsis={hasEllipsis}>{option.content}</Item>;\n                              }}\n                            </Combobox.Option>\n                          </Fragment>\n                        );\n                      })}\n                      {isShowCreateOption && (\n                        <>\n                          {filteredOptions.length > 0 && <MenuItemDivider />}\n                          <Combobox.Option value={newOption} as={Fragment}>\n                            {({ active }) => (\n                              <Item isActive={active} hasEllipsis={menuItemsHaveEllipsis}>\n                                <Arrange gap=\"s\" tag=\"span\">\n                                  <Icon icon={<IconPlus />} color={active ? 'content' : 'icon'} />\n                                  Create {`\"` + query + `\"`}\n                                </Arrange>\n                              </Item>\n                            )}\n                          </Combobox.Option>\n                        </>\n                      )}\n                      {noResults && <EmptyState />}\n                    </Combobox.Options>\n                  </MenuCardTransition>\n                </OptionsRoot>\n              </Root>\n              {errorMessage && (\n                <Box marginTop=\"var(--grn-form-control-label-gap)\" color=\"danger\">\n                  {errorMessage}\n                </Box>\n              )}\n            </>\n          );\n        }}\n      </Combobox>\n    </>\n  );\n};\n\nAutocomplete2.Label = FieldLabel;\nAutocomplete2.Hint = FieldHint;\nAutocomplete2.Item = Item;\n\nAutocomplete2.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.string,\n  onChange: PropTypes.func.isRequired,\n  isCreatable: PropTypes.bool,\n  onCreate: PropTypes.func,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuMaxHeight: types.dimension,\n  menuZIndex: types.zIndex,\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n  placeholder: PropTypes.string,\n  hasError: PropTypes.bool,\n  errorMessage: PropTypes.string,\n  isDisabled: PropTypes.bool,\n  backgroundColor: types.color,\n  onClear: PropTypes.func,\n  hasSpinner: PropTypes.bool,\n  size: PropTypes.oneOf(['m', 'l']),\n};\n"]} */"));
173
173
 
174
174
  const TriggerText = /*#__PURE__*/_styled("span", process.env.NODE_ENV === "production" ? {
175
175
  target: "ea1qwwn0"
@@ -186,7 +186,7 @@ const TriggerText = /*#__PURE__*/_styled("span", process.env.NODE_ENV === "produ
186
186
  paddingLeft
187
187
  } = _ref17;
188
188
  return paddingLeft;
189
- }, ";text-overflow:ellipsis;overflow:hidden;white-space:nowrap;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/autocomplete2.jsx"],"names":[],"mappings":"AAuO+B","file":"../../src/components/autocomplete2.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useState } from 'react';\nimport { Icon, Box, Arrange, Spinner } from '.';\nimport { IconChevronDown, IconPlus, IconCross } from '../icons';\nimport { Combobox } from '@headlessui/react';\nimport { defaultProps, types } from '../types';\nimport {\n  InputField,\n  MenuCard,\n  MenuCardTransition,\n  MenuGroupTitle,\n  FieldLabel,\n  FieldHint,\n  styles,\n  FieldClearButton,\n} from '../foundational';\nimport { MenuItemDivider, useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\nimport { Global } from '@emotion/react';\nimport { getColor, getIconSize, getRadius, getSpace, getTransition } from '../utilities';\nimport styled from '@emotion/styled';\n\nconst transitionSpeed = getTransition('xxFast');\n\nconst Root = styled.div`\n  ${styles.transitions};\n  ${({ hasError, isDisabled }) =>\n    !hasError &&\n    !isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color);\n      &:hover { --borderColor: var(--grn-field-border-color-hover) };\n      &:focus-within { --borderColor: var(--grn-field-border-color-focus) };\n    `};\n\n  ${({ hasError }) => hasError && `--borderColor: ${getColor('danger')};`};\n\n  ${({ backgroundColor, isDisabled }) =>\n    !backgroundColor &&\n    !isDisabled &&\n    `\n      --backgroundColor: transparent;\n    `};\n\n  ${({ backgroundColor }) =>\n    backgroundColor &&\n    `\n      --backgroundColor: ${getColor(backgroundColor)};\n    `};\n\n  ${({ isDisabled }) =>\n    isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color-disabled);\n      --backgroundColor: var(--grn-field-background-disabled);\n\n      * {\n        pointer-events: none;\n      }\n    `};\n\n  position: relative;\n  margin-top: ${({ marginTop }) => getSpace(marginTop)};\n  border-radius: ${getRadius('s')};\n  box-shadow: inset 0 0 0 var(--grn-border-size) var(--borderColor);\n  background-color: var(--backgroundColor);\n\n  .autocompleteInput {\n    line-height: 20px;\n    border: none;\n    opacity: ${({ open }) => (open ? 1 : 0)};\n    pointer-events: ${({ open }) => (open ? 'auto' : 'none')};\n    transition: opacity ${transitionSpeed};\n\n    &:focus-visible,\n    &:placeholder-shown {\n      opacity: 1;\n      pointer-events: auto;\n      transition: opacity ${transitionSpeed};\n    }\n    &:disabled:placeholder-shown {\n      background: transparent;\n    }\n  }\n`;\n\nconst EmptyState = () => (\n  <Box paddingY=\"s\" paddingX=\"m\" color=\"content2\">\n    No results\n  </Box>\n);\n\nconst rightSectionGap = 'var(--grn-space-m)';\n\nconst RightSection = ({ children, right }) => (\n  <Arrange\n    right={right}\n    gap={rightSectionGap}\n    position=\"absolute\"\n    top=\"0px\"\n    bottom=\"0px\"\n    margin=\"auto\"\n    height=\"fit-content\"\n  >\n    {children}\n  </Arrange>\n);\n\nconst ClearButton = ({ onClick, ...props }) => {\n  const handleClear = e => {\n    e.stopPropagation();\n    onClick();\n  };\n  return <FieldClearButton onClick={e => handleClear(e)} {...props} />;\n};\n\nconst getFilteredOptions = (query, options, searchField) => {\n  if (query === '') return options;\n\n  return options.filter(option => option[searchField].toLowerCase().includes(query.toLowerCase()));\n};\n\nconst getShowGroupTitle = (index, option, filteredOptions) => {\n  const firstOptionHasTitle = index === 0 && option.groupTitle;\n  const titleChanged = index > 0 && option.groupTitle !== filteredOptions[index - 1].groupTitle;\n\n  return firstOptionHasTitle || titleChanged;\n};\n\nconst menuItemStyles = `\n  .autocompleteMenuItem {\n    display: flex;\n    align-items: center;\n    list-style: none;\n    min-height: var(--grn-textbox-height-m);\n    border-radius: ${getRadius('s')};\n    appearance: none;\n    padding-block: 0;\n    padding: 4px 12px;\n\n    --checkUri: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%2337312f73\"%3E%3C/path%3E%3C/svg%3E');\n    --checkUriActive: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%23241a17\"%3E%3C/path%3E%3C/svg%3E');\n\n    &[data-headlessui-state='active'],\n    &[data-headlessui-state='active selected'] {\n      background-color: ${getColor('fade1')};\n    }\n\n    &[data-headlessui-state='selected'],\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUri);\n      background-repeat: no-repeat;\n      background-position: center right 12px;\n      background-size: auto ${getIconSize('m')};\n    }\n\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUriActive);\n    }\n\n    &[data-headlessui-state='disabled'] {\n      color: ${getColor('disabledContent')};\n      cursor: default;\n    }\n\n    ul & {\n      cursor: pointer;\n      padding-right: 40px;\n    }\n  }\n\n  .autocompleteMenuItemText {\n    flex-grow: 1;\n\n    &.hasEllipsis {\n      white-space: nowrap;\n      overflow: hidden;\n      text-overflow: ellipsis;\n    }\n  }\n`;\n\nconst Item = forwardRef(({ children, hasEllipsis, paddingLeft = '12px', ...props }, ref) => {\n  return (\n    <li\n      className=\"autocompleteMenuItem\"\n      ref={ref}\n      style={{ paddingLeft: getSpace(paddingLeft) }}\n      {...props}\n    >\n      <span\n        className={\n          hasEllipsis ? 'autocompleteMenuItemText hasEllipsis' : 'autocompleteMenuItemText'\n        }\n      >\n        {children}\n      </span>\n    </li>\n  );\n});\n\nconst Trigger = styled.button`\n  text-align: left;\n  height: ${({ size }) => `var(--grn-textbox-height-${size})`};\n  background-color: transparent;\n  cursor: text;\n  appearance: none;\n  display: block;\n  font: inherit;\n  color: inherit;\n  width: 100%;\n  outline: none;\n  border: none;\n  padding: 0;\n  opacity: ${({ open }) => (open ? 0 : 1)};\n  pointer-events: ${({ open }) => (open ? 'none' : 'auto')};\n  position: absolute;\n  top: 0;\n  left: 0;\n  transition: opacity ${transitionSpeed};\n\n  .autocompleteMenuItem {\n    padding-right: ${({ paddingRight }) => paddingRight};\n  }\n\n  .autocompleteInput:focus-visible + & {\n    opacity: 0;\n    pointer-events: none;\n  }\n`;\n\nconst TriggerText = styled.span`\n  line-height: 20px;\n  display: block;\n  padding-block: 0;\n  padding-right: ${({ paddingRight }) => paddingRight};\n  padding-left: ${({ paddingLeft }) => paddingLeft};\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`;\n\nexport const Autocomplete2 = ({\n  options,\n  value,\n  onChange,\n  isCreatable,\n  onCreate,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  menuZIndex,\n  placeholder,\n  label,\n  hint,\n  menuItemsHaveEllipsis = true,\n  hasPortal = true,\n  hasError,\n  errorMessage,\n  isDisabled,\n  searchField = 'content',\n  backgroundColor,\n  onClear,\n  hasSpinner,\n  size = 'm',\n  ...props\n}) => {\n  const [query, setQuery] = useState('');\n\n  const handleCreatableChange = option => {\n    if (onCreate) {\n      const isCreatableOption = !options.find(item => item.value === option.value);\n      if (isCreatableOption) {\n        onCreate(option);\n        return;\n      }\n    }\n    onChange(option);\n  };\n\n  const handleChange = option => {\n    if (isCreatable) {\n      handleCreatableChange(option);\n    } else {\n      onChange(option);\n    }\n  };\n\n  const filteredOptions = getFilteredOptions(query, options, searchField);\n\n  const isShowCreateOption = isCreatable && query.length > 0;\n\n  const noResults = !Boolean(filteredOptions.length) && !isShowCreateOption;\n\n  const fieldMarginTop = label || hint ? 'var(--grn-form-control-label-gap)' : undefined;\n\n  const selectedOption = useMemo(\n    () => options.find(option => option.value === value),\n    [options, value],\n  );\n\n  const newOption = useMemo(() => ({ value: query, content: query }), [query]);\n\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n\n  const OptionsRoot = hasPortal ? FloatingPortal : Fragment;\n\n  const hasClearButton = onClear && Boolean(selectedOption);\n\n  const inputPaddingX = `var(--grn-field-paddingX-${size})`;\n\n  const chevronWidth = `${IconChevronDown.viewBoxWidth / 2}px`;\n  const textGap = 'var(--grn-space-s2)';\n  const spinnerSpace = `calc(var(--grn-unit) * 3) + ${rightSectionGap}`;\n  const iconCrossWidth = `${IconCross.viewBoxWidth / 2}px`;\n  const clearButtonSpace = `${iconCrossWidth} + ${rightSectionGap}`;\n\n  const paddingRightValues = [textGap];\n  if (hasSpinner) paddingRightValues.push(spinnerSpace);\n  if (hasClearButton) paddingRightValues.push(clearButtonSpace);\n  paddingRightValues.push(chevronWidth, inputPaddingX);\n\n  const inputPaddingRight = `calc(${paddingRightValues.join(' + ')})`;\n\n  return (\n    <>\n      <Global styles={menuItemStyles} />\n      <Combobox\n        disabled={isDisabled}\n        as=\"div\"\n        value={selectedOption || ''}\n        onChange={handleChange}\n        {...props}\n      >\n        {({ open }) => {\n          const hasItem = selectedOption?.item;\n          return (\n            <>\n              {label && <Combobox.Label as={FieldLabel}>{label}</Combobox.Label>}\n              {hint && <FieldHint>{hint}</FieldHint>}\n\n              <Root\n                marginTop={fieldMarginTop}\n                ref={reference}\n                open={open}\n                hasError={hasError}\n                backgroundColor={backgroundColor}\n                isDisabled={isDisabled}\n              >\n                <Combobox.Button as={Box} position=\"relative\">\n                  <Combobox.Input\n                    as={InputField}\n                    autoComplete=\"off\"\n                    onChange={event => setQuery(event.target.value)}\n                    placeholder={placeholder}\n                    displayValue={option => option && option.content}\n                    paddingRight={inputPaddingRight}\n                    hasError={hasError}\n                    className=\"autocompleteInput\"\n                    size={size}\n                  />\n\n                  <Trigger open={open} tabIndex={-1} paddingRight={inputPaddingRight} size={size}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight} paddingLeft={inputPaddingX}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <RightSection right={inputPaddingX}>\n                    {hasSpinner && <Spinner />}\n                    {hasClearButton && <ClearButton onClick={onClear} trimSides />}\n                    <Icon icon={<IconChevronDown />} />\n                  </RightSection>\n                </Combobox.Button>\n\n                <OptionsRoot>\n                  <MenuCardTransition afterLeave={() => setQuery('')}>\n                    <Combobox.Options\n                      static\n                      ref={floating}\n                      placement={menuPlacement}\n                      maxHeight={menuMaxHeight}\n                      as={MenuCard}\n                      zIndex={menuZIndex}\n                      style={floatingStyles}\n                      isOpen={open}\n                    >\n                      {filteredOptions.map((option, index) => {\n                        const showGroupTitle = getShowGroupTitle(index, option, filteredOptions);\n\n                        return (\n                          <Fragment key={index}>\n                            {showGroupTitle && (\n                              <MenuGroupTitle hasDivider={index > 0}>\n                                {option.groupTitle}\n                              </MenuGroupTitle>\n                            )}\n\n                            <Combobox.Option\n                              value={option}\n                              as={Fragment}\n                              disabled={option.isDisabled}\n                            >\n                              {({ active }) => {\n                                const isDisabled = option.isDisabled;\n                                const isSelected = option.value === value;\n                                const isActive = !option.isDisabled && active;\n                                const hasEllipsis = menuItemsHaveEllipsis;\n                                if (option.item) {\n                                  return option.item({\n                                    isActive,\n                                    isSelected,\n                                    isDisabled,\n                                    hasEllipsis,\n                                  });\n                                }\n                                return <Item hasEllipsis={hasEllipsis}>{option.content}</Item>;\n                              }}\n                            </Combobox.Option>\n                          </Fragment>\n                        );\n                      })}\n                      {isShowCreateOption && (\n                        <>\n                          {filteredOptions.length > 0 && <MenuItemDivider />}\n                          <Combobox.Option value={newOption} as={Fragment}>\n                            {({ active }) => (\n                              <Item isActive={active} hasEllipsis={menuItemsHaveEllipsis}>\n                                <Arrange gap=\"s\" tag=\"span\">\n                                  <Icon icon={<IconPlus />} color={active ? 'content' : 'icon'} />\n                                  Create {`\"` + query + `\"`}\n                                </Arrange>\n                              </Item>\n                            )}\n                          </Combobox.Option>\n                        </>\n                      )}\n                      {noResults && <EmptyState />}\n                    </Combobox.Options>\n                  </MenuCardTransition>\n                </OptionsRoot>\n              </Root>\n              {errorMessage && (\n                <Box marginTop=\"var(--grn-form-control-label-gap)\" color=\"danger\">\n                  {errorMessage}\n                </Box>\n              )}\n            </>\n          );\n        }}\n      </Combobox>\n    </>\n  );\n};\n\nAutocomplete2.Label = FieldLabel;\nAutocomplete2.Hint = FieldHint;\nAutocomplete2.Item = Item;\n\nAutocomplete2.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.string,\n  onChange: PropTypes.func.isRequired,\n  isCreatable: PropTypes.bool,\n  onCreate: PropTypes.func,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuMaxHeight: types.dimension,\n  menuZIndex: types.zIndex,\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n  placeholder: PropTypes.string,\n  hasError: PropTypes.bool,\n  errorMessage: PropTypes.string,\n  isDisabled: PropTypes.bool,\n  backgroundColor: types.color,\n  onClear: PropTypes.func,\n  hasSpinner: PropTypes.bool,\n  size: PropTypes.oneOf(['m', 'l']),\n};\n"]} */"));
189
+ }, ";text-overflow:ellipsis;overflow:hidden;white-space:nowrap;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/autocomplete2.jsx"],"names":[],"mappings":"AAuO+B","file":"../../src/components/autocomplete2.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useState } from 'react';\nimport { Icon, Box, Arrange, Spinner } from '.';\nimport { IconChevronDown, IconPlus, IconCross } from '../icons';\nimport { Combobox } from '@headlessui/react';\nimport { defaultProps, types } from '../types';\nimport {\n  InputField,\n  MenuCard,\n  MenuCardTransition,\n  MenuGroupTitle,\n  FieldLabel,\n  FieldHint,\n  styles,\n  FieldClearButton,\n} from '../foundational';\nimport { MenuItemDivider, useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\nimport { Global } from '@emotion/react';\nimport { getColor, getIconSize, getRadius, getSpace, getTransition } from '../utilities';\nimport styled from '@emotion/styled';\n\nconst transitionSpeed = getTransition('xxFast');\n\nconst Root = styled.div`\n  ${styles.transitions};\n  ${({ hasError, isDisabled }) =>\n    !hasError &&\n    !isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color);\n      &:hover { --borderColor: var(--grn-field-border-color-hover) };\n      &:focus-within { --borderColor: var(--grn-field-border-color-focus) };\n    `};\n\n  ${({ hasError }) => hasError && `--borderColor: ${getColor('danger')};`};\n\n  ${({ backgroundColor, isDisabled }) =>\n    !backgroundColor &&\n    !isDisabled &&\n    `\n      --backgroundColor: transparent;\n    `};\n\n  ${({ backgroundColor }) =>\n    backgroundColor &&\n    `\n      --backgroundColor: ${getColor(backgroundColor)};\n    `};\n\n  ${({ isDisabled }) =>\n    isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color-disabled);\n      --backgroundColor: var(--grn-field-background-disabled);\n\n      * {\n        pointer-events: none;\n      }\n    `};\n\n  position: relative;\n  margin-top: ${({ marginTop }) => getSpace(marginTop)};\n  border-radius: ${getRadius('s')};\n  box-shadow: inset 0 0 0 var(--grn-border-size) var(--borderColor);\n  background-color: var(--backgroundColor);\n\n  .autocompleteInput {\n    line-height: 20px;\n    border: none;\n    opacity: ${({ open }) => (open ? 1 : 0)};\n    pointer-events: ${({ open }) => (open ? 'auto' : 'none')};\n    transition: opacity ${transitionSpeed};\n\n    &:focus-visible,\n    &:placeholder-shown {\n      opacity: 1;\n      pointer-events: auto;\n      transition: opacity ${transitionSpeed};\n    }\n    &:disabled:placeholder-shown {\n      background: transparent;\n    }\n  }\n`;\n\nconst EmptyState = () => (\n  <Box paddingY=\"s\" paddingX=\"m\" color=\"content2\">\n    No results\n  </Box>\n);\n\nconst rightSectionGap = 'var(--grn-space-m)';\n\nconst RightSection = ({ children, right }) => (\n  <Arrange\n    right={right}\n    gap={rightSectionGap}\n    position=\"absolute\"\n    top=\"0px\"\n    bottom=\"0px\"\n    margin=\"auto\"\n    height=\"fit-content\"\n  >\n    {children}\n  </Arrange>\n);\n\nconst ClearButton = ({ onClick, ...props }) => {\n  const handleClear = e => {\n    e.stopPropagation();\n    onClick();\n  };\n  return <FieldClearButton onClick={e => handleClear(e)} {...props} />;\n};\n\nconst getFilteredOptions = (query, options, searchField) => {\n  if (query === '') return options;\n\n  return options.filter(option => option[searchField].toLowerCase().includes(query.toLowerCase()));\n};\n\nconst getShowGroupTitle = (index, option, filteredOptions) => {\n  const firstOptionHasTitle = index === 0 && option.groupTitle;\n  const titleChanged = index > 0 && option.groupTitle !== filteredOptions[index - 1].groupTitle;\n\n  return firstOptionHasTitle || titleChanged;\n};\n\nconst menuItemStyles = `\n  .autocompleteMenuItem {\n    display: flex;\n    align-items: center;\n    list-style: none;\n    min-height: var(--grn-textbox-height-m);\n    border-radius: ${getRadius('s')};\n    appearance: none;\n    padding-block: 0;\n    padding: 4px 12px;\n\n    --checkUri: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%2337312f73\"%3E%3C/path%3E%3C/svg%3E');\n    --checkUriActive: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%23241a17\"%3E%3C/path%3E%3C/svg%3E');\n\n    &[data-headlessui-state='active'],\n    &[data-headlessui-state='active selected'] {\n      background-color: ${getColor('fade1')};\n    }\n\n    &[data-headlessui-state='selected'],\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUri);\n      background-repeat: no-repeat;\n      background-position: center right 12px;\n      background-size: auto ${getIconSize('m')};\n    }\n\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUriActive);\n    }\n\n    &[data-headlessui-state='disabled'] {\n      color: ${getColor('disabledContent')};\n      cursor: default;\n    }\n\n    ul & {\n      cursor: pointer;\n      padding-right: 40px;\n    }\n  }\n\n  .autocompleteMenuItemText {\n    flex-grow: 1;\n\n    &.hasEllipsis {\n      white-space: nowrap;\n      overflow: hidden;\n      text-overflow: ellipsis;\n    }\n  }\n`;\n\nconst Item = forwardRef(({ children, hasEllipsis, paddingLeft = '12px', ...props }, ref) => {\n  return (\n    <li\n      className=\"autocompleteMenuItem\"\n      ref={ref}\n      style={{ paddingLeft: getSpace(paddingLeft) }}\n      {...props}\n    >\n      <span\n        className={\n          hasEllipsis ? 'autocompleteMenuItemText hasEllipsis' : 'autocompleteMenuItemText'\n        }\n      >\n        {children}\n      </span>\n    </li>\n  );\n});\n\nconst Trigger = styled.button`\n  text-align: left;\n  height: ${({ size }) => `var(--grn-textbox-height-${size})`};\n  background-color: transparent;\n  cursor: text;\n  appearance: none;\n  display: block;\n  font: inherit;\n  color: inherit;\n  width: 100%;\n  outline: none;\n  border: none;\n  padding: 0;\n  opacity: ${({ open }) => (open ? 0 : 1)};\n  pointer-events: ${({ open }) => (open ? 'none' : 'auto')};\n  position: absolute;\n  top: 0;\n  left: 0;\n  transition: opacity ${transitionSpeed};\n\n  .autocompleteMenuItem {\n    padding-right: ${({ paddingRight }) => paddingRight};\n  }\n\n  .autocompleteInput:focus-visible + & {\n    opacity: 0;\n    pointer-events: none;\n  }\n`;\n\nconst TriggerText = styled.span`\n  line-height: 20px;\n  display: block;\n  padding-block: 0;\n  padding-right: ${({ paddingRight }) => paddingRight};\n  padding-left: ${({ paddingLeft }) => paddingLeft};\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`;\n\nexport const Autocomplete2 = ({\n  options,\n  value,\n  onChange,\n  isCreatable,\n  onCreate,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  menuZIndex,\n  placeholder,\n  label,\n  hint,\n  menuItemsHaveEllipsis = true,\n  hasPortal = true,\n  hasError,\n  errorMessage,\n  isDisabled,\n  searchField = 'content',\n  backgroundColor,\n  onClear,\n  hasSpinner,\n  size = 'm',\n  renderGroupTitle,\n  ...props\n}) => {\n  const [query, setQuery] = useState('');\n\n  const handleCreatableChange = option => {\n    if (onCreate) {\n      const isCreatableOption = !options.find(item => item.value === option.value);\n      if (isCreatableOption) {\n        onCreate(option);\n        return;\n      }\n    }\n    onChange(option);\n  };\n\n  const handleChange = option => {\n    if (isCreatable) {\n      handleCreatableChange(option);\n    } else {\n      onChange(option);\n    }\n  };\n\n  const filteredOptions = getFilteredOptions(query, options, searchField);\n\n  const isShowCreateOption = isCreatable && query.length > 0;\n\n  const noResults = !Boolean(filteredOptions.length) && !isShowCreateOption;\n\n  const fieldMarginTop = label || hint ? 'var(--grn-form-control-label-gap)' : undefined;\n\n  const selectedOption = useMemo(\n    () => options.find(option => option.value === value),\n    [options, value],\n  );\n\n  const newOption = useMemo(() => ({ value: query, content: query }), [query]);\n\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n\n  const OptionsRoot = hasPortal ? FloatingPortal : Fragment;\n\n  const hasClearButton = onClear && Boolean(selectedOption);\n\n  const inputPaddingX = `var(--grn-field-paddingX-${size})`;\n\n  const chevronWidth = `${IconChevronDown.viewBoxWidth / 2}px`;\n  const textGap = 'var(--grn-space-s2)';\n  const spinnerSpace = `calc(var(--grn-unit) * 3) + ${rightSectionGap}`;\n  const iconCrossWidth = `${IconCross.viewBoxWidth / 2}px`;\n  const clearButtonSpace = `${iconCrossWidth} + ${rightSectionGap}`;\n\n  const paddingRightValues = [textGap];\n  if (hasSpinner) paddingRightValues.push(spinnerSpace);\n  if (hasClearButton) paddingRightValues.push(clearButtonSpace);\n  paddingRightValues.push(chevronWidth, inputPaddingX);\n\n  const inputPaddingRight = `calc(${paddingRightValues.join(' + ')})`;\n\n  return (\n    <>\n      <Global styles={menuItemStyles} />\n      <Combobox\n        disabled={isDisabled}\n        as=\"div\"\n        value={selectedOption || ''}\n        onChange={handleChange}\n        {...props}\n      >\n        {({ open }) => {\n          const hasItem = selectedOption?.item;\n          return (\n            <>\n              {label && <Combobox.Label as={FieldLabel}>{label}</Combobox.Label>}\n              {hint && <FieldHint>{hint}</FieldHint>}\n\n              <Root\n                marginTop={fieldMarginTop}\n                ref={reference}\n                open={open}\n                hasError={hasError}\n                backgroundColor={backgroundColor}\n                isDisabled={isDisabled}\n              >\n                <Combobox.Button as={Box} position=\"relative\">\n                  <Combobox.Input\n                    as={InputField}\n                    autoComplete=\"off\"\n                    onChange={event => setQuery(event.target.value)}\n                    placeholder={placeholder}\n                    displayValue={option => option && option.content}\n                    paddingRight={inputPaddingRight}\n                    hasError={hasError}\n                    className=\"autocompleteInput\"\n                    size={size}\n                  />\n\n                  <Trigger open={open} tabIndex={-1} paddingRight={inputPaddingRight} size={size}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight} paddingLeft={inputPaddingX}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <RightSection right={inputPaddingX}>\n                    {hasSpinner && <Spinner />}\n                    {hasClearButton && <ClearButton onClick={onClear} trimSides />}\n                    <Icon icon={<IconChevronDown />} />\n                  </RightSection>\n                </Combobox.Button>\n\n                <OptionsRoot>\n                  <MenuCardTransition afterLeave={() => setQuery('')}>\n                    <Combobox.Options\n                      static\n                      ref={floating}\n                      placement={menuPlacement}\n                      maxHeight={menuMaxHeight}\n                      as={MenuCard}\n                      zIndex={menuZIndex}\n                      style={floatingStyles}\n                      isOpen={open}\n                    >\n                      {filteredOptions.map((option, index) => {\n                        const showGroupTitle = getShowGroupTitle(index, option, filteredOptions);\n\n                        return (\n                          <Fragment key={index}>\n                            {showGroupTitle &&\n                              (renderGroupTitle ? (\n                                renderGroupTitle(option.groupTitle)\n                              ) : (\n                                <MenuGroupTitle>{option.groupTitle}</MenuGroupTitle>\n                              ))}\n\n                            <Combobox.Option\n                              value={option}\n                              as={Fragment}\n                              disabled={option.isDisabled}\n                            >\n                              {({ active }) => {\n                                const isDisabled = option.isDisabled;\n                                const isSelected = option.value === value;\n                                const isActive = !option.isDisabled && active;\n                                const hasEllipsis = menuItemsHaveEllipsis;\n                                if (option.item) {\n                                  return option.item({\n                                    isActive,\n                                    isSelected,\n                                    isDisabled,\n                                    hasEllipsis,\n                                  });\n                                }\n                                return <Item hasEllipsis={hasEllipsis}>{option.content}</Item>;\n                              }}\n                            </Combobox.Option>\n                          </Fragment>\n                        );\n                      })}\n                      {isShowCreateOption && (\n                        <>\n                          {filteredOptions.length > 0 && <MenuItemDivider />}\n                          <Combobox.Option value={newOption} as={Fragment}>\n                            {({ active }) => (\n                              <Item isActive={active} hasEllipsis={menuItemsHaveEllipsis}>\n                                <Arrange gap=\"s\" tag=\"span\">\n                                  <Icon icon={<IconPlus />} color={active ? 'content' : 'icon'} />\n                                  Create {`\"` + query + `\"`}\n                                </Arrange>\n                              </Item>\n                            )}\n                          </Combobox.Option>\n                        </>\n                      )}\n                      {noResults && <EmptyState />}\n                    </Combobox.Options>\n                  </MenuCardTransition>\n                </OptionsRoot>\n              </Root>\n              {errorMessage && (\n                <Box marginTop=\"var(--grn-form-control-label-gap)\" color=\"danger\">\n                  {errorMessage}\n                </Box>\n              )}\n            </>\n          );\n        }}\n      </Combobox>\n    </>\n  );\n};\n\nAutocomplete2.Label = FieldLabel;\nAutocomplete2.Hint = FieldHint;\nAutocomplete2.Item = Item;\n\nAutocomplete2.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.string,\n  onChange: PropTypes.func.isRequired,\n  isCreatable: PropTypes.bool,\n  onCreate: PropTypes.func,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuMaxHeight: types.dimension,\n  menuZIndex: types.zIndex,\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n  placeholder: PropTypes.string,\n  hasError: PropTypes.bool,\n  errorMessage: PropTypes.string,\n  isDisabled: PropTypes.bool,\n  backgroundColor: types.color,\n  onClear: PropTypes.func,\n  hasSpinner: PropTypes.bool,\n  size: PropTypes.oneOf(['m', 'l']),\n};\n"]} */"));
190
190
 
191
191
  export const Autocomplete2 = _ref18 => {
192
192
  let {
@@ -211,7 +211,8 @@ export const Autocomplete2 = _ref18 => {
211
211
  backgroundColor,
212
212
  onClear,
213
213
  hasSpinner,
214
- size = 'm'
214
+ size = 'm',
215
+ renderGroupTitle
215
216
  } = _ref18,
216
217
  props = _objectWithoutProperties(_ref18, _excluded3);
217
218
 
@@ -336,9 +337,7 @@ export const Autocomplete2 = _ref18 => {
336
337
  const showGroupTitle = getShowGroupTitle(index, option, filteredOptions);
337
338
  return ___EmotionJSX(Fragment, {
338
339
  key: index
339
- }, showGroupTitle && ___EmotionJSX(MenuGroupTitle, {
340
- hasDivider: index > 0
341
- }, option.groupTitle), ___EmotionJSX(Combobox.Option, {
340
+ }, showGroupTitle && (renderGroupTitle ? renderGroupTitle(option.groupTitle) : ___EmotionJSX(MenuGroupTitle, null, option.groupTitle)), ___EmotionJSX(Combobox.Option, {
342
341
  value: option,
343
342
  as: Fragment,
344
343
  disabled: option.isDisabled
@@ -11,7 +11,7 @@ function _objectWithoutPropertiesLoose(source, excluded) { if (source == null) r
11
11
 
12
12
  import React, { Fragment, forwardRef, useCallback, useEffect, useImperativeHandle, useRef } from 'react';
13
13
  import PropTypes from 'prop-types';
14
- import { Global } from '@emotion/react';
14
+ import { Global, css } from '@emotion/react';
15
15
  import { getColor, getDimension, getRadius, getShadow, getSpace, getTransition } from '../utilities';
16
16
  import { Box, Icon, Text } from '../components';
17
17
  import { IconCheck } from '../icons';
@@ -126,7 +126,7 @@ const MenuCardUl = /*#__PURE__*/_styled("ul", process.env.NODE_ENV === "producti
126
126
  isDisabled
127
127
  } = _ref7;
128
128
  return isDisabled && "pointer-events: none;";
129
- }, " &:focus{outline:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AAyG4B","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, {\n  Fragment,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useRef,\n} from 'react';\nimport PropTypes from 'prop-types';\nimport { Global } from '@emotion/react';\nimport {\n  getColor,\n  getDimension,\n  getRadius,\n  getShadow,\n  getSpace,\n  getTransition,\n} from '../utilities';\nimport { Box, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\nimport { types } from '../types';\nimport styled from '@emotion/styled';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = `\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(({ children, show, ...props }, ref) => (\n  <Transition\n    as={Fragment}\n    enterFrom=\"menuCardEnterFrom\"\n    enterTo=\"menuCardEnterTo\"\n    leaveFrom=\"menuCardLeaveFrom\"\n    leaveTo=\"menuCardLeaveTo\"\n    ref={ref}\n    show={show}\n    {...props}\n  >\n    {children}\n  </Transition>\n));\n\nconst MenuCardUl = styled.ul`\n  padding: ${cardPaddingPx};\n  margin: 0;\n  background: ${getColor('floatingBackground')};\n  z-index: ${({ zIndex }) => zIndex};\n  box-shadow: ${getShadow('m')};\n  border-radius: ${getRadius('m')};\n  overflow: auto;\n  ${({ maxHeight }) => maxHeight && `max-height: ${getSpace(maxHeight)}`};\n  ${({ maxWidth }) => maxWidth && `max-width: ${getSpace(maxWidth)}`};\n  /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n  ${({ isDisabled }) => isDisabled && `pointer-events: none;`}\n\n  &:focus {\n    outline: none;\n  }\n`;\n\nexport const MenuCard = forwardRef(\n  ({ children, zIndex = 1, maxHeight, maxWidth, isDisabled, ...props }, ref) => {\n    const menuCardRef = useRef(ref);\n    useImperativeHandle(ref, () => menuCardRef.current, []);\n\n    const scrollToActive = useCallback(() => {\n      // Note: this function is rely on data-headlessui-state attribute\n      const menuCard = menuCardRef.current;\n      const activeItem = menuCard?.querySelector('[data-headlessui-state=\"selected\"]');\n      if (!menuCard || !activeItem) return;\n      const menuCardRect = menuCard.getBoundingClientRect();\n      const activeItemRect = activeItem.getBoundingClientRect();\n      menuCard.scrollTop =\n        activeItem.offsetTop - menuCardRect.height / 2 + activeItemRect.height / 2;\n    }, []);\n\n    useEffect(() => {\n      scrollToActive();\n    }, [scrollToActive]);\n\n    return (\n      <>\n        <Global styles={menuCardStyles} />\n        <MenuCardUl\n          ref={menuCardRef}\n          zIndex={zIndex}\n          maxHeight={maxHeight}\n          maxWidth={maxWidth}\n          isDisabled={isDisabled}\n          className=\"MenuCard grn-context\"\n          {...props}\n        >\n          {children}\n        </MenuCardUl>\n      </>\n    );\n  },\n);\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('fade1'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('danger'),\n    iconColor: 'var(--grn-icon-danger-color)',\n    iconColorActive: getColor('danger'),\n    backgroundColorActive: 'var(--grn-clearbutton-danger-background-hover)',\n  },\n};\n\nconst MenuItemRoot = styled.li`\n  display: grid;\n  grid-auto-flow: column;\n  gap: 8px;\n  ${({ isDisabled }) => isDisabled && `color: ${getColor('disabledContent')}`};\n  ${({ isDisabled }) => !isDisabled && `cursor: pointer;`};\n  align-items: center;\n  justify-content: start;\n  list-style: none;\n  padding-block: 10px;\n  padding-inline: ${itemXPadding}px;\n  margin: 0;\n  min-height: var(--grn-textbox-height-m);\n  border-radius: ${getRadius('s')};\n  appearance: none;\n  color: ${({ color }) => color};\n\n  [data-experiment-compact-textbox] & {\n    padding-block: ${getSpace('s')};\n  }\n`;\n\nexport const MenuGroupTitle = ({ children, hasDivider }) => (\n  <>\n    {hasDivider && (\n      <Box\n        height=\"3px\"\n        borderSide={hasDivider ? 'top' : undefined}\n        marginTop={cardPaddingPx}\n        paddingTop={`${cardPadding * 2}px`}\n        marginX={`-${cardPaddingPx}`}\n      />\n    )}\n    <Box\n      paddingX={`${itemXPadding}px`}\n      paddingBottom={cardPaddingPx}\n      paddingTop={hasDivider ? 0 : cardPaddingPx}\n    >\n      <Text variant=\"caps\">{children}</Text>\n    </Box>\n  </>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height=\"1px\" marginY={1} backgroundColor=\"border\" marginX={`-${cardPaddingPx}`} />\n);\n\nexport const MenuItem = forwardRef(\n  (\n    {\n      children,\n      isSelected,\n      isActive,\n      isDisabled,\n      icon,\n      variant = 'neutral',\n      hasEllipsis,\n      checkVariant = 'check',\n      ...props\n    },\n    ref,\n  ) => {\n    const hasCheck = checkVariant === 'check';\n\n    const getColumns = () => {\n      if (Boolean(icon)) return 'auto 1fr';\n      if (hasCheck) return '1fr auto';\n    };\n\n    return (\n      <MenuItemRoot\n        ref={ref}\n        isSelected={isSelected}\n        isDisabled={isDisabled}\n        color={variantStyles[variant].color}\n        hasIcon={Boolean(icon)}\n        style={{\n          background: isActive && variantStyles[variant].backgroundColorActive,\n          gridTemplateColumns: getColumns(),\n        }}\n        {...props}\n      >\n        {icon && (\n          <Icon\n            icon={icon}\n            hasEvenBoundary\n            color={\n              isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n            }\n          />\n        )}\n        <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n          <Text hasEllipsis={hasEllipsis}>{children}</Text>\n        </Box>\n        {hasCheck && (\n          <Box opacity={isSelected ? '1' : '0'}>\n            <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n          </Box>\n        )}\n      </MenuItemRoot>\n    );\n  },\n);\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'none']),\n};\n\nMenuCard.propTypes = {\n  children: PropTypes.node,\n  zIndex: types.zIndex,\n  maxHeight: types.dimension,\n  maxWidth: types.dimension,\n  isDisabled: PropTypes.bool,\n};\n\nMenuCardTransition.propTypes = {\n  children: PropTypes.node,\n  show: PropTypes.bool,\n};\n"]} */"));
129
+ }, " &:focus{outline:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AAyG4B","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, {\n  Fragment,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useRef,\n} from 'react';\nimport PropTypes from 'prop-types';\nimport { Global, css } from '@emotion/react';\nimport {\n  getColor,\n  getDimension,\n  getRadius,\n  getShadow,\n  getSpace,\n  getTransition,\n} from '../utilities';\nimport { Box, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\nimport { types } from '../types';\nimport styled from '@emotion/styled';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = `\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(({ children, show, ...props }, ref) => (\n  <Transition\n    as={Fragment}\n    enterFrom=\"menuCardEnterFrom\"\n    enterTo=\"menuCardEnterTo\"\n    leaveFrom=\"menuCardLeaveFrom\"\n    leaveTo=\"menuCardLeaveTo\"\n    ref={ref}\n    show={show}\n    {...props}\n  >\n    {children}\n  </Transition>\n));\n\nconst MenuCardUl = styled.ul`\n  padding: ${cardPaddingPx};\n  margin: 0;\n  background: ${getColor('floatingBackground')};\n  z-index: ${({ zIndex }) => zIndex};\n  box-shadow: ${getShadow('m')};\n  border-radius: ${getRadius('m')};\n  overflow: auto;\n  ${({ maxHeight }) => maxHeight && `max-height: ${getSpace(maxHeight)}`};\n  ${({ maxWidth }) => maxWidth && `max-width: ${getSpace(maxWidth)}`};\n  /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n  ${({ isDisabled }) => isDisabled && `pointer-events: none;`}\n\n  &:focus {\n    outline: none;\n  }\n`;\n\nexport const MenuCard = forwardRef(\n  ({ children, zIndex = 1, maxHeight, maxWidth, isDisabled, ...props }, ref) => {\n    const menuCardRef = useRef(ref);\n    useImperativeHandle(ref, () => menuCardRef.current, []);\n\n    const scrollToActive = useCallback(() => {\n      // Note: this function is rely on data-headlessui-state attribute\n      const menuCard = menuCardRef.current;\n      const activeItem = menuCard?.querySelector('[data-headlessui-state=\"selected\"]');\n      if (!menuCard || !activeItem) return;\n      const menuCardRect = menuCard.getBoundingClientRect();\n      const activeItemRect = activeItem.getBoundingClientRect();\n      menuCard.scrollTop =\n        activeItem.offsetTop - menuCardRect.height / 2 + activeItemRect.height / 2;\n    }, []);\n\n    useEffect(() => {\n      scrollToActive();\n    }, [scrollToActive]);\n\n    return (\n      <>\n        <Global styles={menuCardStyles} />\n        <MenuCardUl\n          ref={menuCardRef}\n          zIndex={zIndex}\n          maxHeight={maxHeight}\n          maxWidth={maxWidth}\n          isDisabled={isDisabled}\n          className=\"MenuCard grn-context\"\n          {...props}\n        >\n          {children}\n        </MenuCardUl>\n      </>\n    );\n  },\n);\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('fade1'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('danger'),\n    iconColor: 'var(--grn-icon-danger-color)',\n    iconColorActive: getColor('danger'),\n    backgroundColorActive: 'var(--grn-clearbutton-danger-background-hover)',\n  },\n};\n\nconst MenuItemRoot = styled.li`\n  display: grid;\n  grid-auto-flow: column;\n  gap: 8px;\n  ${({ isDisabled }) => isDisabled && `color: ${getColor('disabledContent')}`};\n  ${({ isDisabled }) => !isDisabled && `cursor: pointer;`};\n  align-items: center;\n  justify-content: start;\n  list-style: none;\n  padding-block: 10px;\n  padding-inline: ${itemXPadding}px;\n  margin: 0;\n  min-height: var(--grn-textbox-height-m);\n  border-radius: ${getRadius('s')};\n  appearance: none;\n  color: ${({ color }) => color};\n\n  [data-experiment-compact-textbox] & {\n    padding-block: ${getSpace('s')};\n  }\n`;\n\nexport const MenuGroupTitle = ({ children }) => (\n  <Box\n    css={css`\n      &:first-of-type {\n        --marginTop: calc(-1 * var(--grn-space-s));\n        --borderColor: transparent;\n      }\n      &:not(:first-of-type) {\n        --marginTop: ${cardPaddingPx};\n        --borderColor: var(--grn-color-border);\n      }\n    `}\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingTop=\"m\"\n    paddingBottom={cardPaddingPx}\n    marginTop=\"var(--marginTop)\"\n    borderSide=\"top\"\n    borderColor=\"var(--borderColor)\"\n    marginX={`-${cardPaddingPx}`}\n    position=\"relative\"\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height=\"1px\" marginY={1} backgroundColor=\"border\" marginX={`-${cardPaddingPx}`} />\n);\n\nexport const MenuItem = forwardRef(\n  (\n    {\n      children,\n      isSelected,\n      isActive,\n      isDisabled,\n      icon,\n      variant = 'neutral',\n      hasEllipsis,\n      checkVariant = 'check',\n      ...props\n    },\n    ref,\n  ) => {\n    const hasCheck = checkVariant === 'check';\n\n    const getColumns = () => {\n      if (Boolean(icon)) return 'auto 1fr';\n      if (hasCheck) return '1fr auto';\n    };\n\n    return (\n      <MenuItemRoot\n        ref={ref}\n        isSelected={isSelected}\n        isDisabled={isDisabled}\n        color={variantStyles[variant].color}\n        hasIcon={Boolean(icon)}\n        style={{\n          background: isActive && variantStyles[variant].backgroundColorActive,\n          gridTemplateColumns: getColumns(),\n        }}\n        {...props}\n      >\n        {icon && (\n          <Icon\n            icon={icon}\n            hasEvenBoundary\n            color={\n              isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n            }\n          />\n        )}\n        <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n          <Text hasEllipsis={hasEllipsis}>{children}</Text>\n        </Box>\n        {hasCheck && (\n          <Box opacity={isSelected ? '1' : '0'}>\n            <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n          </Box>\n        )}\n      </MenuItemRoot>\n    );\n  },\n);\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'none']),\n};\n\nMenuCard.propTypes = {\n  children: PropTypes.node,\n  zIndex: types.zIndex,\n  maxHeight: types.dimension,\n  maxWidth: types.dimension,\n  isDisabled: PropTypes.bool,\n};\n\nMenuCardTransition.propTypes = {\n  children: PropTypes.node,\n  show: PropTypes.bool,\n};\n"]} */"));
130
130
 
131
131
  export const MenuCard = /*#__PURE__*/forwardRef((_ref8, ref) => {
132
132
  let {
@@ -196,26 +196,25 @@ const MenuItemRoot = /*#__PURE__*/_styled("li", process.env.NODE_ENV === "produc
196
196
  color
197
197
  } = _ref11;
198
198
  return color;
199
- }, ";[data-experiment-compact-textbox] &{padding-block:", getSpace('s'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AA+K8B","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, {\n  Fragment,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useRef,\n} from 'react';\nimport PropTypes from 'prop-types';\nimport { Global } from '@emotion/react';\nimport {\n  getColor,\n  getDimension,\n  getRadius,\n  getShadow,\n  getSpace,\n  getTransition,\n} from '../utilities';\nimport { Box, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\nimport { types } from '../types';\nimport styled from '@emotion/styled';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = `\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(({ children, show, ...props }, ref) => (\n  <Transition\n    as={Fragment}\n    enterFrom=\"menuCardEnterFrom\"\n    enterTo=\"menuCardEnterTo\"\n    leaveFrom=\"menuCardLeaveFrom\"\n    leaveTo=\"menuCardLeaveTo\"\n    ref={ref}\n    show={show}\n    {...props}\n  >\n    {children}\n  </Transition>\n));\n\nconst MenuCardUl = styled.ul`\n  padding: ${cardPaddingPx};\n  margin: 0;\n  background: ${getColor('floatingBackground')};\n  z-index: ${({ zIndex }) => zIndex};\n  box-shadow: ${getShadow('m')};\n  border-radius: ${getRadius('m')};\n  overflow: auto;\n  ${({ maxHeight }) => maxHeight && `max-height: ${getSpace(maxHeight)}`};\n  ${({ maxWidth }) => maxWidth && `max-width: ${getSpace(maxWidth)}`};\n  /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n  ${({ isDisabled }) => isDisabled && `pointer-events: none;`}\n\n  &:focus {\n    outline: none;\n  }\n`;\n\nexport const MenuCard = forwardRef(\n  ({ children, zIndex = 1, maxHeight, maxWidth, isDisabled, ...props }, ref) => {\n    const menuCardRef = useRef(ref);\n    useImperativeHandle(ref, () => menuCardRef.current, []);\n\n    const scrollToActive = useCallback(() => {\n      // Note: this function is rely on data-headlessui-state attribute\n      const menuCard = menuCardRef.current;\n      const activeItem = menuCard?.querySelector('[data-headlessui-state=\"selected\"]');\n      if (!menuCard || !activeItem) return;\n      const menuCardRect = menuCard.getBoundingClientRect();\n      const activeItemRect = activeItem.getBoundingClientRect();\n      menuCard.scrollTop =\n        activeItem.offsetTop - menuCardRect.height / 2 + activeItemRect.height / 2;\n    }, []);\n\n    useEffect(() => {\n      scrollToActive();\n    }, [scrollToActive]);\n\n    return (\n      <>\n        <Global styles={menuCardStyles} />\n        <MenuCardUl\n          ref={menuCardRef}\n          zIndex={zIndex}\n          maxHeight={maxHeight}\n          maxWidth={maxWidth}\n          isDisabled={isDisabled}\n          className=\"MenuCard grn-context\"\n          {...props}\n        >\n          {children}\n        </MenuCardUl>\n      </>\n    );\n  },\n);\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('fade1'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('danger'),\n    iconColor: 'var(--grn-icon-danger-color)',\n    iconColorActive: getColor('danger'),\n    backgroundColorActive: 'var(--grn-clearbutton-danger-background-hover)',\n  },\n};\n\nconst MenuItemRoot = styled.li`\n  display: grid;\n  grid-auto-flow: column;\n  gap: 8px;\n  ${({ isDisabled }) => isDisabled && `color: ${getColor('disabledContent')}`};\n  ${({ isDisabled }) => !isDisabled && `cursor: pointer;`};\n  align-items: center;\n  justify-content: start;\n  list-style: none;\n  padding-block: 10px;\n  padding-inline: ${itemXPadding}px;\n  margin: 0;\n  min-height: var(--grn-textbox-height-m);\n  border-radius: ${getRadius('s')};\n  appearance: none;\n  color: ${({ color }) => color};\n\n  [data-experiment-compact-textbox] & {\n    padding-block: ${getSpace('s')};\n  }\n`;\n\nexport const MenuGroupTitle = ({ children, hasDivider }) => (\n  <>\n    {hasDivider && (\n      <Box\n        height=\"3px\"\n        borderSide={hasDivider ? 'top' : undefined}\n        marginTop={cardPaddingPx}\n        paddingTop={`${cardPadding * 2}px`}\n        marginX={`-${cardPaddingPx}`}\n      />\n    )}\n    <Box\n      paddingX={`${itemXPadding}px`}\n      paddingBottom={cardPaddingPx}\n      paddingTop={hasDivider ? 0 : cardPaddingPx}\n    >\n      <Text variant=\"caps\">{children}</Text>\n    </Box>\n  </>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height=\"1px\" marginY={1} backgroundColor=\"border\" marginX={`-${cardPaddingPx}`} />\n);\n\nexport const MenuItem = forwardRef(\n  (\n    {\n      children,\n      isSelected,\n      isActive,\n      isDisabled,\n      icon,\n      variant = 'neutral',\n      hasEllipsis,\n      checkVariant = 'check',\n      ...props\n    },\n    ref,\n  ) => {\n    const hasCheck = checkVariant === 'check';\n\n    const getColumns = () => {\n      if (Boolean(icon)) return 'auto 1fr';\n      if (hasCheck) return '1fr auto';\n    };\n\n    return (\n      <MenuItemRoot\n        ref={ref}\n        isSelected={isSelected}\n        isDisabled={isDisabled}\n        color={variantStyles[variant].color}\n        hasIcon={Boolean(icon)}\n        style={{\n          background: isActive && variantStyles[variant].backgroundColorActive,\n          gridTemplateColumns: getColumns(),\n        }}\n        {...props}\n      >\n        {icon && (\n          <Icon\n            icon={icon}\n            hasEvenBoundary\n            color={\n              isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n            }\n          />\n        )}\n        <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n          <Text hasEllipsis={hasEllipsis}>{children}</Text>\n        </Box>\n        {hasCheck && (\n          <Box opacity={isSelected ? '1' : '0'}>\n            <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n          </Box>\n        )}\n      </MenuItemRoot>\n    );\n  },\n);\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'none']),\n};\n\nMenuCard.propTypes = {\n  children: PropTypes.node,\n  zIndex: types.zIndex,\n  maxHeight: types.dimension,\n  maxWidth: types.dimension,\n  isDisabled: PropTypes.bool,\n};\n\nMenuCardTransition.propTypes = {\n  children: PropTypes.node,\n  show: PropTypes.bool,\n};\n"]} */"));
199
+ }, ";[data-experiment-compact-textbox] &{padding-block:", getSpace('s'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AA+K8B","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, {\n  Fragment,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useRef,\n} from 'react';\nimport PropTypes from 'prop-types';\nimport { Global, css } from '@emotion/react';\nimport {\n  getColor,\n  getDimension,\n  getRadius,\n  getShadow,\n  getSpace,\n  getTransition,\n} from '../utilities';\nimport { Box, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\nimport { types } from '../types';\nimport styled from '@emotion/styled';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = `\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(({ children, show, ...props }, ref) => (\n  <Transition\n    as={Fragment}\n    enterFrom=\"menuCardEnterFrom\"\n    enterTo=\"menuCardEnterTo\"\n    leaveFrom=\"menuCardLeaveFrom\"\n    leaveTo=\"menuCardLeaveTo\"\n    ref={ref}\n    show={show}\n    {...props}\n  >\n    {children}\n  </Transition>\n));\n\nconst MenuCardUl = styled.ul`\n  padding: ${cardPaddingPx};\n  margin: 0;\n  background: ${getColor('floatingBackground')};\n  z-index: ${({ zIndex }) => zIndex};\n  box-shadow: ${getShadow('m')};\n  border-radius: ${getRadius('m')};\n  overflow: auto;\n  ${({ maxHeight }) => maxHeight && `max-height: ${getSpace(maxHeight)}`};\n  ${({ maxWidth }) => maxWidth && `max-width: ${getSpace(maxWidth)}`};\n  /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n  ${({ isDisabled }) => isDisabled && `pointer-events: none;`}\n\n  &:focus {\n    outline: none;\n  }\n`;\n\nexport const MenuCard = forwardRef(\n  ({ children, zIndex = 1, maxHeight, maxWidth, isDisabled, ...props }, ref) => {\n    const menuCardRef = useRef(ref);\n    useImperativeHandle(ref, () => menuCardRef.current, []);\n\n    const scrollToActive = useCallback(() => {\n      // Note: this function is rely on data-headlessui-state attribute\n      const menuCard = menuCardRef.current;\n      const activeItem = menuCard?.querySelector('[data-headlessui-state=\"selected\"]');\n      if (!menuCard || !activeItem) return;\n      const menuCardRect = menuCard.getBoundingClientRect();\n      const activeItemRect = activeItem.getBoundingClientRect();\n      menuCard.scrollTop =\n        activeItem.offsetTop - menuCardRect.height / 2 + activeItemRect.height / 2;\n    }, []);\n\n    useEffect(() => {\n      scrollToActive();\n    }, [scrollToActive]);\n\n    return (\n      <>\n        <Global styles={menuCardStyles} />\n        <MenuCardUl\n          ref={menuCardRef}\n          zIndex={zIndex}\n          maxHeight={maxHeight}\n          maxWidth={maxWidth}\n          isDisabled={isDisabled}\n          className=\"MenuCard grn-context\"\n          {...props}\n        >\n          {children}\n        </MenuCardUl>\n      </>\n    );\n  },\n);\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('fade1'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('danger'),\n    iconColor: 'var(--grn-icon-danger-color)',\n    iconColorActive: getColor('danger'),\n    backgroundColorActive: 'var(--grn-clearbutton-danger-background-hover)',\n  },\n};\n\nconst MenuItemRoot = styled.li`\n  display: grid;\n  grid-auto-flow: column;\n  gap: 8px;\n  ${({ isDisabled }) => isDisabled && `color: ${getColor('disabledContent')}`};\n  ${({ isDisabled }) => !isDisabled && `cursor: pointer;`};\n  align-items: center;\n  justify-content: start;\n  list-style: none;\n  padding-block: 10px;\n  padding-inline: ${itemXPadding}px;\n  margin: 0;\n  min-height: var(--grn-textbox-height-m);\n  border-radius: ${getRadius('s')};\n  appearance: none;\n  color: ${({ color }) => color};\n\n  [data-experiment-compact-textbox] & {\n    padding-block: ${getSpace('s')};\n  }\n`;\n\nexport const MenuGroupTitle = ({ children }) => (\n  <Box\n    css={css`\n      &:first-of-type {\n        --marginTop: calc(-1 * var(--grn-space-s));\n        --borderColor: transparent;\n      }\n      &:not(:first-of-type) {\n        --marginTop: ${cardPaddingPx};\n        --borderColor: var(--grn-color-border);\n      }\n    `}\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingTop=\"m\"\n    paddingBottom={cardPaddingPx}\n    marginTop=\"var(--marginTop)\"\n    borderSide=\"top\"\n    borderColor=\"var(--borderColor)\"\n    marginX={`-${cardPaddingPx}`}\n    position=\"relative\"\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height=\"1px\" marginY={1} backgroundColor=\"border\" marginX={`-${cardPaddingPx}`} />\n);\n\nexport const MenuItem = forwardRef(\n  (\n    {\n      children,\n      isSelected,\n      isActive,\n      isDisabled,\n      icon,\n      variant = 'neutral',\n      hasEllipsis,\n      checkVariant = 'check',\n      ...props\n    },\n    ref,\n  ) => {\n    const hasCheck = checkVariant === 'check';\n\n    const getColumns = () => {\n      if (Boolean(icon)) return 'auto 1fr';\n      if (hasCheck) return '1fr auto';\n    };\n\n    return (\n      <MenuItemRoot\n        ref={ref}\n        isSelected={isSelected}\n        isDisabled={isDisabled}\n        color={variantStyles[variant].color}\n        hasIcon={Boolean(icon)}\n        style={{\n          background: isActive && variantStyles[variant].backgroundColorActive,\n          gridTemplateColumns: getColumns(),\n        }}\n        {...props}\n      >\n        {icon && (\n          <Icon\n            icon={icon}\n            hasEvenBoundary\n            color={\n              isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n            }\n          />\n        )}\n        <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n          <Text hasEllipsis={hasEllipsis}>{children}</Text>\n        </Box>\n        {hasCheck && (\n          <Box opacity={isSelected ? '1' : '0'}>\n            <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n          </Box>\n        )}\n      </MenuItemRoot>\n    );\n  },\n);\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'none']),\n};\n\nMenuCard.propTypes = {\n  children: PropTypes.node,\n  zIndex: types.zIndex,\n  maxHeight: types.dimension,\n  maxWidth: types.dimension,\n  isDisabled: PropTypes.bool,\n};\n\nMenuCardTransition.propTypes = {\n  children: PropTypes.node,\n  show: PropTypes.bool,\n};\n"]} */"));
200
200
 
201
201
  export const MenuGroupTitle = _ref12 => {
202
202
  let {
203
- children,
204
- hasDivider
203
+ children
205
204
  } = _ref12;
206
- return ___EmotionJSX(React.Fragment, null, hasDivider && ___EmotionJSX(Box, {
207
- height: "3px",
208
- borderSide: hasDivider ? 'top' : undefined,
209
- marginTop: cardPaddingPx,
210
- paddingTop: "".concat(cardPadding * 2, "px"),
211
- marginX: "-".concat(cardPaddingPx)
212
- }), ___EmotionJSX(Box, {
213
- paddingX: "".concat(itemXPadding, "px"),
205
+ return ___EmotionJSX(Box, {
206
+ css: /*#__PURE__*/css("&:first-of-type{--marginTop:calc(-1 * var(--grn-space-s));--borderColor:transparent;}&:not(:first-of-type){--marginTop:", cardPaddingPx, ";--borderColor:var(--grn-color-border);}" + (process.env.NODE_ENV === "production" ? "" : ";label:MenuGroupTitle;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AAuMY","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, {\n  Fragment,\n  forwardRef,\n  useCallback,\n  useEffect,\n  useImperativeHandle,\n  useRef,\n} from 'react';\nimport PropTypes from 'prop-types';\nimport { Global, css } from '@emotion/react';\nimport {\n  getColor,\n  getDimension,\n  getRadius,\n  getShadow,\n  getSpace,\n  getTransition,\n} from '../utilities';\nimport { Box, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\nimport { types } from '../types';\nimport styled from '@emotion/styled';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = `\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(({ children, show, ...props }, ref) => (\n  <Transition\n    as={Fragment}\n    enterFrom=\"menuCardEnterFrom\"\n    enterTo=\"menuCardEnterTo\"\n    leaveFrom=\"menuCardLeaveFrom\"\n    leaveTo=\"menuCardLeaveTo\"\n    ref={ref}\n    show={show}\n    {...props}\n  >\n    {children}\n  </Transition>\n));\n\nconst MenuCardUl = styled.ul`\n  padding: ${cardPaddingPx};\n  margin: 0;\n  background: ${getColor('floatingBackground')};\n  z-index: ${({ zIndex }) => zIndex};\n  box-shadow: ${getShadow('m')};\n  border-radius: ${getRadius('m')};\n  overflow: auto;\n  ${({ maxHeight }) => maxHeight && `max-height: ${getSpace(maxHeight)}`};\n  ${({ maxWidth }) => maxWidth && `max-width: ${getSpace(maxWidth)}`};\n  /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n  ${({ isDisabled }) => isDisabled && `pointer-events: none;`}\n\n  &:focus {\n    outline: none;\n  }\n`;\n\nexport const MenuCard = forwardRef(\n  ({ children, zIndex = 1, maxHeight, maxWidth, isDisabled, ...props }, ref) => {\n    const menuCardRef = useRef(ref);\n    useImperativeHandle(ref, () => menuCardRef.current, []);\n\n    const scrollToActive = useCallback(() => {\n      // Note: this function is rely on data-headlessui-state attribute\n      const menuCard = menuCardRef.current;\n      const activeItem = menuCard?.querySelector('[data-headlessui-state=\"selected\"]');\n      if (!menuCard || !activeItem) return;\n      const menuCardRect = menuCard.getBoundingClientRect();\n      const activeItemRect = activeItem.getBoundingClientRect();\n      menuCard.scrollTop =\n        activeItem.offsetTop - menuCardRect.height / 2 + activeItemRect.height / 2;\n    }, []);\n\n    useEffect(() => {\n      scrollToActive();\n    }, [scrollToActive]);\n\n    return (\n      <>\n        <Global styles={menuCardStyles} />\n        <MenuCardUl\n          ref={menuCardRef}\n          zIndex={zIndex}\n          maxHeight={maxHeight}\n          maxWidth={maxWidth}\n          isDisabled={isDisabled}\n          className=\"MenuCard grn-context\"\n          {...props}\n        >\n          {children}\n        </MenuCardUl>\n      </>\n    );\n  },\n);\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('fade1'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('danger'),\n    iconColor: 'var(--grn-icon-danger-color)',\n    iconColorActive: getColor('danger'),\n    backgroundColorActive: 'var(--grn-clearbutton-danger-background-hover)',\n  },\n};\n\nconst MenuItemRoot = styled.li`\n  display: grid;\n  grid-auto-flow: column;\n  gap: 8px;\n  ${({ isDisabled }) => isDisabled && `color: ${getColor('disabledContent')}`};\n  ${({ isDisabled }) => !isDisabled && `cursor: pointer;`};\n  align-items: center;\n  justify-content: start;\n  list-style: none;\n  padding-block: 10px;\n  padding-inline: ${itemXPadding}px;\n  margin: 0;\n  min-height: var(--grn-textbox-height-m);\n  border-radius: ${getRadius('s')};\n  appearance: none;\n  color: ${({ color }) => color};\n\n  [data-experiment-compact-textbox] & {\n    padding-block: ${getSpace('s')};\n  }\n`;\n\nexport const MenuGroupTitle = ({ children }) => (\n  <Box\n    css={css`\n      &:first-of-type {\n        --marginTop: calc(-1 * var(--grn-space-s));\n        --borderColor: transparent;\n      }\n      &:not(:first-of-type) {\n        --marginTop: ${cardPaddingPx};\n        --borderColor: var(--grn-color-border);\n      }\n    `}\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingTop=\"m\"\n    paddingBottom={cardPaddingPx}\n    marginTop=\"var(--marginTop)\"\n    borderSide=\"top\"\n    borderColor=\"var(--borderColor)\"\n    marginX={`-${cardPaddingPx}`}\n    position=\"relative\"\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height=\"1px\" marginY={1} backgroundColor=\"border\" marginX={`-${cardPaddingPx}`} />\n);\n\nexport const MenuItem = forwardRef(\n  (\n    {\n      children,\n      isSelected,\n      isActive,\n      isDisabled,\n      icon,\n      variant = 'neutral',\n      hasEllipsis,\n      checkVariant = 'check',\n      ...props\n    },\n    ref,\n  ) => {\n    const hasCheck = checkVariant === 'check';\n\n    const getColumns = () => {\n      if (Boolean(icon)) return 'auto 1fr';\n      if (hasCheck) return '1fr auto';\n    };\n\n    return (\n      <MenuItemRoot\n        ref={ref}\n        isSelected={isSelected}\n        isDisabled={isDisabled}\n        color={variantStyles[variant].color}\n        hasIcon={Boolean(icon)}\n        style={{\n          background: isActive && variantStyles[variant].backgroundColorActive,\n          gridTemplateColumns: getColumns(),\n        }}\n        {...props}\n      >\n        {icon && (\n          <Icon\n            icon={icon}\n            hasEvenBoundary\n            color={\n              isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n            }\n          />\n        )}\n        <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n          <Text hasEllipsis={hasEllipsis}>{children}</Text>\n        </Box>\n        {hasCheck && (\n          <Box opacity={isSelected ? '1' : '0'}>\n            <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n          </Box>\n        )}\n      </MenuItemRoot>\n    );\n  },\n);\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'none']),\n};\n\nMenuCard.propTypes = {\n  children: PropTypes.node,\n  zIndex: types.zIndex,\n  maxHeight: types.dimension,\n  maxWidth: types.dimension,\n  isDisabled: PropTypes.bool,\n};\n\nMenuCardTransition.propTypes = {\n  children: PropTypes.node,\n  show: PropTypes.bool,\n};\n"]} */"),
207
+ paddingX: "".concat(cardPadding + itemXPadding, "px"),
208
+ paddingTop: "m",
214
209
  paddingBottom: cardPaddingPx,
215
- paddingTop: hasDivider ? 0 : cardPaddingPx
210
+ marginTop: "var(--marginTop)",
211
+ borderSide: "top",
212
+ borderColor: "var(--borderColor)",
213
+ marginX: "-".concat(cardPaddingPx),
214
+ position: "relative"
216
215
  }, ___EmotionJSX(Text, {
217
216
  variant: "caps"
218
- }, children)));
217
+ }, children));
219
218
  };
220
219
  export const MenuItemDivider = () => ___EmotionJSX(Box, {
221
220
  height: "1px",
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@flodesk/grain",
3
- "version": "11.47.2",
3
+ "version": "11.48.0",
4
4
  "description": "Flodesk design system",
5
5
  "module": "es/index.js",
6
6
  "types": "es/types/index.d.ts",