@flodesk/grain 11.46.1 → 11.47.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/es/components/autocomplete2.js +51 -33
- package/es/components/button.js +31 -13
- package/es/components/select.js +48 -31
- package/es/components/text-input.js +27 -15
- package/es/foundational/field.js +30 -21
- package/es/styles/foundational.js +3 -3
- package/es/types/components/link.d.ts +1 -1
- package/package.json +1 -1
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import _styled from "@emotion/styled/base";
|
|
2
2
|
const _excluded = ["onClick"],
|
|
3
3
|
_excluded2 = ["children", "hasEllipsis", "paddingLeft"],
|
|
4
|
-
_excluded3 = ["options", "value", "onChange", "isCreatable", "onCreate", "menuPlacement", "menuWidth", "menuMaxHeight", "menuZIndex", "placeholder", "label", "hint", "menuItemsHaveEllipsis", "hasPortal", "hasError", "errorMessage", "isDisabled", "searchField", "backgroundColor", "onClear", "hasSpinner"];
|
|
4
|
+
_excluded3 = ["options", "value", "onChange", "isCreatable", "onCreate", "menuPlacement", "menuWidth", "menuMaxHeight", "menuZIndex", "placeholder", "label", "hint", "menuItemsHaveEllipsis", "hasPortal", "hasError", "errorMessage", "isDisabled", "searchField", "backgroundColor", "onClear", "hasSpinner", "size"];
|
|
5
5
|
import "core-js/modules/web.dom-collections.iterator.js";
|
|
6
6
|
|
|
7
7
|
function _extends() { _extends = Object.assign ? Object.assign.bind() : function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); }
|
|
@@ -71,7 +71,7 @@ const Root = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "production" ?
|
|
|
71
71
|
open
|
|
72
72
|
} = _ref8;
|
|
73
73
|
return open ? 'auto' : 'none';
|
|
74
|
-
}, ";transition:opacity ", transitionSpeed, ";&:focus-visible,&:placeholder-shown{opacity:1;pointer-events:auto;transition:opacity ", transitionSpeed, ";}&:disabled:placeholder-shown{background:transparent;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/autocomplete2.jsx"],"names":[],"mappings":"AAwBuB","file":"../../src/components/autocomplete2.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useState } from 'react';\nimport { Icon, Box, Arrange, Spinner } from '.';\nimport { IconChevronDown, IconPlus, IconCross } from '../icons';\nimport { Combobox } from '@headlessui/react';\nimport { defaultProps, types } from '../types';\nimport {\n  InputField,\n  MenuCard,\n  MenuCardTransition,\n  MenuGroupTitle,\n  FieldLabel,\n  FieldHint,\n  styles,\n  FieldClearButton,\n} from '../foundational';\nimport { MenuItemDivider, useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\nimport { Global } from '@emotion/react';\nimport { getColor, getIconSize, getRadius, getSpace, getTransition } from '../utilities';\nimport styled from '@emotion/styled';\n\nconst transitionSpeed = getTransition('xxFast');\n\nconst Root = styled.div`\n  ${styles.transitions};\n  ${({ hasError, isDisabled }) =>\n    !hasError &&\n    !isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color);\n      &:hover { --borderColor: var(--grn-field-border-color-hover) };\n      &:focus-within { --borderColor: var(--grn-field-border-color-focus) };\n    `};\n\n  ${({ hasError }) => hasError && `--borderColor: ${getColor('danger')};`};\n\n  ${({ backgroundColor, isDisabled }) =>\n    !backgroundColor &&\n    !isDisabled &&\n    `\n      --backgroundColor: transparent;\n    `};\n\n  ${({ backgroundColor }) =>\n    backgroundColor &&\n    `\n      --backgroundColor: ${getColor(backgroundColor)};\n    `};\n\n  ${({ isDisabled }) =>\n    isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color-disabled);\n      --backgroundColor: var(--grn-field-background-disabled);\n\n      * {\n        pointer-events: none;\n      }\n    `};\n\n  position: relative;\n  margin-top: ${({ marginTop }) => getSpace(marginTop)};\n  border-radius: ${getRadius('s')};\n  box-shadow: inset 0 0 0 var(--grn-border-size) var(--borderColor);\n  background-color: var(--backgroundColor);\n\n  .autocompleteInput {\n    line-height: 20px;\n    border: none;\n    opacity: ${({ open }) => (open ? 1 : 0)};\n    pointer-events: ${({ open }) => (open ? 'auto' : 'none')};\n    transition: opacity ${transitionSpeed};\n\n    &:focus-visible,\n    &:placeholder-shown {\n      opacity: 1;\n      pointer-events: auto;\n      transition: opacity ${transitionSpeed};\n    }\n    &:disabled:placeholder-shown {\n      background: transparent;\n    }\n  }\n`;\n\nconst EmptyState = () => (\n  <Box paddingY=\"s\" paddingX=\"m\" color=\"content2\">\n    No results\n  </Box>\n);\n\nconst rightSectionGap = 'var(--grn-space-m)';\n\nconst RightSection = ({ children }) => (\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
|
+
}, ";transition:opacity ", transitionSpeed, ";&:focus-visible,&:placeholder-shown{opacity:1;pointer-events:auto;transition:opacity ", transitionSpeed, ";}&:disabled:placeholder-shown{background:transparent;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/autocomplete2.jsx"],"names":[],"mappings":"AAwBuB","file":"../../src/components/autocomplete2.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useState } from 'react';\nimport { Icon, Box, Arrange, Spinner } from '.';\nimport { IconChevronDown, IconPlus, IconCross } from '../icons';\nimport { Combobox } from '@headlessui/react';\nimport { defaultProps, types } from '../types';\nimport {\n  InputField,\n  MenuCard,\n  MenuCardTransition,\n  MenuGroupTitle,\n  FieldLabel,\n  FieldHint,\n  styles,\n  FieldClearButton,\n} from '../foundational';\nimport { MenuItemDivider, useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\nimport { Global } from '@emotion/react';\nimport { getColor, getIconSize, getRadius, getSpace, getTransition } from '../utilities';\nimport styled from '@emotion/styled';\n\nconst transitionSpeed = getTransition('xxFast');\n\nconst Root = styled.div`\n  ${styles.transitions};\n  ${({ hasError, isDisabled }) =>\n    !hasError &&\n    !isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color);\n      &:hover { --borderColor: var(--grn-field-border-color-hover) };\n      &:focus-within { --borderColor: var(--grn-field-border-color-focus) };\n    `};\n\n  ${({ hasError }) => hasError && `--borderColor: ${getColor('danger')};`};\n\n  ${({ backgroundColor, isDisabled }) =>\n    !backgroundColor &&\n    !isDisabled &&\n    `\n      --backgroundColor: transparent;\n    `};\n\n  ${({ backgroundColor }) =>\n    backgroundColor &&\n    `\n      --backgroundColor: ${getColor(backgroundColor)};\n    `};\n\n  ${({ isDisabled }) =>\n    isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color-disabled);\n      --backgroundColor: var(--grn-field-background-disabled);\n\n      * {\n        pointer-events: none;\n      }\n    `};\n\n  position: relative;\n  margin-top: ${({ marginTop }) => getSpace(marginTop)};\n  border-radius: ${getRadius('s')};\n  box-shadow: inset 0 0 0 var(--grn-border-size) var(--borderColor);\n  background-color: var(--backgroundColor);\n\n  .autocompleteInput {\n    line-height: 20px;\n    border: none;\n    opacity: ${({ open }) => (open ? 1 : 0)};\n    pointer-events: ${({ open }) => (open ? 'auto' : 'none')};\n    transition: opacity ${transitionSpeed};\n\n    &:focus-visible,\n    &:placeholder-shown {\n      opacity: 1;\n      pointer-events: auto;\n      transition: opacity ${transitionSpeed};\n    }\n    &:disabled:placeholder-shown {\n      background: transparent;\n    }\n  }\n`;\n\nconst EmptyState = () => (\n  <Box paddingY=\"s\" paddingX=\"m\" color=\"content2\">\n    No results\n  </Box>\n);\n\nconst rightSectionGap = 'var(--grn-space-m)';\n\nconst RightSection = ({ children, right }) => (\n  <Arrange\n    right={right}\n    gap={rightSectionGap}\n    position=\"absolute\"\n    top=\"0px\"\n    bottom=\"0px\"\n    margin=\"auto\"\n    height=\"fit-content\"\n  >\n    {children}\n  </Arrange>\n);\n\nconst ClearButton = ({ onClick, ...props }) => {\n  const handleClear = e => {\n    e.stopPropagation();\n    onClick();\n  };\n  return <FieldClearButton onClick={e => handleClear(e)} {...props} />;\n};\n\nconst getFilteredOptions = (query, options, searchField) => {\n  if (query === '') return options;\n\n  return options.filter(option => option[searchField].toLowerCase().includes(query.toLowerCase()));\n};\n\nconst getShowGroupTitle = (index, option, filteredOptions) => {\n  const firstOptionHasTitle = index === 0 && option.groupTitle;\n  const titleChanged = index > 0 && option.groupTitle !== filteredOptions[index - 1].groupTitle;\n\n  return firstOptionHasTitle || titleChanged;\n};\n\nconst menuItemStyles = `\n  .autocompleteMenuItem {\n    display: flex;\n    align-items: center;\n    list-style: none;\n    min-height: var(--grn-textbox-height-m);\n    border-radius: ${getRadius('s')};\n    appearance: none;\n    padding-block: 0;\n    padding: 4px 12px;\n\n    --checkUri: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%2337312f73\"%3E%3C/path%3E%3C/svg%3E');\n    --checkUriActive: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%23241a17\"%3E%3C/path%3E%3C/svg%3E');\n\n    &[data-headlessui-state='active'],\n    &[data-headlessui-state='active selected'] {\n      background-color: ${getColor('fade1')};\n    }\n\n    &[data-headlessui-state='selected'],\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUri);\n      background-repeat: no-repeat;\n      background-position: center right 12px;\n      background-size: auto ${getIconSize('m')};\n    }\n\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUriActive);\n    }\n\n    &[data-headlessui-state='disabled'] {\n      color: ${getColor('disabledContent')};\n      cursor: default;\n    }\n\n    ul & {\n      cursor: pointer;\n      padding-right: 40px;\n    }\n  }\n\n  .autocompleteMenuItemText {\n    flex-grow: 1;\n\n    &.hasEllipsis {\n      white-space: nowrap;\n      overflow: hidden;\n      text-overflow: ellipsis;\n    }\n  }\n`;\n\nconst Item = forwardRef(({ children, hasEllipsis, paddingLeft = '12px', ...props }, ref) => {\n  return (\n    <li\n      className=\"autocompleteMenuItem\"\n      ref={ref}\n      style={{ paddingLeft: getSpace(paddingLeft) }}\n      {...props}\n    >\n      <span\n        className={\n          hasEllipsis ? 'autocompleteMenuItemText hasEllipsis' : 'autocompleteMenuItemText'\n        }\n      >\n        {children}\n      </span>\n    </li>\n  );\n});\n\nconst Trigger = styled.button`\n  text-align: left;\n  height: ${({ size }) => `var(--grn-textbox-height-${size})`};\n  background-color: transparent;\n  cursor: text;\n  appearance: none;\n  display: block;\n  font: inherit;\n  color: inherit;\n  width: 100%;\n  outline: none;\n  border: none;\n  padding: 0;\n  opacity: ${({ open }) => (open ? 0 : 1)};\n  pointer-events: ${({ open }) => (open ? 'none' : 'auto')};\n  position: absolute;\n  top: 0;\n  left: 0;\n  transition: opacity ${transitionSpeed};\n\n  .autocompleteMenuItem {\n    padding-right: ${({ paddingRight }) => paddingRight};\n  }\n\n  .autocompleteInput:focus-visible + & {\n    opacity: 0;\n    pointer-events: none;\n  }\n`;\n\nconst TriggerText = styled.span`\n  line-height: 20px;\n  display: block;\n  padding-block: 0;\n  padding-right: ${({ paddingRight }) => paddingRight};\n  padding-left: ${({ paddingLeft }) => paddingLeft};\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`;\n\nexport const Autocomplete2 = ({\n  options,\n  value,\n  onChange,\n  isCreatable,\n  onCreate,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  menuZIndex,\n  placeholder,\n  label,\n  hint,\n  menuItemsHaveEllipsis = true,\n  hasPortal = true,\n  hasError,\n  errorMessage,\n  isDisabled,\n  searchField = 'content',\n  backgroundColor,\n  onClear,\n  hasSpinner,\n  size = 'm',\n  ...props\n}) => {\n  const [query, setQuery] = useState('');\n\n  const handleCreatableChange = option => {\n    if (onCreate) {\n      const isCreatableOption = !options.find(item => item.value === option.value);\n      if (isCreatableOption) {\n        onCreate(option);\n        return;\n      }\n    }\n    onChange(option);\n  };\n\n  const handleChange = option => {\n    if (isCreatable) {\n      handleCreatableChange(option);\n    } else {\n      onChange(option);\n    }\n  };\n\n  const filteredOptions = getFilteredOptions(query, options, searchField);\n\n  const isShowCreateOption = isCreatable && query.length > 0;\n\n  const noResults = !Boolean(filteredOptions.length) && !isShowCreateOption;\n\n  const fieldMarginTop = label || hint ? 'var(--grn-form-control-label-gap)' : undefined;\n\n  const selectedOption = useMemo(\n    () => options.find(option => option.value === value),\n    [options, value],\n  );\n\n  const newOption = useMemo(() => ({ value: query, content: query }), [query]);\n\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n\n  const OptionsRoot = hasPortal ? FloatingPortal : Fragment;\n\n  const hasClearButton = onClear && Boolean(selectedOption);\n\n  const inputPaddingX = `var(--grn-field-paddingX-${size})`;\n\n  const chevronWidth = `${IconChevronDown.viewBoxWidth / 2}px`;\n  const textGap = 'var(--grn-space-s2)';\n  const spinnerSpace = `calc(var(--grn-unit) * 3) + ${rightSectionGap}`;\n  const iconCrossWidth = `${IconCross.viewBoxWidth / 2}px`;\n  const clearButtonSpace = `${iconCrossWidth} + ${rightSectionGap}`;\n\n  const paddingRightValues = [textGap];\n  if (hasSpinner) paddingRightValues.push(spinnerSpace);\n  if (hasClearButton) paddingRightValues.push(clearButtonSpace);\n  paddingRightValues.push(chevronWidth, inputPaddingX);\n\n  const inputPaddingRight = `calc(${paddingRightValues.join(' + ')})`;\n\n  return (\n    <>\n      <Global styles={menuItemStyles} />\n      <Combobox\n        disabled={isDisabled}\n        as=\"div\"\n        value={selectedOption || ''}\n        onChange={handleChange}\n        {...props}\n      >\n        {({ open }) => {\n          const hasItem = selectedOption?.item;\n          return (\n            <>\n              {label && <Combobox.Label as={FieldLabel}>{label}</Combobox.Label>}\n              {hint && <FieldHint>{hint}</FieldHint>}\n\n              <Root\n                marginTop={fieldMarginTop}\n                ref={reference}\n                open={open}\n                hasError={hasError}\n                backgroundColor={backgroundColor}\n                isDisabled={isDisabled}\n              >\n                <Combobox.Button as={Box} position=\"relative\">\n                  <Combobox.Input\n                    as={InputField}\n                    autoComplete=\"off\"\n                    onChange={event => setQuery(event.target.value)}\n                    placeholder={placeholder}\n                    displayValue={option => option && option.content}\n                    paddingRight={inputPaddingRight}\n                    hasError={hasError}\n                    className=\"autocompleteInput\"\n                    size={size}\n                  />\n\n                  <Trigger open={open} tabIndex={-1} paddingRight={inputPaddingRight} size={size}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight} paddingLeft={inputPaddingX}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <RightSection right={inputPaddingX}>\n                    {hasSpinner && <Spinner />}\n                    {hasClearButton && <ClearButton onClick={onClear} trimSides />}\n                    <Icon icon={<IconChevronDown />} />\n                  </RightSection>\n                </Combobox.Button>\n\n                <OptionsRoot>\n                  <MenuCardTransition afterLeave={() => setQuery('')}>\n                    <Combobox.Options\n                      static\n                      ref={floating}\n                      placement={menuPlacement}\n                      maxHeight={menuMaxHeight}\n                      as={MenuCard}\n                      zIndex={menuZIndex}\n                      style={floatingStyles}\n                      isOpen={open}\n                    >\n                      {filteredOptions.map((option, index) => {\n                        const showGroupTitle = getShowGroupTitle(index, option, filteredOptions);\n\n                        return (\n                          <Fragment key={index}>\n                            {showGroupTitle && (\n                              <MenuGroupTitle hasDivider={index > 0}>\n                                {option.groupTitle}\n                              </MenuGroupTitle>\n                            )}\n\n                            <Combobox.Option\n                              value={option}\n                              as={Fragment}\n                              disabled={option.isDisabled}\n                            >\n                              {({ active }) => {\n                                const isDisabled = option.isDisabled;\n                                const isSelected = option.value === value;\n                                const isActive = !option.isDisabled && active;\n                                const hasEllipsis = menuItemsHaveEllipsis;\n                                if (option.item) {\n                                  return option.item({\n                                    isActive,\n                                    isSelected,\n                                    isDisabled,\n                                    hasEllipsis,\n                                  });\n                                }\n                                return <Item hasEllipsis={hasEllipsis}>{option.content}</Item>;\n                              }}\n                            </Combobox.Option>\n                          </Fragment>\n                        );\n                      })}\n                      {isShowCreateOption && (\n                        <>\n                          {filteredOptions.length > 0 && <MenuItemDivider />}\n                          <Combobox.Option value={newOption} as={Fragment}>\n                            {({ active }) => (\n                              <Item isActive={active} hasEllipsis={menuItemsHaveEllipsis}>\n                                <Arrange gap=\"s\" tag=\"span\">\n                                  <Icon icon={<IconPlus />} color={active ? 'content' : 'icon'} />\n                                  Create {`\"` + query + `\"`}\n                                </Arrange>\n                              </Item>\n                            )}\n                          </Combobox.Option>\n                        </>\n                      )}\n                      {noResults && <EmptyState />}\n                    </Combobox.Options>\n                  </MenuCardTransition>\n                </OptionsRoot>\n              </Root>\n              {errorMessage && (\n                <Box marginTop=\"var(--grn-form-control-label-gap)\" color=\"danger\">\n                  {errorMessage}\n                </Box>\n              )}\n            </>\n          );\n        }}\n      </Combobox>\n    </>\n  );\n};\n\nAutocomplete2.Label = FieldLabel;\nAutocomplete2.Hint = FieldHint;\nAutocomplete2.Item = Item;\n\nAutocomplete2.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.string,\n  onChange: PropTypes.func.isRequired,\n  isCreatable: PropTypes.bool,\n  onCreate: PropTypes.func,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuMaxHeight: types.dimension,\n  menuZIndex: types.zIndex,\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n  placeholder: PropTypes.string,\n  hasError: PropTypes.bool,\n  errorMessage: PropTypes.string,\n  isDisabled: PropTypes.bool,\n  backgroundColor: types.color,\n  onClear: PropTypes.func,\n  hasSpinner: PropTypes.bool,\n  size: PropTypes.oneOf(['m', 'l']),\n};\n"]} */"));
|
|
75
75
|
|
|
76
76
|
const EmptyState = () => ___EmotionJSX(Box, {
|
|
77
77
|
paddingY: "s",
|
|
@@ -83,10 +83,11 @@ const rightSectionGap = 'var(--grn-space-m)';
|
|
|
83
83
|
|
|
84
84
|
const RightSection = _ref9 => {
|
|
85
85
|
let {
|
|
86
|
-
children
|
|
86
|
+
children,
|
|
87
|
+
right
|
|
87
88
|
} = _ref9;
|
|
88
89
|
return ___EmotionJSX(Arrange, {
|
|
89
|
-
right:
|
|
90
|
+
right: right,
|
|
90
91
|
gap: rightSectionGap,
|
|
91
92
|
position: "absolute",
|
|
92
93
|
top: "0px",
|
|
@@ -148,36 +149,46 @@ const Trigger = /*#__PURE__*/_styled("button", process.env.NODE_ENV === "product
|
|
|
148
149
|
} : {
|
|
149
150
|
target: "ea1qwwn1",
|
|
150
151
|
label: "Trigger"
|
|
151
|
-
})("text-align:left;height:
|
|
152
|
+
})("text-align:left;height:", _ref12 => {
|
|
152
153
|
let {
|
|
153
|
-
|
|
154
|
+
size
|
|
154
155
|
} = _ref12;
|
|
155
|
-
return
|
|
156
|
-
}, ";
|
|
156
|
+
return "var(--grn-textbox-height-".concat(size, ")");
|
|
157
|
+
}, ";background-color:transparent;cursor:text;appearance:none;display:block;font:inherit;color:inherit;width:100%;outline:none;border:none;padding:0;opacity:", _ref13 => {
|
|
157
158
|
let {
|
|
158
159
|
open
|
|
159
160
|
} = _ref13;
|
|
161
|
+
return open ? 0 : 1;
|
|
162
|
+
}, ";pointer-events:", _ref14 => {
|
|
163
|
+
let {
|
|
164
|
+
open
|
|
165
|
+
} = _ref14;
|
|
160
166
|
return open ? 'none' : 'auto';
|
|
161
|
-
}, ";position:absolute;top:0;left:0;transition:opacity ", transitionSpeed, ";.autocompleteMenuItem{padding-right:",
|
|
167
|
+
}, ";position:absolute;top:0;left:0;transition:opacity ", transitionSpeed, ";.autocompleteMenuItem{padding-right:", _ref15 => {
|
|
162
168
|
let {
|
|
163
169
|
paddingRight
|
|
164
|
-
} =
|
|
170
|
+
} = _ref15;
|
|
165
171
|
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"]} */"));
|
|
172
|
+
}, ";}.autocompleteInput:focus-visible+&{opacity:0;pointer-events:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/autocomplete2.jsx"],"names":[],"mappings":"AAyM6B","file":"../../src/components/autocomplete2.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useState } from 'react';\nimport { Icon, Box, Arrange, Spinner } from '.';\nimport { IconChevronDown, IconPlus, IconCross } from '../icons';\nimport { Combobox } from '@headlessui/react';\nimport { defaultProps, types } from '../types';\nimport {\n  InputField,\n  MenuCard,\n  MenuCardTransition,\n  MenuGroupTitle,\n  FieldLabel,\n  FieldHint,\n  styles,\n  FieldClearButton,\n} from '../foundational';\nimport { MenuItemDivider, useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\nimport { Global } from '@emotion/react';\nimport { getColor, getIconSize, getRadius, getSpace, getTransition } from '../utilities';\nimport styled from '@emotion/styled';\n\nconst transitionSpeed = getTransition('xxFast');\n\nconst Root = styled.div`\n  ${styles.transitions};\n  ${({ hasError, isDisabled }) =>\n    !hasError &&\n    !isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color);\n      &:hover { --borderColor: var(--grn-field-border-color-hover) };\n      &:focus-within { --borderColor: var(--grn-field-border-color-focus) };\n    `};\n\n  ${({ hasError }) => hasError && `--borderColor: ${getColor('danger')};`};\n\n  ${({ backgroundColor, isDisabled }) =>\n    !backgroundColor &&\n    !isDisabled &&\n    `\n      --backgroundColor: transparent;\n    `};\n\n  ${({ backgroundColor }) =>\n    backgroundColor &&\n    `\n      --backgroundColor: ${getColor(backgroundColor)};\n    `};\n\n  ${({ isDisabled }) =>\n    isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color-disabled);\n      --backgroundColor: var(--grn-field-background-disabled);\n\n      * {\n        pointer-events: none;\n      }\n    `};\n\n  position: relative;\n  margin-top: ${({ marginTop }) => getSpace(marginTop)};\n  border-radius: ${getRadius('s')};\n  box-shadow: inset 0 0 0 var(--grn-border-size) var(--borderColor);\n  background-color: var(--backgroundColor);\n\n  .autocompleteInput {\n    line-height: 20px;\n    border: none;\n    opacity: ${({ open }) => (open ? 1 : 0)};\n    pointer-events: ${({ open }) => (open ? 'auto' : 'none')};\n    transition: opacity ${transitionSpeed};\n\n    &:focus-visible,\n    &:placeholder-shown {\n      opacity: 1;\n      pointer-events: auto;\n      transition: opacity ${transitionSpeed};\n    }\n    &:disabled:placeholder-shown {\n      background: transparent;\n    }\n  }\n`;\n\nconst EmptyState = () => (\n  <Box paddingY=\"s\" paddingX=\"m\" color=\"content2\">\n    No results\n  </Box>\n);\n\nconst rightSectionGap = 'var(--grn-space-m)';\n\nconst RightSection = ({ children, right }) => (\n  <Arrange\n    right={right}\n    gap={rightSectionGap}\n    position=\"absolute\"\n    top=\"0px\"\n    bottom=\"0px\"\n    margin=\"auto\"\n    height=\"fit-content\"\n  >\n    {children}\n  </Arrange>\n);\n\nconst ClearButton = ({ onClick, ...props }) => {\n  const handleClear = e => {\n    e.stopPropagation();\n    onClick();\n  };\n  return <FieldClearButton onClick={e => handleClear(e)} {...props} />;\n};\n\nconst getFilteredOptions = (query, options, searchField) => {\n  if (query === '') return options;\n\n  return options.filter(option => option[searchField].toLowerCase().includes(query.toLowerCase()));\n};\n\nconst getShowGroupTitle = (index, option, filteredOptions) => {\n  const firstOptionHasTitle = index === 0 && option.groupTitle;\n  const titleChanged = index > 0 && option.groupTitle !== filteredOptions[index - 1].groupTitle;\n\n  return firstOptionHasTitle || titleChanged;\n};\n\nconst menuItemStyles = `\n  .autocompleteMenuItem {\n    display: flex;\n    align-items: center;\n    list-style: none;\n    min-height: var(--grn-textbox-height-m);\n    border-radius: ${getRadius('s')};\n    appearance: none;\n    padding-block: 0;\n    padding: 4px 12px;\n\n    --checkUri: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%2337312f73\"%3E%3C/path%3E%3C/svg%3E');\n    --checkUriActive: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%23241a17\"%3E%3C/path%3E%3C/svg%3E');\n\n    &[data-headlessui-state='active'],\n    &[data-headlessui-state='active selected'] {\n      background-color: ${getColor('fade1')};\n    }\n\n    &[data-headlessui-state='selected'],\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUri);\n      background-repeat: no-repeat;\n      background-position: center right 12px;\n      background-size: auto ${getIconSize('m')};\n    }\n\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUriActive);\n    }\n\n    &[data-headlessui-state='disabled'] {\n      color: ${getColor('disabledContent')};\n      cursor: default;\n    }\n\n    ul & {\n      cursor: pointer;\n      padding-right: 40px;\n    }\n  }\n\n  .autocompleteMenuItemText {\n    flex-grow: 1;\n\n    &.hasEllipsis {\n      white-space: nowrap;\n      overflow: hidden;\n      text-overflow: ellipsis;\n    }\n  }\n`;\n\nconst Item = forwardRef(({ children, hasEllipsis, paddingLeft = '12px', ...props }, ref) => {\n  return (\n    <li\n      className=\"autocompleteMenuItem\"\n      ref={ref}\n      style={{ paddingLeft: getSpace(paddingLeft) }}\n      {...props}\n    >\n      <span\n        className={\n          hasEllipsis ? 'autocompleteMenuItemText hasEllipsis' : 'autocompleteMenuItemText'\n        }\n      >\n        {children}\n      </span>\n    </li>\n  );\n});\n\nconst Trigger = styled.button`\n  text-align: left;\n  height: ${({ size }) => `var(--grn-textbox-height-${size})`};\n  background-color: transparent;\n  cursor: text;\n  appearance: none;\n  display: block;\n  font: inherit;\n  color: inherit;\n  width: 100%;\n  outline: none;\n  border: none;\n  padding: 0;\n  opacity: ${({ open }) => (open ? 0 : 1)};\n  pointer-events: ${({ open }) => (open ? 'none' : 'auto')};\n  position: absolute;\n  top: 0;\n  left: 0;\n  transition: opacity ${transitionSpeed};\n\n  .autocompleteMenuItem {\n    padding-right: ${({ paddingRight }) => paddingRight};\n  }\n\n  .autocompleteInput:focus-visible + & {\n    opacity: 0;\n    pointer-events: none;\n  }\n`;\n\nconst TriggerText = styled.span`\n  line-height: 20px;\n  display: block;\n  padding-block: 0;\n  padding-right: ${({ paddingRight }) => paddingRight};\n  padding-left: ${({ paddingLeft }) => paddingLeft};\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`;\n\nexport const Autocomplete2 = ({\n  options,\n  value,\n  onChange,\n  isCreatable,\n  onCreate,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  menuZIndex,\n  placeholder,\n  label,\n  hint,\n  menuItemsHaveEllipsis = true,\n  hasPortal = true,\n  hasError,\n  errorMessage,\n  isDisabled,\n  searchField = 'content',\n  backgroundColor,\n  onClear,\n  hasSpinner,\n  size = 'm',\n  ...props\n}) => {\n  const [query, setQuery] = useState('');\n\n  const handleCreatableChange = option => {\n    if (onCreate) {\n      const isCreatableOption = !options.find(item => item.value === option.value);\n      if (isCreatableOption) {\n        onCreate(option);\n        return;\n      }\n    }\n    onChange(option);\n  };\n\n  const handleChange = option => {\n    if (isCreatable) {\n      handleCreatableChange(option);\n    } else {\n      onChange(option);\n    }\n  };\n\n  const filteredOptions = getFilteredOptions(query, options, searchField);\n\n  const isShowCreateOption = isCreatable && query.length > 0;\n\n  const noResults = !Boolean(filteredOptions.length) && !isShowCreateOption;\n\n  const fieldMarginTop = label || hint ? 'var(--grn-form-control-label-gap)' : undefined;\n\n  const selectedOption = useMemo(\n    () => options.find(option => option.value === value),\n    [options, value],\n  );\n\n  const newOption = useMemo(() => ({ value: query, content: query }), [query]);\n\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n\n  const OptionsRoot = hasPortal ? FloatingPortal : Fragment;\n\n  const hasClearButton = onClear && Boolean(selectedOption);\n\n  const inputPaddingX = `var(--grn-field-paddingX-${size})`;\n\n  const chevronWidth = `${IconChevronDown.viewBoxWidth / 2}px`;\n  const textGap = 'var(--grn-space-s2)';\n  const spinnerSpace = `calc(var(--grn-unit) * 3) + ${rightSectionGap}`;\n  const iconCrossWidth = `${IconCross.viewBoxWidth / 2}px`;\n  const clearButtonSpace = `${iconCrossWidth} + ${rightSectionGap}`;\n\n  const paddingRightValues = [textGap];\n  if (hasSpinner) paddingRightValues.push(spinnerSpace);\n  if (hasClearButton) paddingRightValues.push(clearButtonSpace);\n  paddingRightValues.push(chevronWidth, inputPaddingX);\n\n  const inputPaddingRight = `calc(${paddingRightValues.join(' + ')})`;\n\n  return (\n    <>\n      <Global styles={menuItemStyles} />\n      <Combobox\n        disabled={isDisabled}\n        as=\"div\"\n        value={selectedOption || ''}\n        onChange={handleChange}\n        {...props}\n      >\n        {({ open }) => {\n          const hasItem = selectedOption?.item;\n          return (\n            <>\n              {label && <Combobox.Label as={FieldLabel}>{label}</Combobox.Label>}\n              {hint && <FieldHint>{hint}</FieldHint>}\n\n              <Root\n                marginTop={fieldMarginTop}\n                ref={reference}\n                open={open}\n                hasError={hasError}\n                backgroundColor={backgroundColor}\n                isDisabled={isDisabled}\n              >\n                <Combobox.Button as={Box} position=\"relative\">\n                  <Combobox.Input\n                    as={InputField}\n                    autoComplete=\"off\"\n                    onChange={event => setQuery(event.target.value)}\n                    placeholder={placeholder}\n                    displayValue={option => option && option.content}\n                    paddingRight={inputPaddingRight}\n                    hasError={hasError}\n                    className=\"autocompleteInput\"\n                    size={size}\n                  />\n\n                  <Trigger open={open} tabIndex={-1} paddingRight={inputPaddingRight} size={size}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight} paddingLeft={inputPaddingX}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <RightSection right={inputPaddingX}>\n                    {hasSpinner && <Spinner />}\n                    {hasClearButton && <ClearButton onClick={onClear} trimSides />}\n                    <Icon icon={<IconChevronDown />} />\n                  </RightSection>\n                </Combobox.Button>\n\n                <OptionsRoot>\n                  <MenuCardTransition afterLeave={() => setQuery('')}>\n                    <Combobox.Options\n                      static\n                      ref={floating}\n                      placement={menuPlacement}\n                      maxHeight={menuMaxHeight}\n                      as={MenuCard}\n                      zIndex={menuZIndex}\n                      style={floatingStyles}\n                      isOpen={open}\n                    >\n                      {filteredOptions.map((option, index) => {\n                        const showGroupTitle = getShowGroupTitle(index, option, filteredOptions);\n\n                        return (\n                          <Fragment key={index}>\n                            {showGroupTitle && (\n                              <MenuGroupTitle hasDivider={index > 0}>\n                                {option.groupTitle}\n                              </MenuGroupTitle>\n                            )}\n\n                            <Combobox.Option\n                              value={option}\n                              as={Fragment}\n                              disabled={option.isDisabled}\n                            >\n                              {({ active }) => {\n                                const isDisabled = option.isDisabled;\n                                const isSelected = option.value === value;\n                                const isActive = !option.isDisabled && active;\n                                const hasEllipsis = menuItemsHaveEllipsis;\n                                if (option.item) {\n                                  return option.item({\n                                    isActive,\n                                    isSelected,\n                                    isDisabled,\n                                    hasEllipsis,\n                                  });\n                                }\n                                return <Item hasEllipsis={hasEllipsis}>{option.content}</Item>;\n                              }}\n                            </Combobox.Option>\n                          </Fragment>\n                        );\n                      })}\n                      {isShowCreateOption && (\n                        <>\n                          {filteredOptions.length > 0 && <MenuItemDivider />}\n                          <Combobox.Option value={newOption} as={Fragment}>\n                            {({ active }) => (\n                              <Item isActive={active} hasEllipsis={menuItemsHaveEllipsis}>\n                                <Arrange gap=\"s\" tag=\"span\">\n                                  <Icon icon={<IconPlus />} color={active ? 'content' : 'icon'} />\n                                  Create {`\"` + query + `\"`}\n                                </Arrange>\n                              </Item>\n                            )}\n                          </Combobox.Option>\n                        </>\n                      )}\n                      {noResults && <EmptyState />}\n                    </Combobox.Options>\n                  </MenuCardTransition>\n                </OptionsRoot>\n              </Root>\n              {errorMessage && (\n                <Box marginTop=\"var(--grn-form-control-label-gap)\" color=\"danger\">\n                  {errorMessage}\n                </Box>\n              )}\n            </>\n          );\n        }}\n      </Combobox>\n    </>\n  );\n};\n\nAutocomplete2.Label = FieldLabel;\nAutocomplete2.Hint = FieldHint;\nAutocomplete2.Item = Item;\n\nAutocomplete2.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.string,\n  onChange: PropTypes.func.isRequired,\n  isCreatable: PropTypes.bool,\n  onCreate: PropTypes.func,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuMaxHeight: types.dimension,\n  menuZIndex: types.zIndex,\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n  placeholder: PropTypes.string,\n  hasError: PropTypes.bool,\n  errorMessage: PropTypes.string,\n  isDisabled: PropTypes.bool,\n  backgroundColor: types.color,\n  onClear: PropTypes.func,\n  hasSpinner: PropTypes.bool,\n  size: PropTypes.oneOf(['m', 'l']),\n};\n"]} */"));
|
|
167
173
|
|
|
168
174
|
const TriggerText = /*#__PURE__*/_styled("span", process.env.NODE_ENV === "production" ? {
|
|
169
175
|
target: "ea1qwwn0"
|
|
170
176
|
} : {
|
|
171
177
|
target: "ea1qwwn0",
|
|
172
178
|
label: "TriggerText"
|
|
173
|
-
})("line-height:20px;display:block;padding-block:0;padding-
|
|
179
|
+
})("line-height:20px;display:block;padding-block:0;padding-right:", _ref16 => {
|
|
174
180
|
let {
|
|
175
181
|
paddingRight
|
|
176
|
-
} =
|
|
182
|
+
} = _ref16;
|
|
177
183
|
return paddingRight;
|
|
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"]} */"));
|
|
184
|
+
}, ";padding-left:", _ref17 => {
|
|
185
|
+
let {
|
|
186
|
+
paddingLeft
|
|
187
|
+
} = _ref17;
|
|
188
|
+
return paddingLeft;
|
|
189
|
+
}, ";text-overflow:ellipsis;overflow:hidden;white-space:nowrap;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/autocomplete2.jsx"],"names":[],"mappings":"AAuO+B","file":"../../src/components/autocomplete2.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useState } from 'react';\nimport { Icon, Box, Arrange, Spinner } from '.';\nimport { IconChevronDown, IconPlus, IconCross } from '../icons';\nimport { Combobox } from '@headlessui/react';\nimport { defaultProps, types } from '../types';\nimport {\n  InputField,\n  MenuCard,\n  MenuCardTransition,\n  MenuGroupTitle,\n  FieldLabel,\n  FieldHint,\n  styles,\n  FieldClearButton,\n} from '../foundational';\nimport { MenuItemDivider, useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\nimport { Global } from '@emotion/react';\nimport { getColor, getIconSize, getRadius, getSpace, getTransition } from '../utilities';\nimport styled from '@emotion/styled';\n\nconst transitionSpeed = getTransition('xxFast');\n\nconst Root = styled.div`\n  ${styles.transitions};\n  ${({ hasError, isDisabled }) =>\n    !hasError &&\n    !isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color);\n      &:hover { --borderColor: var(--grn-field-border-color-hover) };\n      &:focus-within { --borderColor: var(--grn-field-border-color-focus) };\n    `};\n\n  ${({ hasError }) => hasError && `--borderColor: ${getColor('danger')};`};\n\n  ${({ backgroundColor, isDisabled }) =>\n    !backgroundColor &&\n    !isDisabled &&\n    `\n      --backgroundColor: transparent;\n    `};\n\n  ${({ backgroundColor }) =>\n    backgroundColor &&\n    `\n      --backgroundColor: ${getColor(backgroundColor)};\n    `};\n\n  ${({ isDisabled }) =>\n    isDisabled &&\n    `\n      --borderColor: var(--grn-field-border-color-disabled);\n      --backgroundColor: var(--grn-field-background-disabled);\n\n      * {\n        pointer-events: none;\n      }\n    `};\n\n  position: relative;\n  margin-top: ${({ marginTop }) => getSpace(marginTop)};\n  border-radius: ${getRadius('s')};\n  box-shadow: inset 0 0 0 var(--grn-border-size) var(--borderColor);\n  background-color: var(--backgroundColor);\n\n  .autocompleteInput {\n    line-height: 20px;\n    border: none;\n    opacity: ${({ open }) => (open ? 1 : 0)};\n    pointer-events: ${({ open }) => (open ? 'auto' : 'none')};\n    transition: opacity ${transitionSpeed};\n\n    &:focus-visible,\n    &:placeholder-shown {\n      opacity: 1;\n      pointer-events: auto;\n      transition: opacity ${transitionSpeed};\n    }\n    &:disabled:placeholder-shown {\n      background: transparent;\n    }\n  }\n`;\n\nconst EmptyState = () => (\n  <Box paddingY=\"s\" paddingX=\"m\" color=\"content2\">\n    No results\n  </Box>\n);\n\nconst rightSectionGap = 'var(--grn-space-m)';\n\nconst RightSection = ({ children, right }) => (\n  <Arrange\n    right={right}\n    gap={rightSectionGap}\n    position=\"absolute\"\n    top=\"0px\"\n    bottom=\"0px\"\n    margin=\"auto\"\n    height=\"fit-content\"\n  >\n    {children}\n  </Arrange>\n);\n\nconst ClearButton = ({ onClick, ...props }) => {\n  const handleClear = e => {\n    e.stopPropagation();\n    onClick();\n  };\n  return <FieldClearButton onClick={e => handleClear(e)} {...props} />;\n};\n\nconst getFilteredOptions = (query, options, searchField) => {\n  if (query === '') return options;\n\n  return options.filter(option => option[searchField].toLowerCase().includes(query.toLowerCase()));\n};\n\nconst getShowGroupTitle = (index, option, filteredOptions) => {\n  const firstOptionHasTitle = index === 0 && option.groupTitle;\n  const titleChanged = index > 0 && option.groupTitle !== filteredOptions[index - 1].groupTitle;\n\n  return firstOptionHasTitle || titleChanged;\n};\n\nconst menuItemStyles = `\n  .autocompleteMenuItem {\n    display: flex;\n    align-items: center;\n    list-style: none;\n    min-height: var(--grn-textbox-height-m);\n    border-radius: ${getRadius('s')};\n    appearance: none;\n    padding-block: 0;\n    padding: 4px 12px;\n\n    --checkUri: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%2337312f73\"%3E%3C/path%3E%3C/svg%3E');\n    --checkUriActive: url('data:image/svg+xml,%3Csvg viewBox=\"0 0 22 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"%3E%3Cpath fill-rule=\"evenodd\" clip-rule=\"evenodd\" d=\"M21.707 6.207L8.5 19.414.293 11.207l1.414-1.414L8.5 16.586 20.293 4.793l1.414 1.414z\" fill=\"%23241a17\"%3E%3C/path%3E%3C/svg%3E');\n\n    &[data-headlessui-state='active'],\n    &[data-headlessui-state='active selected'] {\n      background-color: ${getColor('fade1')};\n    }\n\n    &[data-headlessui-state='selected'],\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUri);\n      background-repeat: no-repeat;\n      background-position: center right 12px;\n      background-size: auto ${getIconSize('m')};\n    }\n\n    &[data-headlessui-state='active selected'] {\n      background-image: var(--checkUriActive);\n    }\n\n    &[data-headlessui-state='disabled'] {\n      color: ${getColor('disabledContent')};\n      cursor: default;\n    }\n\n    ul & {\n      cursor: pointer;\n      padding-right: 40px;\n    }\n  }\n\n  .autocompleteMenuItemText {\n    flex-grow: 1;\n\n    &.hasEllipsis {\n      white-space: nowrap;\n      overflow: hidden;\n      text-overflow: ellipsis;\n    }\n  }\n`;\n\nconst Item = forwardRef(({ children, hasEllipsis, paddingLeft = '12px', ...props }, ref) => {\n  return (\n    <li\n      className=\"autocompleteMenuItem\"\n      ref={ref}\n      style={{ paddingLeft: getSpace(paddingLeft) }}\n      {...props}\n    >\n      <span\n        className={\n          hasEllipsis ? 'autocompleteMenuItemText hasEllipsis' : 'autocompleteMenuItemText'\n        }\n      >\n        {children}\n      </span>\n    </li>\n  );\n});\n\nconst Trigger = styled.button`\n  text-align: left;\n  height: ${({ size }) => `var(--grn-textbox-height-${size})`};\n  background-color: transparent;\n  cursor: text;\n  appearance: none;\n  display: block;\n  font: inherit;\n  color: inherit;\n  width: 100%;\n  outline: none;\n  border: none;\n  padding: 0;\n  opacity: ${({ open }) => (open ? 0 : 1)};\n  pointer-events: ${({ open }) => (open ? 'none' : 'auto')};\n  position: absolute;\n  top: 0;\n  left: 0;\n  transition: opacity ${transitionSpeed};\n\n  .autocompleteMenuItem {\n    padding-right: ${({ paddingRight }) => paddingRight};\n  }\n\n  .autocompleteInput:focus-visible + & {\n    opacity: 0;\n    pointer-events: none;\n  }\n`;\n\nconst TriggerText = styled.span`\n  line-height: 20px;\n  display: block;\n  padding-block: 0;\n  padding-right: ${({ paddingRight }) => paddingRight};\n  padding-left: ${({ paddingLeft }) => paddingLeft};\n  text-overflow: ellipsis;\n  overflow: hidden;\n  white-space: nowrap;\n`;\n\nexport const Autocomplete2 = ({\n  options,\n  value,\n  onChange,\n  isCreatable,\n  onCreate,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  menuZIndex,\n  placeholder,\n  label,\n  hint,\n  menuItemsHaveEllipsis = true,\n  hasPortal = true,\n  hasError,\n  errorMessage,\n  isDisabled,\n  searchField = 'content',\n  backgroundColor,\n  onClear,\n  hasSpinner,\n  size = 'm',\n  ...props\n}) => {\n  const [query, setQuery] = useState('');\n\n  const handleCreatableChange = option => {\n    if (onCreate) {\n      const isCreatableOption = !options.find(item => item.value === option.value);\n      if (isCreatableOption) {\n        onCreate(option);\n        return;\n      }\n    }\n    onChange(option);\n  };\n\n  const handleChange = option => {\n    if (isCreatable) {\n      handleCreatableChange(option);\n    } else {\n      onChange(option);\n    }\n  };\n\n  const filteredOptions = getFilteredOptions(query, options, searchField);\n\n  const isShowCreateOption = isCreatable && query.length > 0;\n\n  const noResults = !Boolean(filteredOptions.length) && !isShowCreateOption;\n\n  const fieldMarginTop = label || hint ? 'var(--grn-form-control-label-gap)' : undefined;\n\n  const selectedOption = useMemo(\n    () => options.find(option => option.value === value),\n    [options, value],\n  );\n\n  const newOption = useMemo(() => ({ value: query, content: query }), [query]);\n\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n\n  const OptionsRoot = hasPortal ? FloatingPortal : Fragment;\n\n  const hasClearButton = onClear && Boolean(selectedOption);\n\n  const inputPaddingX = `var(--grn-field-paddingX-${size})`;\n\n  const chevronWidth = `${IconChevronDown.viewBoxWidth / 2}px`;\n  const textGap = 'var(--grn-space-s2)';\n  const spinnerSpace = `calc(var(--grn-unit) * 3) + ${rightSectionGap}`;\n  const iconCrossWidth = `${IconCross.viewBoxWidth / 2}px`;\n  const clearButtonSpace = `${iconCrossWidth} + ${rightSectionGap}`;\n\n  const paddingRightValues = [textGap];\n  if (hasSpinner) paddingRightValues.push(spinnerSpace);\n  if (hasClearButton) paddingRightValues.push(clearButtonSpace);\n  paddingRightValues.push(chevronWidth, inputPaddingX);\n\n  const inputPaddingRight = `calc(${paddingRightValues.join(' + ')})`;\n\n  return (\n    <>\n      <Global styles={menuItemStyles} />\n      <Combobox\n        disabled={isDisabled}\n        as=\"div\"\n        value={selectedOption || ''}\n        onChange={handleChange}\n        {...props}\n      >\n        {({ open }) => {\n          const hasItem = selectedOption?.item;\n          return (\n            <>\n              {label && <Combobox.Label as={FieldLabel}>{label}</Combobox.Label>}\n              {hint && <FieldHint>{hint}</FieldHint>}\n\n              <Root\n                marginTop={fieldMarginTop}\n                ref={reference}\n                open={open}\n                hasError={hasError}\n                backgroundColor={backgroundColor}\n                isDisabled={isDisabled}\n              >\n                <Combobox.Button as={Box} position=\"relative\">\n                  <Combobox.Input\n                    as={InputField}\n                    autoComplete=\"off\"\n                    onChange={event => setQuery(event.target.value)}\n                    placeholder={placeholder}\n                    displayValue={option => option && option.content}\n                    paddingRight={inputPaddingRight}\n                    hasError={hasError}\n                    className=\"autocompleteInput\"\n                    size={size}\n                  />\n\n                  <Trigger open={open} tabIndex={-1} paddingRight={inputPaddingRight} size={size}>\n                    {hasItem &&\n                      selectedOption.item({\n                        isActive: false,\n                        isSelected: false,\n                        isDisabled: false,\n                      })}\n\n                    {!hasItem && selectedOption && (\n                      <TriggerText paddingRight={inputPaddingRight} paddingLeft={inputPaddingX}>\n                        {selectedOption?.content}\n                      </TriggerText>\n                    )}\n                  </Trigger>\n\n                  <RightSection right={inputPaddingX}>\n                    {hasSpinner && <Spinner />}\n                    {hasClearButton && <ClearButton onClick={onClear} trimSides />}\n                    <Icon icon={<IconChevronDown />} />\n                  </RightSection>\n                </Combobox.Button>\n\n                <OptionsRoot>\n                  <MenuCardTransition afterLeave={() => setQuery('')}>\n                    <Combobox.Options\n                      static\n                      ref={floating}\n                      placement={menuPlacement}\n                      maxHeight={menuMaxHeight}\n                      as={MenuCard}\n                      zIndex={menuZIndex}\n                      style={floatingStyles}\n                      isOpen={open}\n                    >\n                      {filteredOptions.map((option, index) => {\n                        const showGroupTitle = getShowGroupTitle(index, option, filteredOptions);\n\n                        return (\n                          <Fragment key={index}>\n                            {showGroupTitle && (\n                              <MenuGroupTitle hasDivider={index > 0}>\n                                {option.groupTitle}\n                              </MenuGroupTitle>\n                            )}\n\n                            <Combobox.Option\n                              value={option}\n                              as={Fragment}\n                              disabled={option.isDisabled}\n                            >\n                              {({ active }) => {\n                                const isDisabled = option.isDisabled;\n                                const isSelected = option.value === value;\n                                const isActive = !option.isDisabled && active;\n                                const hasEllipsis = menuItemsHaveEllipsis;\n                                if (option.item) {\n                                  return option.item({\n                                    isActive,\n                                    isSelected,\n                                    isDisabled,\n                                    hasEllipsis,\n                                  });\n                                }\n                                return <Item hasEllipsis={hasEllipsis}>{option.content}</Item>;\n                              }}\n                            </Combobox.Option>\n                          </Fragment>\n                        );\n                      })}\n                      {isShowCreateOption && (\n                        <>\n                          {filteredOptions.length > 0 && <MenuItemDivider />}\n                          <Combobox.Option value={newOption} as={Fragment}>\n                            {({ active }) => (\n                              <Item isActive={active} hasEllipsis={menuItemsHaveEllipsis}>\n                                <Arrange gap=\"s\" tag=\"span\">\n                                  <Icon icon={<IconPlus />} color={active ? 'content' : 'icon'} />\n                                  Create {`\"` + query + `\"`}\n                                </Arrange>\n                              </Item>\n                            )}\n                          </Combobox.Option>\n                        </>\n                      )}\n                      {noResults && <EmptyState />}\n                    </Combobox.Options>\n                  </MenuCardTransition>\n                </OptionsRoot>\n              </Root>\n              {errorMessage && (\n                <Box marginTop=\"var(--grn-form-control-label-gap)\" color=\"danger\">\n                  {errorMessage}\n                </Box>\n              )}\n            </>\n          );\n        }}\n      </Combobox>\n    </>\n  );\n};\n\nAutocomplete2.Label = FieldLabel;\nAutocomplete2.Hint = FieldHint;\nAutocomplete2.Item = Item;\n\nAutocomplete2.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.string,\n  onChange: PropTypes.func.isRequired,\n  isCreatable: PropTypes.bool,\n  onCreate: PropTypes.func,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuMaxHeight: types.dimension,\n  menuZIndex: types.zIndex,\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n  placeholder: PropTypes.string,\n  hasError: PropTypes.bool,\n  errorMessage: PropTypes.string,\n  isDisabled: PropTypes.bool,\n  backgroundColor: types.color,\n  onClear: PropTypes.func,\n  hasSpinner: PropTypes.bool,\n  size: PropTypes.oneOf(['m', 'l']),\n};\n"]} */"));
|
|
179
190
|
|
|
180
|
-
export const Autocomplete2 =
|
|
191
|
+
export const Autocomplete2 = _ref18 => {
|
|
181
192
|
let {
|
|
182
193
|
options,
|
|
183
194
|
value,
|
|
@@ -199,9 +210,10 @@ export const Autocomplete2 = _ref16 => {
|
|
|
199
210
|
searchField = 'content',
|
|
200
211
|
backgroundColor,
|
|
201
212
|
onClear,
|
|
202
|
-
hasSpinner
|
|
203
|
-
|
|
204
|
-
|
|
213
|
+
hasSpinner,
|
|
214
|
+
size = 'm'
|
|
215
|
+
} = _ref18,
|
|
216
|
+
props = _objectWithoutProperties(_ref18, _excluded3);
|
|
205
217
|
|
|
206
218
|
const [query, setQuery] = useState('');
|
|
207
219
|
|
|
@@ -245,8 +257,8 @@ export const Autocomplete2 = _ref16 => {
|
|
|
245
257
|
});
|
|
246
258
|
const OptionsRoot = hasPortal ? FloatingPortal : Fragment;
|
|
247
259
|
const hasClearButton = onClear && Boolean(selectedOption);
|
|
248
|
-
const
|
|
249
|
-
const
|
|
260
|
+
const inputPaddingX = "var(--grn-field-paddingX-".concat(size, ")");
|
|
261
|
+
const chevronWidth = "".concat(IconChevronDown.viewBoxWidth / 2, "px");
|
|
250
262
|
const textGap = 'var(--grn-space-s2)';
|
|
251
263
|
const spinnerSpace = "calc(var(--grn-unit) * 3) + ".concat(rightSectionGap);
|
|
252
264
|
const iconCrossWidth = "".concat(IconCross.viewBoxWidth / 2, "px");
|
|
@@ -254,7 +266,7 @@ export const Autocomplete2 = _ref16 => {
|
|
|
254
266
|
const paddingRightValues = [textGap];
|
|
255
267
|
if (hasSpinner) paddingRightValues.push(spinnerSpace);
|
|
256
268
|
if (hasClearButton) paddingRightValues.push(clearButtonSpace);
|
|
257
|
-
paddingRightValues.push(
|
|
269
|
+
paddingRightValues.push(chevronWidth, inputPaddingX);
|
|
258
270
|
const inputPaddingRight = "calc(".concat(paddingRightValues.join(' + '), ")");
|
|
259
271
|
return ___EmotionJSX(React.Fragment, null, ___EmotionJSX(Global, {
|
|
260
272
|
styles: menuItemStyles
|
|
@@ -263,10 +275,10 @@ export const Autocomplete2 = _ref16 => {
|
|
|
263
275
|
as: "div",
|
|
264
276
|
value: selectedOption || '',
|
|
265
277
|
onChange: handleChange
|
|
266
|
-
}, props),
|
|
278
|
+
}, props), _ref19 => {
|
|
267
279
|
let {
|
|
268
280
|
open
|
|
269
|
-
} =
|
|
281
|
+
} = _ref19;
|
|
270
282
|
const hasItem = selectedOption === null || selectedOption === void 0 ? void 0 : selectedOption.item;
|
|
271
283
|
return ___EmotionJSX(React.Fragment, null, label && ___EmotionJSX(Combobox.Label, {
|
|
272
284
|
as: FieldLabel
|
|
@@ -288,20 +300,25 @@ export const Autocomplete2 = _ref16 => {
|
|
|
288
300
|
displayValue: option => option && option.content,
|
|
289
301
|
paddingRight: inputPaddingRight,
|
|
290
302
|
hasError: hasError,
|
|
291
|
-
className: "autocompleteInput"
|
|
303
|
+
className: "autocompleteInput",
|
|
304
|
+
size: size
|
|
292
305
|
}), ___EmotionJSX(Trigger, {
|
|
293
306
|
open: open,
|
|
294
307
|
tabIndex: -1,
|
|
295
|
-
paddingRight: inputPaddingRight
|
|
308
|
+
paddingRight: inputPaddingRight,
|
|
309
|
+
size: size
|
|
296
310
|
}, hasItem && selectedOption.item({
|
|
297
311
|
isActive: false,
|
|
298
312
|
isSelected: false,
|
|
299
313
|
isDisabled: false
|
|
300
314
|
}), !hasItem && selectedOption && ___EmotionJSX(TriggerText, {
|
|
301
|
-
paddingRight: inputPaddingRight
|
|
302
|
-
|
|
315
|
+
paddingRight: inputPaddingRight,
|
|
316
|
+
paddingLeft: inputPaddingX
|
|
317
|
+
}, selectedOption === null || selectedOption === void 0 ? void 0 : selectedOption.content)), ___EmotionJSX(RightSection, {
|
|
318
|
+
right: inputPaddingX
|
|
319
|
+
}, hasSpinner && ___EmotionJSX(Spinner, null), hasClearButton && ___EmotionJSX(ClearButton, {
|
|
303
320
|
onClick: onClear,
|
|
304
|
-
|
|
321
|
+
trimSides: true
|
|
305
322
|
}), ___EmotionJSX(Icon, {
|
|
306
323
|
icon: ___EmotionJSX(IconChevronDown, null)
|
|
307
324
|
}))), ___EmotionJSX(OptionsRoot, null, ___EmotionJSX(MenuCardTransition, {
|
|
@@ -325,10 +342,10 @@ export const Autocomplete2 = _ref16 => {
|
|
|
325
342
|
value: option,
|
|
326
343
|
as: Fragment,
|
|
327
344
|
disabled: option.isDisabled
|
|
328
|
-
},
|
|
345
|
+
}, _ref20 => {
|
|
329
346
|
let {
|
|
330
347
|
active
|
|
331
|
-
} =
|
|
348
|
+
} = _ref20;
|
|
332
349
|
const isDisabled = option.isDisabled;
|
|
333
350
|
const isSelected = option.value === value;
|
|
334
351
|
const isActive = !option.isDisabled && active;
|
|
@@ -350,10 +367,10 @@ export const Autocomplete2 = _ref16 => {
|
|
|
350
367
|
}), isShowCreateOption && ___EmotionJSX(React.Fragment, null, filteredOptions.length > 0 && ___EmotionJSX(MenuItemDivider, null), ___EmotionJSX(Combobox.Option, {
|
|
351
368
|
value: newOption,
|
|
352
369
|
as: Fragment
|
|
353
|
-
},
|
|
370
|
+
}, _ref21 => {
|
|
354
371
|
let {
|
|
355
372
|
active
|
|
356
|
-
} =
|
|
373
|
+
} = _ref21;
|
|
357
374
|
return ___EmotionJSX(Item, {
|
|
358
375
|
isActive: active,
|
|
359
376
|
hasEllipsis: menuItemsHaveEllipsis
|
|
@@ -393,5 +410,6 @@ Autocomplete2.propTypes = {
|
|
|
393
410
|
isDisabled: PropTypes.bool,
|
|
394
411
|
backgroundColor: types.color,
|
|
395
412
|
onClear: PropTypes.func,
|
|
396
|
-
hasSpinner: PropTypes.bool
|
|
413
|
+
hasSpinner: PropTypes.bool,
|
|
414
|
+
size: PropTypes.oneOf(['m', 'l'])
|
|
397
415
|
};
|
package/es/components/button.js
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import _styled from "@emotion/styled/base";
|
|
2
|
-
const _excluded = ["children", "variant", "isDisabled", "icon", "iconPosition", "hasFullWidth", "hasSpinner", "type", "tag"];
|
|
2
|
+
const _excluded = ["children", "variant", "isDisabled", "icon", "iconPosition", "hasFullWidth", "hasSpinner", "type", "tag", "size"];
|
|
3
3
|
|
|
4
4
|
var _templateObject;
|
|
5
5
|
|
|
@@ -11,7 +11,7 @@ function _objectWithoutPropertiesLoose(source, excluded) { if (source == null) r
|
|
|
11
11
|
|
|
12
12
|
function _taggedTemplateLiteral(strings, raw) { if (!raw) { raw = strings.slice(0); } return Object.freeze(Object.defineProperties(strings, { raw: { value: Object.freeze(raw) } })); }
|
|
13
13
|
|
|
14
|
-
import { getColor,
|
|
14
|
+
import { getColor, getSpace, getTransition } from '../utilities';
|
|
15
15
|
import { Icon } from './icon';
|
|
16
16
|
import React, { forwardRef } from 'react';
|
|
17
17
|
import PropTypes from 'prop-types';
|
|
@@ -61,19 +61,34 @@ const Wrapper = /*#__PURE__*/_styled("button", process.env.NODE_ENV === "product
|
|
|
61
61
|
} : {
|
|
62
62
|
target: "e1blxzk90",
|
|
63
63
|
label: "Wrapper"
|
|
64
|
-
})(styles.transitions, ";", styles.buttonReset, ";position:relative;border:none;border-width:var(--grn-border-size);border-style:solid;border-radius:",
|
|
64
|
+
})(styles.transitions, ";", styles.buttonReset, ";position:relative;border:none;border-width:var(--grn-border-size);border-style:solid;border-radius:", _ref => {
|
|
65
65
|
let {
|
|
66
|
-
|
|
66
|
+
size
|
|
67
67
|
} = _ref;
|
|
68
|
+
return "var(--grn-textbox-radius-".concat(size, ")");
|
|
69
|
+
}, ";height:", _ref2 => {
|
|
70
|
+
let {
|
|
71
|
+
size
|
|
72
|
+
} = _ref2;
|
|
73
|
+
return "var(--grn-textbox-height-".concat(size, ")");
|
|
74
|
+
}, ";padding-block:0;padding-inline:", _ref3 => {
|
|
75
|
+
let {
|
|
76
|
+
size
|
|
77
|
+
} = _ref3;
|
|
78
|
+
return "var(--grn-textbox-paddingX-".concat(size, ")");
|
|
79
|
+
}, ";background-color:transparent;display:flex;align-items:center;justify-content:center;gap:12px;", p => p.hasFullWidth && "width: 100%", ";", p => tagStyles[p.as], ";", _ref4 => {
|
|
80
|
+
let {
|
|
81
|
+
variant
|
|
82
|
+
} = _ref4;
|
|
68
83
|
return "\n --iconColor: ".concat(getColor(variantColors[variant].icon), ";\n border-color: ").concat(getColor(variantColors[variant].border), ";\n color: ").concat(getColor(variantColors[variant].content), ";\n background: ").concat(getColor(variantColors[variant].background), ";\n\n @media(hover: hover) {\n &:hover {\n --iconColor: currentColor;\n ").concat(variantColors[variant].backgroundHover && "background: ".concat(getColor(variantColors[variant].backgroundHover)), ";\n ").concat(variantColors[variant].borderHover && "border-color:".concat(getColor(variantColors[variant].borderHover)), ";\n\n // set color to prevent inheriting from outisde reset\n color: ").concat(getColor(variantColors[variant].content), ";\n }\n\n &:active {\n ").concat(variantColors[variant].backgroundActive && "background: ".concat(getColor(variantColors[variant].backgroundActive)), ";\n ").concat(variantColors[variant].borderActive && "border-color:".concat(getColor(variantColors[variant].borderActive)), ";\n }\n }\n ");
|
|
69
|
-
}, " &:disabled{--iconColor:", getColor('disabledIcon'), ";color:", getColor('disabledContent'), ";background-color:", getColor('disabledBackground'), ";border-color:transparent;pointer-events:none;}.Icon{", p => p.iconPosition === 'right' && "order: 1", ";", styles.transitions, ";}.buttonText,.Icon{",
|
|
84
|
+
}, " &:disabled{--iconColor:", getColor('disabledIcon'), ";color:", getColor('disabledContent'), ";background-color:", getColor('disabledBackground'), ";border-color:transparent;pointer-events:none;}.Icon{", p => p.iconPosition === 'right' && "order: 1", ";", styles.transitions, ";}.buttonText,.Icon{", _ref5 => {
|
|
70
85
|
let {
|
|
71
86
|
hasSpinner
|
|
72
|
-
} =
|
|
87
|
+
} = _ref5;
|
|
73
88
|
return hasSpinner && "\n opacity: 0;\n transition: opacity ".concat(getTransition('fast'), ";\n ");
|
|
74
|
-
}, ";}.buttonSpinner{animation:", fadeIn, " ", getTransition('slow'), ";}[data-experiment-rounded-button] &{border-radius:var(--grn-textbox-height-m);}[data-experiment-compact-textbox] &{padding:0 ", getSpace('s2'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
89
|
+
}, ";}.buttonSpinner{animation:", fadeIn, " ", getTransition('slow'), ";}[data-experiment-rounded-button] &{border-radius:var(--grn-textbox-height-m);}[data-experiment-compact-textbox] &{padding:0 ", getSpace('s2'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb21wb25lbnRzL2J1dHRvbi5qc3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBNkQ2QiIsImZpbGUiOiIuLi8uLi9zcmMvY29tcG9uZW50cy9idXR0b24uanN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZ2V0Q29sb3IsIGdldFNwYWNlLCBnZXRUcmFuc2l0aW9uIH0gZnJvbSAnLi4vdXRpbGl0aWVzJztcbmltcG9ydCB7IEljb24gfSBmcm9tICcuL2ljb24nO1xuaW1wb3J0IFJlYWN0LCB7IGZvcndhcmRSZWYgfSBmcm9tICdyZWFjdCc7XG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCc7XG5pbXBvcnQgUHJvcFR5cGVzIGZyb20gJ3Byb3AtdHlwZXMnO1xuaW1wb3J0IHsgc3R5bGVzIH0gZnJvbSAnLi4vZm91bmRhdGlvbmFsJztcbmltcG9ydCB7IFNwaW5uZXIgfSBmcm9tICcuL3NwaW5uZXInO1xuaW1wb3J0IHsgdHlwZXMgfSBmcm9tICcuLi90eXBlcyc7XG5pbXBvcnQgeyBCb3ggfSBmcm9tICcuL2JveCc7XG5pbXBvcnQgeyBUZXh0IH0gZnJvbSAnLi90ZXh0JztcbmltcG9ydCB7IGtleWZyYW1lcyB9IGZyb20gJ0BlbW90aW9uL3JlYWN0JztcblxuY29uc3QgaWNvbkludmVydCA9ICdoc2wodmFyKC0tSFMpIHZhcigtLWJhY2tncm91bmRMKSAvIHZhcigtLWZhZGUxMC1hKSknO1xuXG5jb25zdCB2YXJpYW50Q29sb3JzID0ge1xuICBhY2NlbnQ6IHtcbiAgICBpY29uOiBpY29uSW52ZXJ0LFxuICAgIGNvbnRlbnQ6ICdzaGFkZTInLFxuICAgIGJvcmRlcjogJ3RyYW5zcGFyZW50JyxcbiAgICBiYWNrZ3JvdW5kOiAnc2hhZGUxMicsXG4gICAgYmFja2dyb3VuZEhvdmVyOiAnc2hhZGUxMCcsXG4gICAgYmFja2dyb3VuZEFjdGl2ZTogJ3NoYWRlOScsXG4gIH0sXG4gIGRhbmdlcjoge1xuICAgIGljb246ICdoc2woMCAwJSAxMDAlIC8gdmFyKC0tZmFkZTgtYSkpJyxcbiAgICBjb250ZW50OiAndmFyKC0td2hpdGUpJyxcbiAgICBib3JkZXI6ICd0cmFuc3BhcmVudCcsXG4gICAgYmFja2dyb3VuZDogJ2RhbmdlcicsXG4gICAgYmFja2dyb3VuZEhvdmVyOiAnaHNsKHZhcigtLXJlZEhTKSBjYWxjKHZhcigtLXJlZEwpICsgNCUpKScsXG4gICAgYmFja2dyb3VuZEFjdGl2ZTogJ2hzbCh2YXIoLS1yZWRIUykgY2FsYyh2YXIoLS1yZWRMKSArIDclKSknLFxuICB9LFxuICBuZXV0cmFsOiB7XG4gICAgaWNvbjogJ2ljb24nLFxuICAgIGNvbnRlbnQ6ICdjb250ZW50JyxcbiAgICBib3JkZXI6ICdmYWRlNCcsXG4gICAgYm9yZGVySG92ZXI6ICdmYWRlNicsXG4gICAgYm9yZGVyQWN0aXZlOiAnc2hhZGU4JyxcbiAgICBiYWNrZ3JvdW5kOiAndHJhbnNwYXJlbnQnLFxuICB9LFxufTtcblxuY29uc3QgZmFkZUluID0ga2V5ZnJhbWVzYFxuICAwJSwgMjUlIHsgb3BhY2l0eTogMCB9XG5gO1xuXG5jb25zdCB0YWdTdHlsZXMgPSB7XG4gIGE6IGBcbiAgICB0ZXh0LWRlY29yYXRpb246IG5vbmU7XG4gICAgd2lkdGg6IGZpdC1jb250ZW50O1xuXG4gICAgJjpob3ZlciB7XG4gICAgICB0ZXh0LWRlY29yYXRpb246IG5vbmU7XG4gICAgfVxuICBgLFxuICBidXR0b246IGBgLFxuICBzcGFuOiBgXG4gICAgd2lkdGg6IGZpdC1jb250ZW50O1xuICAgIGN1cnNvcjogcG9pbnRlcjtcbiAgYCxcbn07XG5cbmNvbnN0IFdyYXBwZXIgPSBzdHlsZWQuYnV0dG9uYFxuICAke3N0eWxlcy50cmFuc2l0aW9uc307XG4gICR7c3R5bGVzLmJ1dHRvblJlc2V0fTtcblxuICBwb3NpdGlvbjogcmVsYXRpdmU7XG4gIGJvcmRlcjogbm9uZTtcbiAgYm9yZGVyLXdpZHRoOiB2YXIoLS1ncm4tYm9yZGVyLXNpemUpO1xuICBib3JkZXItc3R5bGU6IHNvbGlkO1xuICBib3JkZXItcmFkaXVzOiAkeyh7IHNpemUgfSkgPT4gYHZhcigtLWdybi10ZXh0Ym94LXJhZGl1cy0ke3NpemV9KWB9O1xuICBoZWlnaHQ6ICR7KHsgc2l6ZSB9KSA9PiBgdmFyKC0tZ3JuLXRleHRib3gtaGVpZ2h0LSR7c2l6ZX0pYH07XG4gIHBhZGRpbmctYmxvY2s6IDA7XG4gIHBhZGRpbmctaW5saW5lOiAkeyh7IHNpemUgfSkgPT4gYHZhcigtLWdybi10ZXh0Ym94LXBhZGRpbmdYLSR7c2l6ZX0pYH07XG4gIGJhY2tncm91bmQtY29sb3I6IHRyYW5zcGFyZW50O1xuICBkaXNwbGF5OiBmbGV4O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBqdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjtcbiAgZ2FwOiAxMnB4O1xuICAke3AgPT4gcC5oYXNGdWxsV2lkdGggJiYgYHdpZHRoOiAxMDAlYH07XG4gICR7cCA9PiB0YWdTdHlsZXNbcC5hc119O1xuXG4gICR7KHsgdmFyaWFudCB9KSA9PiBgXG4gICAgLS1pY29uQ29sb3I6ICR7Z2V0Q29sb3IodmFyaWFudENvbG9yc1t2YXJpYW50XS5pY29uKX07XG4gICAgYm9yZGVyLWNvbG9yOiAke2dldENvbG9yKHZhcmlhbnRDb2xvcnNbdmFyaWFudF0uYm9yZGVyKX07XG4gICAgY29sb3I6ICR7Z2V0Q29sb3IodmFyaWFudENvbG9yc1t2YXJpYW50XS5jb250ZW50KX07XG4gICAgYmFja2dyb3VuZDogJHtnZXRDb2xvcih2YXJpYW50Q29sb3JzW3ZhcmlhbnRdLmJhY2tncm91bmQpfTtcblxuICAgIEBtZWRpYShob3ZlcjogaG92ZXIpIHtcbiAgICAgICY6aG92ZXIge1xuICAgICAgICAtLWljb25Db2xvcjogY3VycmVudENvbG9yO1xuICAgICAgICAke1xuICAgICAgICAgIHZhcmlhbnRDb2xvcnNbdmFyaWFudF0uYmFja2dyb3VuZEhvdmVyICYmXG4gICAgICAgICAgYGJhY2tncm91bmQ6ICR7Z2V0Q29sb3IodmFyaWFudENvbG9yc1t2YXJpYW50XS5iYWNrZ3JvdW5kSG92ZXIpfWBcbiAgICAgICAgfTtcbiAgICAgICAgJHtcbiAgICAgICAgICB2YXJpYW50Q29sb3JzW3ZhcmlhbnRdLmJvcmRlckhvdmVyICYmXG4gICAgICAgICAgYGJvcmRlci1jb2xvcjoke2dldENvbG9yKHZhcmlhbnRDb2xvcnNbdmFyaWFudF0uYm9yZGVySG92ZXIpfWBcbiAgICAgICAgfTtcblxuICAgICAgICAvLyBzZXQgY29sb3IgdG8gcHJldmVudCBpbmhlcml0aW5nIGZyb20gb3V0aXNkZSByZXNldFxuICAgICAgICBjb2xvcjogJHtnZXRDb2xvcih2YXJpYW50Q29sb3JzW3ZhcmlhbnRdLmNvbnRlbnQpfTtcbiAgICAgIH1cblxuICAgICAgJjphY3RpdmUge1xuICAgICAgICAke1xuICAgICAgICAgIHZhcmlhbnRDb2xvcnNbdmFyaWFudF0uYmFja2dyb3VuZEFjdGl2ZSAmJlxuICAgICAgICAgIGBiYWNrZ3JvdW5kOiAke2dldENvbG9yKHZhcmlhbnRDb2xvcnNbdmFyaWFudF0uYmFja2dyb3VuZEFjdGl2ZSl9YFxuICAgICAgICB9O1xuICAgICAgICAgJHtcbiAgICAgICAgICAgdmFyaWFudENvbG9yc1t2YXJpYW50XS5ib3JkZXJBY3RpdmUgJiZcbiAgICAgICAgICAgYGJvcmRlci1jb2xvcjoke2dldENvbG9yKHZhcmlhbnRDb2xvcnNbdmFyaWFudF0uYm9yZGVyQWN0aXZlKX1gXG4gICAgICAgICB9O1xuICAgICAgfVxuICAgIH1cbiAgYH1cblxuICAmOmRpc2FibGVkIHtcbiAgICAtLWljb25Db2xvcjogJHtnZXRDb2xvcignZGlzYWJsZWRJY29uJyl9O1xuICAgIGNvbG9yOiAke2dldENvbG9yKCdkaXNhYmxlZENvbnRlbnQnKX07XG4gICAgYmFja2dyb3VuZC1jb2xvcjogJHtnZXRDb2xvcignZGlzYWJsZWRCYWNrZ3JvdW5kJyl9O1xuICAgIGJvcmRlci1jb2xvcjogdHJhbnNwYXJlbnQ7XG4gICAgcG9pbnRlci1ldmVudHM6IG5vbmU7XG4gIH1cblxuICAuSWNvbiB7XG4gICAgJHtwID0+IHAuaWNvblBvc2l0aW9uID09PSAncmlnaHQnICYmIGBvcmRlcjogMWB9O1xuICAgICR7c3R5bGVzLnRyYW5zaXRpb25zfTtcbiAgfVxuXG4gIC5idXR0b25UZXh0LFxuICAuSWNvbiB7XG4gICAgJHsoeyBoYXNTcGlubmVyIH0pID0+XG4gICAgICBoYXNTcGlubmVyICYmXG4gICAgICBgXG4gICAgICBvcGFjaXR5OiAwO1xuICAgICAgdHJhbnNpdGlvbjogb3BhY2l0eSAke2dldFRyYW5zaXRpb24oJ2Zhc3QnKX07XG4gICAgYH1cbiAgfVxuXG4gIC5idXR0b25TcGlubmVyIHtcbiAgICBhbmltYXRpb246ICR7ZmFkZUlufSAke2dldFRyYW5zaXRpb24oJ3Nsb3cnKX07XG4gIH1cblxuICBbZGF0YS1leHBlcmltZW50LXJvdW5kZWQtYnV0dG9uXSAmIHtcbiAgICBib3JkZXItcmFkaXVzOiB2YXIoLS1ncm4tdGV4dGJveC1oZWlnaHQtbSk7XG4gIH1cblxuICBbZGF0YS1leHBlcmltZW50LWNvbXBhY3QtdGV4dGJveF0gJiB7XG4gICAgcGFkZGluZzogMCAke2dldFNwYWNlKCdzMicpfTtcbiAgfVxuYDtcblxuZXhwb3J0IGNvbnN0IEJ1dHRvbiA9IGZvcndhcmRSZWYoXG4gIChcbiAgICB7XG4gICAgICBjaGlsZHJlbixcbiAgICAgIHZhcmlhbnQgPSAnbmV1dHJhbCcsXG4gICAgICBpc0Rpc2FibGVkLFxuICAgICAgaWNvbixcbiAgICAgIGljb25Qb3NpdGlvbiA9ICdsZWZ0JyxcbiAgICAgIGhhc0Z1bGxXaWR0aCxcbiAgICAgIGhhc1NwaW5uZXIsXG4gICAgICB0eXBlID0gJ2J1dHRvbicsXG4gICAgICB0YWcgPSAnYnV0dG9uJyxcbiAgICAgIHNpemUgPSAnbScsXG4gICAgICAuLi5wcm9wc1xuICAgIH0sXG4gICAgcmVmLFxuICApID0+IHtcbiAgICByZXR1cm4gKFxuICAgICAgPFdyYXBwZXJcbiAgICAgICAgdHlwZT17dHlwZX1cbiAgICAgICAgdmFyaWFudD17dmFyaWFudH1cbiAgICAgICAgZGlzYWJsZWQ9e2lzRGlzYWJsZWR9XG4gICAgICAgIGljb249e2ljb259XG4gICAgICAgIGljb25Qb3NpdGlvbj17aWNvblBvc2l0aW9ufVxuICAgICAgICBoYXNGdWxsV2lkdGg9e2hhc0Z1bGxXaWR0aH1cbiAgICAgICAgaGFzU3Bpbm5lcj17aGFzU3Bpbm5lcn1cbiAgICAgICAgcmVmPXtyZWZ9XG4gICAgICAgIGFzPXt0YWd9XG4gICAgICAgIHNpemU9e3NpemV9XG4gICAgICAgIHsuLi5wcm9wc31cbiAgICAgID5cbiAgICAgICAge2hhc1NwaW5uZXIgJiYgKFxuICAgICAgICAgIDxCb3ggcG9zaXRpb249XCJhYnNvbHV0ZVwiIGNsYXNzTmFtZT1cImJ1dHRvblNwaW5uZXJcIj5cbiAgICAgICAgICAgIDxTcGlubmVyIGNvbG9yPVwiY3VycmVudENvbG9yXCIgLz5cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgKX1cbiAgICAgICAge2ljb24gJiYgPEljb24gaWNvbj17aWNvbn0gY29sb3I9XCJ2YXIoLS1pY29uQ29sb3IpXCIgLz59XG4gICAgICAgIDxUZXh0IGNsYXNzTmFtZT1cImJ1dHRvblRleHRcIj57Y2hpbGRyZW59PC9UZXh0PlxuICAgICAgPC9XcmFwcGVyPlxuICAgICk7XG4gIH0sXG4pO1xuXG5CdXR0b24ucHJvcFR5cGVzID0ge1xuICBjaGlsZHJlbjogUHJvcFR5cGVzLm5vZGUsXG4gIGljb246IFByb3BUeXBlcy5ub2RlLFxuICBpY29uUG9zaXRpb246IHR5cGVzLmljb25Qb3NpdGlvbixcbiAgdmFyaWFudDogdHlwZXMuYnV0dG9uVmFyaWFudCxcbiAgb25DbGljazogUHJvcFR5cGVzLmZ1bmMsXG4gIGlzRGlzYWJsZWQ6IFByb3BUeXBlcy5ib29sLFxuICBoYXNGdWxsV2lkdGg6IFByb3BUeXBlcy5ib29sLFxuICBoYXNTcGlubmVyOiBQcm9wVHlwZXMuYm9vbCxcbiAgdHlwZTogdHlwZXMuYnV0dG9uVHlwZSxcbiAgdGFnOiB0eXBlcy5idXR0b25UYWcsXG4gIHNpemU6IFByb3BUeXBlcy5vbmVPZihbJ20nLCAnbCddKSxcbn07XG4iXX0= */"));
|
|
75
90
|
|
|
76
|
-
export const Button = /*#__PURE__*/forwardRef((
|
|
91
|
+
export const Button = /*#__PURE__*/forwardRef((_ref6, ref) => {
|
|
77
92
|
let {
|
|
78
93
|
children,
|
|
79
94
|
variant = 'neutral',
|
|
@@ -83,9 +98,10 @@ export const Button = /*#__PURE__*/forwardRef((_ref3, ref) => {
|
|
|
83
98
|
hasFullWidth,
|
|
84
99
|
hasSpinner,
|
|
85
100
|
type = 'button',
|
|
86
|
-
tag = 'button'
|
|
87
|
-
|
|
88
|
-
|
|
101
|
+
tag = 'button',
|
|
102
|
+
size = 'm'
|
|
103
|
+
} = _ref6,
|
|
104
|
+
props = _objectWithoutProperties(_ref6, _excluded);
|
|
89
105
|
|
|
90
106
|
return ___EmotionJSX(Wrapper, _extends({
|
|
91
107
|
type: type,
|
|
@@ -96,7 +112,8 @@ export const Button = /*#__PURE__*/forwardRef((_ref3, ref) => {
|
|
|
96
112
|
hasFullWidth: hasFullWidth,
|
|
97
113
|
hasSpinner: hasSpinner,
|
|
98
114
|
ref: ref,
|
|
99
|
-
as: tag
|
|
115
|
+
as: tag,
|
|
116
|
+
size: size
|
|
100
117
|
}, props), hasSpinner && ___EmotionJSX(Box, {
|
|
101
118
|
position: "absolute",
|
|
102
119
|
className: "buttonSpinner"
|
|
@@ -119,5 +136,6 @@ Button.propTypes = {
|
|
|
119
136
|
hasFullWidth: PropTypes.bool,
|
|
120
137
|
hasSpinner: PropTypes.bool,
|
|
121
138
|
type: types.buttonType,
|
|
122
|
-
tag: types.buttonTag
|
|
139
|
+
tag: types.buttonTag,
|
|
140
|
+
size: PropTypes.oneOf(['m', 'l'])
|
|
123
141
|
};
|