@flodesk/grain 11.40.0 → 11.41.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.
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
import _styled from "@emotion/styled/base";
|
|
2
|
-
const _excluded = ["
|
|
3
|
-
_excluded2 = ["
|
|
2
|
+
const _excluded = ["onClick"],
|
|
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"];
|
|
4
5
|
import "core-js/modules/web.dom-collections.iterator.js";
|
|
5
6
|
|
|
6
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); }
|
|
@@ -11,8 +12,8 @@ function _objectWithoutPropertiesLoose(source, excluded) { if (source == null) r
|
|
|
11
12
|
|
|
12
13
|
import PropTypes from 'prop-types';
|
|
13
14
|
import React, { forwardRef, Fragment, useMemo, useState } from 'react';
|
|
14
|
-
import { Icon, Box, Arrange } from '.';
|
|
15
|
-
import { IconChevronDown, IconPlus } from '../icons';
|
|
15
|
+
import { Icon, Box, Arrange, Spinner } from '.';
|
|
16
|
+
import { IconChevronDown, IconPlus, IconCross } from '../icons';
|
|
16
17
|
import { Combobox } from '@headlessui/react';
|
|
17
18
|
import { defaultProps, types } from '../types';
|
|
18
19
|
import { InputField, MenuCard, MenuCardTransition, MenuGroupTitle, FieldLabel, FieldHint, styles, FieldClearButton } from '../foundational';
|
|
@@ -70,7 +71,7 @@ const Root = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "production" ?
|
|
|
70
71
|
open
|
|
71
72
|
} = _ref8;
|
|
72
73
|
return open ? 'auto' : 'none';
|
|
73
|
-
}, ";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 } from '.';\nimport { IconChevronDown, IconPlus } 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 ActionsSection = ({ children }) => (\n  <Arrange\n    right=\"var(--grn-field-paddingX)\"\n    gap=\"xs\"\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 }) => {\n  const handleClear = e => {\n    e.stopPropagation();\n    onClick();\n  };\n  return <FieldClearButton onClick={e => handleClear(e)} />;\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: var(--grn-textbox-height-m);\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: 34px;\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-left: var(--grn-field-paddingX);\n  padding-right: ${({ paddingRight }) => paddingRight};\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  ...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 chevronWidth = '12px';\n  const chevronGap = getSpace('s');\n  const paddingRightWithoutClear = `var(--grn-field-paddingX) + ${chevronWidth} + ${chevronGap}`;\n  const inputPaddingRight = hasClearButton\n    ? `calc(${paddingRightWithoutClear} + var(--grn-clearbutton-height))`\n    : `calc(${paddingRightWithoutClear})`;\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                  />\n\n                  <Trigger open={open} tabIndex={-1}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <ActionsSection>\n                    {hasClearButton && <ClearButton onClick={onClear} />}\n                    <Icon icon={<IconChevronDown />} />\n                  </ActionsSection>\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};\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 }) => (\n  <Arrange\n    right=\"var(--grn-field-paddingX)\"\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: var(--grn-textbox-height-m);\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-left: var(--grn-field-paddingX);\n  padding-right: ${({ paddingRight }) => paddingRight};\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  ...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 iconChevronDownWidth = `${IconChevronDown.viewBoxWidth / 2}px`;\n  const chevronRightSpace = 'var(--grn-field-paddingX)';\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(iconChevronDownWidth, chevronRightSpace);\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                  />\n\n                  <Trigger open={open} tabIndex={-1} paddingRight={inputPaddingRight}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <RightSection>\n                    {hasSpinner && <Spinner />}\n                    {hasClearButton && <ClearButton onClick={onClear} trimSide=\"x\" />}\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};\n"]} */"));
|
|
74
75
|
|
|
75
76
|
const EmptyState = () => ___EmotionJSX(Box, {
|
|
76
77
|
paddingY: "s",
|
|
@@ -78,13 +79,15 @@ const EmptyState = () => ___EmotionJSX(Box, {
|
|
|
78
79
|
color: "content2"
|
|
79
80
|
}, "No results");
|
|
80
81
|
|
|
81
|
-
const
|
|
82
|
+
const rightSectionGap = 'var(--grn-space-m)';
|
|
83
|
+
|
|
84
|
+
const RightSection = _ref9 => {
|
|
82
85
|
let {
|
|
83
86
|
children
|
|
84
87
|
} = _ref9;
|
|
85
88
|
return ___EmotionJSX(Arrange, {
|
|
86
89
|
right: "var(--grn-field-paddingX)",
|
|
87
|
-
gap:
|
|
90
|
+
gap: rightSectionGap,
|
|
88
91
|
position: "absolute",
|
|
89
92
|
top: "0px",
|
|
90
93
|
bottom: "0px",
|
|
@@ -96,16 +99,17 @@ const ActionsSection = _ref9 => {
|
|
|
96
99
|
const ClearButton = _ref10 => {
|
|
97
100
|
let {
|
|
98
101
|
onClick
|
|
99
|
-
} = _ref10
|
|
102
|
+
} = _ref10,
|
|
103
|
+
props = _objectWithoutProperties(_ref10, _excluded);
|
|
100
104
|
|
|
101
105
|
const handleClear = e => {
|
|
102
106
|
e.stopPropagation();
|
|
103
107
|
onClick();
|
|
104
108
|
};
|
|
105
109
|
|
|
106
|
-
return ___EmotionJSX(FieldClearButton, {
|
|
110
|
+
return ___EmotionJSX(FieldClearButton, _extends({
|
|
107
111
|
onClick: e => handleClear(e)
|
|
108
|
-
});
|
|
112
|
+
}, props));
|
|
109
113
|
};
|
|
110
114
|
|
|
111
115
|
const getFilteredOptions = (query, options, searchField) => {
|
|
@@ -126,7 +130,7 @@ const Item = /*#__PURE__*/forwardRef((_ref11, ref) => {
|
|
|
126
130
|
hasEllipsis,
|
|
127
131
|
paddingLeft = '12px'
|
|
128
132
|
} = _ref11,
|
|
129
|
-
props = _objectWithoutProperties(_ref11,
|
|
133
|
+
props = _objectWithoutProperties(_ref11, _excluded2);
|
|
130
134
|
|
|
131
135
|
return ___EmotionJSX("li", _extends({
|
|
132
136
|
className: "autocompleteMenuItem",
|
|
@@ -154,21 +158,26 @@ const Trigger = /*#__PURE__*/_styled("button", process.env.NODE_ENV === "product
|
|
|
154
158
|
open
|
|
155
159
|
} = _ref13;
|
|
156
160
|
return open ? 'none' : 'auto';
|
|
157
|
-
}, ";position:absolute;top:0;left:0;transition:opacity ", transitionSpeed, ";.autocompleteMenuItem{padding-right:34px;}.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":"AAuM6B","file":"../../src/components/autocomplete2.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useState } from 'react';\nimport { Icon, Box, Arrange } from '.';\nimport { IconChevronDown, IconPlus } 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 ActionsSection = ({ children }) => (\n  <Arrange\n    right=\"var(--grn-field-paddingX)\"\n    gap=\"xs\"\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 }) => {\n  const handleClear = e => {\n    e.stopPropagation();\n    onClick();\n  };\n  return <FieldClearButton onClick={e => handleClear(e)} />;\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: var(--grn-textbox-height-m);\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: 34px;\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-left: var(--grn-field-paddingX);\n  padding-right: ${({ paddingRight }) => paddingRight};\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  ...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 chevronWidth = '12px';\n  const chevronGap = getSpace('s');\n  const paddingRightWithoutClear = `var(--grn-field-paddingX) + ${chevronWidth} + ${chevronGap}`;\n  const inputPaddingRight = hasClearButton\n    ? `calc(${paddingRightWithoutClear} + var(--grn-clearbutton-height))`\n    : `calc(${paddingRightWithoutClear})`;\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                  />\n\n                  <Trigger open={open} tabIndex={-1}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <ActionsSection>\n                    {hasClearButton && <ClearButton onClick={onClear} />}\n                    <Icon icon={<IconChevronDown />} />\n                  </ActionsSection>\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};\n"]} */"));
|
|
161
|
+
}, ";position:absolute;top:0;left:0;transition:opacity ", transitionSpeed, ";.autocompleteMenuItem{padding-right:", _ref14 => {
|
|
162
|
+
let {
|
|
163
|
+
paddingRight
|
|
164
|
+
} = _ref14;
|
|
165
|
+
return paddingRight;
|
|
166
|
+
}, ";}.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 }) => (\n  <Arrange\n    right=\"var(--grn-field-paddingX)\"\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: var(--grn-textbox-height-m);\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-left: var(--grn-field-paddingX);\n  padding-right: ${({ paddingRight }) => paddingRight};\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  ...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 iconChevronDownWidth = `${IconChevronDown.viewBoxWidth / 2}px`;\n  const chevronRightSpace = 'var(--grn-field-paddingX)';\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(iconChevronDownWidth, chevronRightSpace);\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                  />\n\n                  <Trigger open={open} tabIndex={-1} paddingRight={inputPaddingRight}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <RightSection>\n                    {hasSpinner && <Spinner />}\n                    {hasClearButton && <ClearButton onClick={onClear} trimSide=\"x\" />}\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};\n"]} */"));
|
|
158
167
|
|
|
159
168
|
const TriggerText = /*#__PURE__*/_styled("span", process.env.NODE_ENV === "production" ? {
|
|
160
169
|
target: "ea1qwwn0"
|
|
161
170
|
} : {
|
|
162
171
|
target: "ea1qwwn0",
|
|
163
172
|
label: "TriggerText"
|
|
164
|
-
})("line-height:20px;display:block;padding-block:0;padding-left:var(--grn-field-paddingX);padding-right:",
|
|
173
|
+
})("line-height:20px;display:block;padding-block:0;padding-left:var(--grn-field-paddingX);padding-right:", _ref15 => {
|
|
165
174
|
let {
|
|
166
175
|
paddingRight
|
|
167
|
-
} =
|
|
176
|
+
} = _ref15;
|
|
168
177
|
return paddingRight;
|
|
169
|
-
}, ";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":"AAqO+B","file":"../../src/components/autocomplete2.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useState } from 'react';\nimport { Icon, Box, Arrange } from '.';\nimport { IconChevronDown, IconPlus } 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 ActionsSection = ({ children }) => (\n  <Arrange\n    right=\"var(--grn-field-paddingX)\"\n    gap=\"xs\"\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 }) => {\n  const handleClear = e => {\n    e.stopPropagation();\n    onClick();\n  };\n  return <FieldClearButton onClick={e => handleClear(e)} />;\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: var(--grn-textbox-height-m);\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: 34px;\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-left: var(--grn-field-paddingX);\n  padding-right: ${({ paddingRight }) => paddingRight};\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  ...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 chevronWidth = '12px';\n  const chevronGap = getSpace('s');\n  const paddingRightWithoutClear = `var(--grn-field-paddingX) + ${chevronWidth} + ${chevronGap}`;\n  const inputPaddingRight = hasClearButton\n    ? `calc(${paddingRightWithoutClear} + var(--grn-clearbutton-height))`\n    : `calc(${paddingRightWithoutClear})`;\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                  />\n\n                  <Trigger open={open} tabIndex={-1}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <ActionsSection>\n                    {hasClearButton && <ClearButton onClick={onClear} />}\n                    <Icon icon={<IconChevronDown />} />\n                  </ActionsSection>\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};\n"]} */"));
|
|
178
|
+
}, ";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 }) => (\n  <Arrange\n    right=\"var(--grn-field-paddingX)\"\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: var(--grn-textbox-height-m);\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-left: var(--grn-field-paddingX);\n  padding-right: ${({ paddingRight }) => paddingRight};\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  ...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 iconChevronDownWidth = `${IconChevronDown.viewBoxWidth / 2}px`;\n  const chevronRightSpace = 'var(--grn-field-paddingX)';\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(iconChevronDownWidth, chevronRightSpace);\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                  />\n\n                  <Trigger open={open} tabIndex={-1} paddingRight={inputPaddingRight}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <RightSection>\n                    {hasSpinner && <Spinner />}\n                    {hasClearButton && <ClearButton onClick={onClear} trimSide=\"x\" />}\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};\n"]} */"));
|
|
170
179
|
|
|
171
|
-
export const Autocomplete2 =
|
|
180
|
+
export const Autocomplete2 = _ref16 => {
|
|
172
181
|
let {
|
|
173
182
|
options,
|
|
174
183
|
value,
|
|
@@ -189,9 +198,10 @@ export const Autocomplete2 = _ref15 => {
|
|
|
189
198
|
isDisabled,
|
|
190
199
|
searchField = 'content',
|
|
191
200
|
backgroundColor,
|
|
192
|
-
onClear
|
|
193
|
-
|
|
194
|
-
|
|
201
|
+
onClear,
|
|
202
|
+
hasSpinner
|
|
203
|
+
} = _ref16,
|
|
204
|
+
props = _objectWithoutProperties(_ref16, _excluded3);
|
|
195
205
|
|
|
196
206
|
const [query, setQuery] = useState('');
|
|
197
207
|
|
|
@@ -235,10 +245,17 @@ export const Autocomplete2 = _ref15 => {
|
|
|
235
245
|
});
|
|
236
246
|
const OptionsRoot = hasPortal ? FloatingPortal : Fragment;
|
|
237
247
|
const hasClearButton = onClear && Boolean(selectedOption);
|
|
238
|
-
const
|
|
239
|
-
const
|
|
240
|
-
const
|
|
241
|
-
const
|
|
248
|
+
const iconChevronDownWidth = "".concat(IconChevronDown.viewBoxWidth / 2, "px");
|
|
249
|
+
const chevronRightSpace = 'var(--grn-field-paddingX)';
|
|
250
|
+
const textGap = 'var(--grn-space-s2)';
|
|
251
|
+
const spinnerSpace = "calc(var(--grn-unit) * 3) + ".concat(rightSectionGap);
|
|
252
|
+
const iconCrossWidth = "".concat(IconCross.viewBoxWidth / 2, "px");
|
|
253
|
+
const clearButtonSpace = "".concat(iconCrossWidth, " + ").concat(rightSectionGap);
|
|
254
|
+
const paddingRightValues = [textGap];
|
|
255
|
+
if (hasSpinner) paddingRightValues.push(spinnerSpace);
|
|
256
|
+
if (hasClearButton) paddingRightValues.push(clearButtonSpace);
|
|
257
|
+
paddingRightValues.push(iconChevronDownWidth, chevronRightSpace);
|
|
258
|
+
const inputPaddingRight = "calc(".concat(paddingRightValues.join(' + '), ")");
|
|
242
259
|
return ___EmotionJSX(React.Fragment, null, ___EmotionJSX(Global, {
|
|
243
260
|
styles: menuItemStyles
|
|
244
261
|
}), ___EmotionJSX(Combobox, _extends({
|
|
@@ -246,10 +263,10 @@ export const Autocomplete2 = _ref15 => {
|
|
|
246
263
|
as: "div",
|
|
247
264
|
value: selectedOption || '',
|
|
248
265
|
onChange: handleChange
|
|
249
|
-
}, props),
|
|
266
|
+
}, props), _ref17 => {
|
|
250
267
|
let {
|
|
251
268
|
open
|
|
252
|
-
} =
|
|
269
|
+
} = _ref17;
|
|
253
270
|
const hasItem = selectedOption === null || selectedOption === void 0 ? void 0 : selectedOption.item;
|
|
254
271
|
return ___EmotionJSX(React.Fragment, null, label && ___EmotionJSX(Combobox.Label, {
|
|
255
272
|
as: FieldLabel
|
|
@@ -274,15 +291,17 @@ export const Autocomplete2 = _ref15 => {
|
|
|
274
291
|
className: "autocompleteInput"
|
|
275
292
|
}), ___EmotionJSX(Trigger, {
|
|
276
293
|
open: open,
|
|
277
|
-
tabIndex: -1
|
|
294
|
+
tabIndex: -1,
|
|
295
|
+
paddingRight: inputPaddingRight
|
|
278
296
|
}, hasItem && selectedOption.item({
|
|
279
297
|
isActive: false,
|
|
280
298
|
isSelected: false,
|
|
281
299
|
isDisabled: false
|
|
282
300
|
}), !hasItem && selectedOption && ___EmotionJSX(TriggerText, {
|
|
283
301
|
paddingRight: inputPaddingRight
|
|
284
|
-
}, selectedOption === null || selectedOption === void 0 ? void 0 : selectedOption.content)), ___EmotionJSX(
|
|
285
|
-
onClick: onClear
|
|
302
|
+
}, selectedOption === null || selectedOption === void 0 ? void 0 : selectedOption.content)), ___EmotionJSX(RightSection, null, hasSpinner && ___EmotionJSX(Spinner, null), hasClearButton && ___EmotionJSX(ClearButton, {
|
|
303
|
+
onClick: onClear,
|
|
304
|
+
trimSide: "x"
|
|
286
305
|
}), ___EmotionJSX(Icon, {
|
|
287
306
|
icon: ___EmotionJSX(IconChevronDown, null)
|
|
288
307
|
}))), ___EmotionJSX(OptionsRoot, null, ___EmotionJSX(MenuCardTransition, {
|
|
@@ -306,10 +325,10 @@ export const Autocomplete2 = _ref15 => {
|
|
|
306
325
|
value: option,
|
|
307
326
|
as: Fragment,
|
|
308
327
|
disabled: option.isDisabled
|
|
309
|
-
},
|
|
328
|
+
}, _ref18 => {
|
|
310
329
|
let {
|
|
311
330
|
active
|
|
312
|
-
} =
|
|
331
|
+
} = _ref18;
|
|
313
332
|
const isDisabled = option.isDisabled;
|
|
314
333
|
const isSelected = option.value === value;
|
|
315
334
|
const isActive = !option.isDisabled && active;
|
|
@@ -331,10 +350,10 @@ export const Autocomplete2 = _ref15 => {
|
|
|
331
350
|
}), isShowCreateOption && ___EmotionJSX(React.Fragment, null, filteredOptions.length > 0 && ___EmotionJSX(MenuItemDivider, null), ___EmotionJSX(Combobox.Option, {
|
|
332
351
|
value: newOption,
|
|
333
352
|
as: Fragment
|
|
334
|
-
},
|
|
353
|
+
}, _ref19 => {
|
|
335
354
|
let {
|
|
336
355
|
active
|
|
337
|
-
} =
|
|
356
|
+
} = _ref19;
|
|
338
357
|
return ___EmotionJSX(Item, {
|
|
339
358
|
isActive: active,
|
|
340
359
|
hasEllipsis: menuItemsHaveEllipsis
|
|
@@ -373,5 +392,6 @@ Autocomplete2.propTypes = {
|
|
|
373
392
|
errorMessage: PropTypes.string,
|
|
374
393
|
isDisabled: PropTypes.bool,
|
|
375
394
|
backgroundColor: types.color,
|
|
376
|
-
onClear: PropTypes.func
|
|
395
|
+
onClear: PropTypes.func,
|
|
396
|
+
hasSpinner: PropTypes.bool
|
|
377
397
|
};
|
|
@@ -3,14 +3,12 @@ const _excluded = ["children", "isDisabled", "icon", "color", "variant", "type",
|
|
|
3
3
|
|
|
4
4
|
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); }
|
|
5
5
|
|
|
6
|
-
import "core-js/modules/web.dom-collections.iterator.js";
|
|
7
|
-
|
|
8
6
|
function _objectWithoutProperties(source, excluded) { if (source == null) return {}; var target = _objectWithoutPropertiesLoose(source, excluded); var key, i; if (Object.getOwnPropertySymbols) { var sourceSymbolKeys = Object.getOwnPropertySymbols(source); for (i = 0; i < sourceSymbolKeys.length; i++) { key = sourceSymbolKeys[i]; if (excluded.indexOf(key) >= 0) continue; if (!Object.prototype.propertyIsEnumerable.call(source, key)) continue; target[key] = source[key]; } } return target; }
|
|
9
7
|
|
|
10
8
|
function _objectWithoutPropertiesLoose(source, excluded) { if (source == null) return {}; var target = {}; var sourceKeys = Object.keys(source); var key, i; for (i = 0; i < sourceKeys.length; i++) { key = sourceKeys[i]; if (excluded.indexOf(key) >= 0) continue; target[key] = source[key]; } return target; }
|
|
11
9
|
|
|
12
10
|
import { Icon } from './icon';
|
|
13
|
-
import React, { forwardRef
|
|
11
|
+
import React, { forwardRef } from 'react';
|
|
14
12
|
import PropTypes from 'prop-types';
|
|
15
13
|
import { types } from '../types';
|
|
16
14
|
import { getClearButtonStyles } from '../foundational';
|
|
@@ -42,9 +40,11 @@ const Wrapper = /*#__PURE__*/_styled("button", process.env.NODE_ENV === "product
|
|
|
42
40
|
iconWidth,
|
|
43
41
|
trimSide
|
|
44
42
|
});
|
|
45
|
-
}, ";.Icon{", p => p.clearButtonStyles.icon, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
43
|
+
}, ";.Icon{", p => p.clearButtonStyles.icon, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb21wb25lbnRzL2ljb24tYnV0dG9uLmpzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFpQjZCIiwiZmlsZSI6Ii4uLy4uL3NyYy9jb21wb25lbnRzL2ljb24tYnV0dG9uLmpzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEljb24gfSBmcm9tICcuL2ljb24nO1xuaW1wb3J0IFJlYWN0LCB7IGZvcndhcmRSZWYgfSBmcm9tICdyZWFjdCc7XG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCc7XG5pbXBvcnQgUHJvcFR5cGVzIGZyb20gJ3Byb3AtdHlwZXMnO1xuaW1wb3J0IHsgdHlwZXMgfSBmcm9tICcuLi90eXBlcyc7XG5pbXBvcnQgeyBnZXRDbGVhckJ1dHRvblN0eWxlcyB9IGZyb20gJy4uL2ZvdW5kYXRpb25hbCc7XG5cbmNvbnN0IGdldFRyaW1TaWRlU3R5bGVzID0gKHsgaWNvbldpZHRoLCB0cmltU2lkZSB9KSA9PiB7XG4gIGNvbnN0IG9mZnNldCA9IGBjYWxjKCh2YXIoLS1ncm4tY2xlYXJidXR0b24taGVpZ2h0KSAtICR7aWNvbldpZHRofXB4KSAvIC0yKWA7XG5cbiAgaWYgKCF0cmltU2lkZSkgcmV0dXJuICcnO1xuXG4gIGlmICh0cmltU2lkZSA9PT0gJ2xlZnQnKSByZXR1cm4gYG1hcmdpbi1sZWZ0OiAke29mZnNldH07YDtcbiAgaWYgKHRyaW1TaWRlID09PSAncmlnaHQnKSByZXR1cm4gYG1hcmdpbi1yaWdodDogJHtvZmZzZXR9O2A7XG4gIGlmICh0cmltU2lkZSA9PT0gJ3gnKSByZXR1cm4gYG1hcmdpbi1pbmxpbmU6ICR7b2Zmc2V0fTtgO1xufTtcblxuY29uc3QgV3JhcHBlciA9IHN0eWxlZC5idXR0b25gXG4gICR7cCA9PiBwLmNsZWFyQnV0dG9uU3R5bGVzLmJveH07XG4gIG1pbi13aWR0aDogdmFyKC0tZ3JuLWNsZWFyYnV0dG9uLWhlaWdodCk7XG4gIHBhZGRpbmctbGVmdDogOHB4O1xuICBwYWRkaW5nLXJpZ2h0OiA4cHg7XG4gICR7KHsgdHJpbVNpZGUsIGljb25XaWR0aCB9KSA9PiB0cmltU2lkZSAmJiBnZXRUcmltU2lkZVN0eWxlcyh7IGljb25XaWR0aCwgdHJpbVNpZGUgfSl9O1xuXG4gIC5JY29uIHtcbiAgICAke3AgPT4gcC5jbGVhckJ1dHRvblN0eWxlcy5pY29ufTtcbiAgfVxuYDtcblxuZXhwb3J0IGNvbnN0IEljb25CdXR0b24gPSBmb3J3YXJkUmVmKFxuICAoXG4gICAge1xuICAgICAgY2hpbGRyZW4sXG4gICAgICBpc0Rpc2FibGVkLFxuICAgICAgaWNvbixcbiAgICAgIGNvbG9yLFxuICAgICAgdmFyaWFudCA9ICduZXV0cmFsJyxcbiAgICAgIHR5cGUgPSAnYnV0dG9uJyxcbiAgICAgIGlzQWN0aXZlLFxuICAgICAgdGFnID0gJ2J1dHRvbicsXG4gICAgICB0cmltU2lkZSxcbiAgICAgIC4uLnByb3BzXG4gICAgfSxcbiAgICByZWYsXG4gICkgPT4ge1xuICAgIGNvbnN0IGNsZWFyQnV0dG9uU3R5bGVzID0gZ2V0Q2xlYXJCdXR0b25TdHlsZXMoeyB2YXJpYW50LCBpc0Rpc2FibGVkLCBpc0FjdGl2ZSB9KTtcbiAgICBjb25zdCBpY29uV2lkdGggPSBpY29uPy50eXBlPy52aWV3Qm94V2lkdGggLyAyIHx8IDE2O1xuXG4gICAgcmV0dXJuIChcbiAgICAgIDxXcmFwcGVyXG4gICAgICAgIHR5cGU9e3R5cGV9XG4gICAgICAgIGRpc2FibGVkPXtpc0Rpc2FibGVkfVxuICAgICAgICBpY29uPXtpY29ufVxuICAgICAgICByZWY9e3JlZn1cbiAgICAgICAgdmFyaWFudD17dmFyaWFudH1cbiAgICAgICAgY2xlYXJCdXR0b25TdHlsZXM9e2NsZWFyQnV0dG9uU3R5bGVzfVxuICAgICAgICBhcz17dGFnfVxuICAgICAgICBpY29uV2lkdGg9e2ljb25XaWR0aH1cbiAgICAgICAgdHJpbVNpZGU9e3RyaW1TaWRlfVxuICAgICAgICB7Li4ucHJvcHN9XG4gICAgICA+XG4gICAgICAgIHtpY29uICYmIDxJY29uIGljb249e2ljb259IC8+fVxuICAgICAgICB7Y2hpbGRyZW4gJiYgY2hpbGRyZW59XG4gICAgICA8L1dyYXBwZXI+XG4gICAgKTtcbiAgfSxcbik7XG5cbkljb25CdXR0b24ucHJvcFR5cGVzID0ge1xuICBjaGlsZHJlbjogUHJvcFR5cGVzLm5vZGUsXG4gIGljb246IFByb3BUeXBlcy5ub2RlLFxuICBvbkNsaWNrOiBQcm9wVHlwZXMuZnVuYyxcbiAgaXNEaXNhYmxlZDogUHJvcFR5cGVzLmJvb2wsXG4gIGlzQWN0aXZlOiBQcm9wVHlwZXMuYm9vbCxcbiAgdmFyaWFudDogdHlwZXMuY2xlYXJCdXR0b25WYXJpYW50LFxuICB0eXBlOiB0eXBlcy5idXR0b25UeXBlLFxuICB0YWc6IHR5cGVzLmJ1dHRvblRhZyxcbiAgdHJpbVNpZGU6IFByb3BUeXBlcy5vbmVPZihbJ2xlZnQnLCAncmlnaHQnLCAneCddKSxcbn07XG4iXX0= */"));
|
|
46
44
|
|
|
47
45
|
export const IconButton = /*#__PURE__*/forwardRef((_ref3, ref) => {
|
|
46
|
+
var _icon$type;
|
|
47
|
+
|
|
48
48
|
let {
|
|
49
49
|
children,
|
|
50
50
|
isDisabled,
|
|
@@ -58,18 +58,12 @@ export const IconButton = /*#__PURE__*/forwardRef((_ref3, ref) => {
|
|
|
58
58
|
} = _ref3,
|
|
59
59
|
props = _objectWithoutProperties(_ref3, _excluded);
|
|
60
60
|
|
|
61
|
-
const [iconWidth, setIconWidth] = React.useState(18);
|
|
62
61
|
const clearButtonStyles = getClearButtonStyles({
|
|
63
62
|
variant,
|
|
64
63
|
isDisabled,
|
|
65
64
|
isActive
|
|
66
65
|
});
|
|
67
|
-
const
|
|
68
|
-
useEffect(() => {
|
|
69
|
-
if (trimSide && iconRef.current) {
|
|
70
|
-
setIconWidth(iconRef.current.getBoundingClientRect().width);
|
|
71
|
-
}
|
|
72
|
-
}, []);
|
|
66
|
+
const iconWidth = (icon === null || icon === void 0 ? void 0 : (_icon$type = icon.type) === null || _icon$type === void 0 ? void 0 : _icon$type.viewBoxWidth) / 2 || 16;
|
|
73
67
|
return ___EmotionJSX(Wrapper, _extends({
|
|
74
68
|
type: type,
|
|
75
69
|
disabled: isDisabled,
|
|
@@ -81,8 +75,7 @@ export const IconButton = /*#__PURE__*/forwardRef((_ref3, ref) => {
|
|
|
81
75
|
iconWidth: iconWidth,
|
|
82
76
|
trimSide: trimSide
|
|
83
77
|
}, props), icon && ___EmotionJSX(Icon, {
|
|
84
|
-
icon: icon
|
|
85
|
-
ref: iconRef
|
|
78
|
+
icon: icon
|
|
86
79
|
}), children && children);
|
|
87
80
|
});
|
|
88
81
|
IconButton.propTypes = {
|