@flodesk/grain 10.5.3 → 10.5.5
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.
|
@@ -92,7 +92,7 @@ var getShowGroupTitle = function getShowGroupTitle(index, option, filteredOption
|
|
|
92
92
|
return firstOptionHasTitle || titleChanged;
|
|
93
93
|
};
|
|
94
94
|
|
|
95
|
-
var menuItemStyles = /*#__PURE__*/css(".autocompleteMenuItem{display:flex;gap:8px;align-items:center;list-style:none;padding:4px 12px;margin:0 8px;min-height:", componentVars.textBoxHeight, ";border-radius:", getRadius('s'), ";appearance:none;&.hasPreContent{padding-left:0;}}.autocompleteMenuItemText{flex-grow:1;&.hasEllipsis{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;}}" + (process.env.NODE_ENV === "production" ? "" : ";label:menuItemStyles;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/autocomplete.jsx"],"names":[],"mappings":"AAsD0B","file":"../../src/components/autocomplete.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useRef, useState } from 'react';\nimport { Icon, Box, Text } from '.';\nimport { IconCheck, IconChevronDown } 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  componentVars,\n} from '../foundational';\nimport { useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\nimport { css, Global } from '@emotion/react';\nimport { getColor, getRadius } from '../utilities';\n\nconst Trigger = forwardRef(({ ...props }, ref) => <Box ref={ref} {...props} position=\"relative\" />);\n\nconst EmptyState = () => (\n  <Box paddingY=\"s\" paddingX=\"m\">\n    <Text color=\"content2\">No results</Text>\n  </Box>\n);\n\nconst ExpandIcon = () => (\n  <Box\n    right=\"fieldPaddingX\"\n    position=\"absolute\"\n    top=\"0px\"\n    bottom=\"0px\"\n    margin=\"auto\"\n    height=\"fit-content\"\n  >\n    <Icon icon={<IconChevronDown />} />\n  </Box>\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 = css`\n  .autocompleteMenuItem {\n    display: flex;\n    gap: 8px;\n    align-items: center;\n    list-style: none;\n    padding: 4px 12px;\n    margin: 0 8px;\n    min-height: ${componentVars.textBoxHeight};\n    border-radius: ${getRadius('s')};\n    appearance: none;\n\n    &.hasPreContent {\n      padding-left: 0;\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\nexport const Autocomplete = ({\n  options,\n  value,\n  onChange,\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  searchField = 'content',\n  ...props\n}) => {\n  const [query, setQuery] = useState('');\n\n  const handleChange = option => {\n    onChange(option);\n    setQuery('');\n  };\n\n  const filteredOptions = getFilteredOptions(query, options, searchField);\n\n  const noResults = !Boolean(filteredOptions.length);\n\n  const fieldMarginTop = label || hint ? 'betweenFormControlAndLabel' : undefined;\n\n  const selectedOption = useMemo(\n    () => options.find(option => option.value === value),\n    [options, value],\n  );\n\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n\n  const OptionsRoot = hasPortal ? FloatingPortal : Fragment;\n\n  const preContentRef = useRef(null);\n\n  const [preContentWidth, setPreContentWidth] = useState(0);\n\n  React.useEffect(() => {\n    if (preContentRef.current) {\n      setPreContentWidth(preContentRef.current.offsetWidth);\n    }\n  }, [selectedOption, preContentRef]);\n\n  return (\n    <>\n      <Global styles={menuItemStyles} />\n      <Combobox as=\"div\" value={selectedOption || ''} onChange={handleChange} {...props}>\n        {({ open }) => {\n          const hasPreContent = selectedOption?.preContent;\n          return (\n            <>\n              {label && <Combobox.Label as={FieldLabel}>{label}</Combobox.Label>}\n              {hint && <FieldHint>{hint}</FieldHint>}\n\n              <Box position=\"relative\" marginTop={fieldMarginTop} ref={reference}>\n                <Combobox.Button as={Trigger}>\n                  {hasPreContent && !open && (\n                    <Box\n                      ref={preContentRef}\n                      width=\"fit-content\"\n                      position=\"absolute\"\n                      top=\"0px\"\n                      bottom=\"0px\"\n                      margin=\"auto\"\n                      height=\"fit-content\"\n                      left=\"0\"\n                    >\n                      {selectedOption.preContent}\n                    </Box>\n                  )}\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=\"32px\"\n                    paddingLeft={!open && hasPreContent ? `${preContentWidth + 8}px` : undefined}\n                    hasError={hasError}\n                  />\n                  <ExpandIcon />\n                </Combobox.Button>\n\n                <OptionsRoot>\n                  <MenuCardTransition>\n                    <Combobox.Options\n                      static\n                      className=\"grn-context\"\n                      ref={floating}\n                      placement={menuPlacement}\n                      maxHeight={menuMaxHeight}\n                      as={MenuCard}\n                      zIndex={menuZIndex}\n                      style={floatingStyles}\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                                const hasPreContent = option.preContent;\n                                return (\n                                  <li\n                                    className={\n                                      hasPreContent\n                                        ? 'autocompleteMenuItem hasPreContent'\n                                        : 'autocompleteMenuItem'\n                                    }\n                                    style={{\n                                      backgroundColor: isActive && getColor('fade1'),\n                                      color: isDisabled && getColor('disabledContent'),\n                                      cursor: !isDisabled && 'pointer',\n                                    }}\n                                  >\n                                    {hasPreContent && option.preContent}\n                                    <span\n                                      className={\n                                        hasEllipsis\n                                          ? 'autocompleteMenuItemText hasEllipsis'\n                                          : 'autocompleteMenuItemText'\n                                      }\n                                    >\n                                      {option.content}\n                                    </span>\n                                    <Icon\n                                      style={{ opacity: isSelected ? 1 : 0 }}\n                                      icon={<IconCheck />}\n                                      color={isActive ? 'content' : 'icon'}\n                                    />\n                                  </li>\n                                );\n                              }}\n                            </Combobox.Option>\n                          </Fragment>\n                        );\n                      })}\n                      {noResults && <EmptyState />}\n                    </Combobox.Options>\n                  </MenuCardTransition>\n                </OptionsRoot>\n              </Box>\n              {errorMessage && (\n                <Box marginTop=\"betweenFormControlAndLabel\" color=\"danger\">\n                  {errorMessage}\n                </Box>\n              )}\n            </>\n          );\n        }}\n      </Combobox>\n    </>\n  );\n};\n\nAutocomplete.Label = FieldLabel;\nAutocomplete.Hint = FieldHint;\n\nAutocomplete.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.string,\n  onChange: PropTypes.func.isRequired,\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  hasError: PropTypes.bool,\n  errorMessage: PropTypes.string,\n};\n"]} */");
|
|
95
|
+
var menuItemStyles = /*#__PURE__*/css(".autocompleteMenuItem{display:flex;gap:8px;align-items:center;list-style:none;padding:4px 12px;margin:0 8px;min-height:", componentVars.textBoxHeight, ";border-radius:", getRadius('s'), ";appearance:none;&.hasPreContent{padding-left:0;}}.autocompleteMenuItemText{flex-grow:1;&.hasEllipsis{white-space:nowrap;overflow:hidden;text-overflow:ellipsis;}}" + (process.env.NODE_ENV === "production" ? "" : ";label:menuItemStyles;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/autocomplete.jsx"],"names":[],"mappings":"AAsD0B","file":"../../src/components/autocomplete.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, useRef, useState } from 'react';\nimport { Icon, Box, Text } from '.';\nimport { IconCheck, IconChevronDown } 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  componentVars,\n} from '../foundational';\nimport { useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\nimport { css, Global } from '@emotion/react';\nimport { getColor, getRadius } from '../utilities';\n\nconst Trigger = forwardRef(({ ...props }, ref) => <Box ref={ref} {...props} position=\"relative\" />);\n\nconst EmptyState = () => (\n  <Box paddingY=\"s\" paddingX=\"m\">\n    <Text color=\"content2\">No results</Text>\n  </Box>\n);\n\nconst ExpandIcon = () => (\n  <Box\n    right=\"fieldPaddingX\"\n    position=\"absolute\"\n    top=\"0px\"\n    bottom=\"0px\"\n    margin=\"auto\"\n    height=\"fit-content\"\n  >\n    <Icon icon={<IconChevronDown />} />\n  </Box>\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 = css`\n  .autocompleteMenuItem {\n    display: flex;\n    gap: 8px;\n    align-items: center;\n    list-style: none;\n    padding: 4px 12px;\n    margin: 0 8px;\n    min-height: ${componentVars.textBoxHeight};\n    border-radius: ${getRadius('s')};\n    appearance: none;\n\n    &.hasPreContent {\n      padding-left: 0;\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\nexport const Autocomplete = ({\n  options,\n  value,\n  onChange,\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  searchField = 'content',\n  ...props\n}) => {\n  const [query, setQuery] = useState('');\n\n  const handleChange = option => {\n    onChange(option);\n    setQuery('');\n  };\n\n  const filteredOptions = getFilteredOptions(query, options, searchField);\n\n  const noResults = !Boolean(filteredOptions.length);\n\n  const fieldMarginTop = label || hint ? 'betweenFormControlAndLabel' : undefined;\n\n  const selectedOption = useMemo(\n    () => options.find(option => option.value === value),\n    [options, value],\n  );\n\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n\n  const OptionsRoot = hasPortal ? FloatingPortal : Fragment;\n\n  const preContentRef = useRef(null);\n\n  const [preContentWidth, setPreContentWidth] = useState(0);\n\n  React.useEffect(() => {\n    if (preContentRef.current) {\n      setPreContentWidth(preContentRef.current.offsetWidth);\n    }\n  }, [selectedOption, preContentRef]);\n\n  return (\n    <>\n      <Global styles={menuItemStyles} />\n      <Combobox as=\"div\" value={selectedOption || ''} onChange={handleChange} {...props}>\n        {({ open }) => {\n          const hasPreContent = selectedOption?.preContent;\n          return (\n            <>\n              {label && <Combobox.Label as={FieldLabel}>{label}</Combobox.Label>}\n              {hint && <FieldHint>{hint}</FieldHint>}\n\n              <Box position=\"relative\" marginTop={fieldMarginTop} ref={reference}>\n                <Combobox.Button as={Trigger}>\n                  {hasPreContent && !open && (\n                    <Box\n                      ref={preContentRef}\n                      width=\"fit-content\"\n                      position=\"absolute\"\n                      top=\"0px\"\n                      bottom=\"0px\"\n                      margin=\"auto\"\n                      height=\"fit-content\"\n                      left=\"0\"\n                    >\n                      {selectedOption.preContent}\n                    </Box>\n                  )}\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=\"32px\"\n                    paddingLeft={!open && hasPreContent ? `${preContentWidth + 8}px` : undefined}\n                    hasError={hasError}\n                  />\n                  <ExpandIcon />\n                </Combobox.Button>\n\n                <OptionsRoot>\n                  <MenuCardTransition>\n                    <Combobox.Options\n                      static\n                      className=\"grn-context\"\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                                const hasPreContent = option.preContent;\n                                return (\n                                  <li\n                                    className={\n                                      hasPreContent\n                                        ? 'autocompleteMenuItem hasPreContent'\n                                        : 'autocompleteMenuItem'\n                                    }\n                                    style={{\n                                      backgroundColor: isActive && getColor('fade1'),\n                                      color: isDisabled && getColor('disabledContent'),\n                                      cursor: !isDisabled && 'pointer',\n                                    }}\n                                  >\n                                    {hasPreContent && option.preContent}\n                                    <span\n                                      className={\n                                        hasEllipsis\n                                          ? 'autocompleteMenuItemText hasEllipsis'\n                                          : 'autocompleteMenuItemText'\n                                      }\n                                    >\n                                      {option.content}\n                                    </span>\n                                    <Icon\n                                      style={{ opacity: isSelected ? 1 : 0 }}\n                                      icon={<IconCheck />}\n                                      color={isActive ? 'content' : 'icon'}\n                                    />\n                                  </li>\n                                );\n                              }}\n                            </Combobox.Option>\n                          </Fragment>\n                        );\n                      })}\n                      {noResults && <EmptyState />}\n                    </Combobox.Options>\n                  </MenuCardTransition>\n                </OptionsRoot>\n              </Box>\n              {errorMessage && (\n                <Box marginTop=\"betweenFormControlAndLabel\" color=\"danger\">\n                  {errorMessage}\n                </Box>\n              )}\n            </>\n          );\n        }}\n      </Combobox>\n    </>\n  );\n};\n\nAutocomplete.Label = FieldLabel;\nAutocomplete.Hint = FieldHint;\n\nAutocomplete.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.string,\n  onChange: PropTypes.func.isRequired,\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  hasError: PropTypes.bool,\n  errorMessage: PropTypes.string,\n};\n"]} */");
|
|
96
96
|
export var Autocomplete = function Autocomplete(_ref2) {
|
|
97
97
|
var options = _ref2.options,
|
|
98
98
|
value = _ref2.value,
|
|
@@ -203,7 +203,8 @@ export var Autocomplete = function Autocomplete(_ref2) {
|
|
|
203
203
|
maxHeight: menuMaxHeight,
|
|
204
204
|
as: MenuCard,
|
|
205
205
|
zIndex: menuZIndex,
|
|
206
|
-
style: floatingStyles
|
|
206
|
+
style: floatingStyles,
|
|
207
|
+
isOpen: open
|
|
207
208
|
}, filteredOptions.map(function (option, index) {
|
|
208
209
|
var showGroupTitle = getShowGroupTitle(index, option, filteredOptions);
|
|
209
210
|
return ___EmotionJSX(Fragment, {
|
|
@@ -76,7 +76,8 @@ export var Dropdown = function Dropdown(_ref2) {
|
|
|
76
76
|
as: MenuCard,
|
|
77
77
|
placement: menuPlacement,
|
|
78
78
|
zIndex: menuZIndex,
|
|
79
|
-
style: floatingStyles
|
|
79
|
+
style: floatingStyles,
|
|
80
|
+
isOpen: open
|
|
80
81
|
}, options.map(function (option, index) {
|
|
81
82
|
if (option.type === 'divider') return ___EmotionJSX(MenuItemDivider, {
|
|
82
83
|
key: index
|
package/es/components/select.js
CHANGED
|
@@ -5,7 +5,7 @@ import "core-js/modules/es.object.assign.js";
|
|
|
5
5
|
import _styled from "@emotion/styled/base";
|
|
6
6
|
var _excluded = ["children", "variant"],
|
|
7
7
|
_excluded2 = ["option", "isSelected", "menuItemsHaveEllipsis", "allowMultiple"],
|
|
8
|
-
_excluded3 = ["children", "floating", "hasPortal"];
|
|
8
|
+
_excluded3 = ["children", "floating", "hasPortal", "isOpen"];
|
|
9
9
|
import "core-js/modules/es.array.map.js";
|
|
10
10
|
import "core-js/modules/es.object.to-string.js";
|
|
11
11
|
import "core-js/modules/es.array.iterator.js";
|
|
@@ -38,7 +38,7 @@ var TriggerButton = /*#__PURE__*/_styled("button", process.env.NODE_ENV === "pro
|
|
|
38
38
|
} : {
|
|
39
39
|
target: "e17qd7kh1",
|
|
40
40
|
label: "TriggerButton"
|
|
41
|
-
})(styles.buttonReset, ";", styles.transitions, ";color:inherit;display:flex;align-items:center;gap:8px;border:none;background:", getColor('fade1'), ";padding:0 ", getSpace('fieldPaddingX'), ";min-height:", componentVars.textBoxHeight, ";border-radius:", getRadius('s'), ";width:100%;text-align:left;&:hover{background:", getColor('fade2'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb21wb25lbnRzL3NlbGVjdC5qc3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBb0JtQyIsImZpbGUiOiIuLi8uLi9zcmMvY29tcG9uZW50cy9zZWxlY3QuanN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFByb3BUeXBlcyBmcm9tICdwcm9wLXR5cGVzJztcbmltcG9ydCBSZWFjdCwgeyBmb3J3YXJkUmVmLCBGcmFnbWVudCwgdXNlTWVtbyB9IGZyb20gJ3JlYWN0JztcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJztcbmltcG9ydCB7IEJveCwgSWNvbiwgVGV4dCwgVGV4dEJ1dHRvbiB9IGZyb20gJy4nO1xuaW1wb3J0IHsgSWNvbkNoZXZyb25Eb3duIH0gZnJvbSAnLi4vaWNvbnMnO1xuaW1wb3J0IHsgTGlzdGJveCB9IGZyb20gJ0BoZWFkbGVzc3VpL3JlYWN0JztcbmltcG9ydCB7IGdldENvbG9yLCBnZXRSYWRpdXMsIGdldFNwYWNlIH0gZnJvbSAnLi4vdXRpbGl0aWVzJztcbmltcG9ydCB7IGRlZmF1bHRQcm9wcywgdHlwZXMgfSBmcm9tICcuLi90eXBlcyc7XG5pbXBvcnQge1xuICBjb21wb25lbnRWYXJzLFxuICBGaWVsZEhpbnQsXG4gIEZpZWxkTGFiZWwsXG4gIE1lbnVDYXJkLFxuICBNZW51Q2FyZFRyYW5zaXRpb24sXG4gIE1lbnVJdGVtLFxuICBzdHlsZXMsXG59IGZyb20gJy4uL2ZvdW5kYXRpb25hbCc7XG5pbXBvcnQgeyB1c2VNZW51UG9zaXRpb24gfSBmcm9tICcuLi9mb3VuZGF0aW9uYWwvbWVudSc7XG5pbXBvcnQgeyBGbG9hdGluZ1BvcnRhbCB9IGZyb20gJ0BmbG9hdGluZy11aS9yZWFjdC1kb20taW50ZXJhY3Rpb25zJztcblxuY29uc3QgVHJpZ2dlckJ1dHRvbiA9IHN0eWxlZC5idXR0b25gXG4gICR7c3R5bGVzLmJ1dHRvblJlc2V0fTtcbiAgJHtzdHlsZXMudHJhbnNpdGlvbnN9O1xuXG4gIGNvbG9yOiBpbmhlcml0O1xuICBkaXNwbGF5OiBmbGV4O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBnYXA6IDhweDtcbiAgYm9yZGVyOiBub25lO1xuICBiYWNrZ3JvdW5kOiAke2dldENvbG9yKCdmYWRlMScpfTtcbiAgcGFkZGluZzogMCAke2dldFNwYWNlKCdmaWVsZFBhZGRpbmdYJyl9O1xuICBtaW4taGVpZ2h0OiAke2NvbXBvbmVudFZhcnMudGV4dEJveEhlaWdodH07XG4gIGJvcmRlci1yYWRpdXM6ICR7Z2V0UmFkaXVzKCdzJyl9O1xuICB3aWR0aDogMTAwJTtcbiAgdGV4dC1hbGlnbjogbGVmdDtcblxuICAmOmhvdmVyIHtcbiAgICBiYWNrZ3JvdW5kOiAke2dldENvbG9yKCdmYWRlMicpfTtcbiAgfVxuYDtcblxuY29uc3QgVHJpZ2dlciA9IGZvcndhcmRSZWYoKHsgY2hpbGRyZW4sIHZhcmlhbnQgPSAnYm94JywgLi4ucHJvcHMgfSwgcmVmKSA9PiAoXG4gIDw+XG4gICAge3ZhcmlhbnQgPT09ICdib3gnICYmIChcbiAgICAgIDxUcmlnZ2VyQnV0dG9uIHJlZj17cmVmfSB0eXBlPVwiYnV0dG9uXCIgey4uLnByb3BzfT5cbiAgICAgICAgPFRleHQgaGFzRWxsaXBzaXM+e2NoaWxkcmVufTwvVGV4dD5cbiAgICAgICAgPEJveCBtYXJnaW5MZWZ0PVwiYXV0b1wiPlxuICAgICAgICAgIDxJY29uIGljb249ezxJY29uQ2hldnJvbkRvd24gLz59IC8+XG4gICAgICAgIDwvQm94PlxuICAgICAgPC9UcmlnZ2VyQnV0dG9uPlxuICAgICl9XG4gICAge3ZhcmlhbnQgPT09ICd0ZXh0JyAmJiAoXG4gICAgICA8VGV4dEJ1dHRvbiByZWY9e3JlZn0gaWNvbj17PEljb25DaGV2cm9uRG93biAvPn0gaWNvblBvc2l0aW9uPVwicmlnaHRcIiB7Li4ucHJvcHN9PlxuICAgICAgICB7Y2hpbGRyZW59XG4gICAgICA8L1RleHRCdXR0b24+XG4gICAgKX1cbiAgPC8+XG4pKTtcblxuY29uc3QgZ2V0VHJpZ2dlckNvbnRlbnQgPSAoaXNNdWx0aXBsZSwgc2VsZWN0ZWQpID0+IHtcbiAgaWYgKCFzZWxlY3RlZCkgcmV0dXJuIG51bGw7XG4gIGlmIChpc011bHRpcGxlKSB7XG4gICAgcmV0dXJuIHNlbGVjdGVkLm1hcChvcHRpb24gPT4gb3B0aW9uLmNvbnRlbnQpLmpvaW4oJywgJyk7XG4gIH1cbiAgcmV0dXJuIHNlbGVjdGVkLmNvbnRlbnQ7XG59O1xuXG5jb25zdCBTZWxlY3RCdXR0b24gPSAoeyBjdXN0b21UcmlnZ2VyLCB2YXJpYW50LCBpc011bHRpcGxlLCBzZWxlY3RlZCB9KSA9PiB7XG4gIGNvbnN0IHByb3BzID0ge1xuICAgIHZhcmlhbnQsXG4gICAgY2hpbGRyZW46IGdldFRyaWdnZXJDb250ZW50KGlzTXVsdGlwbGUsIHNlbGVjdGVkKSxcbiAgfTtcbiAgcmV0dXJuIChcbiAgICA8TGlzdGJveC5CdXR0b24gYXM9e0ZyYWdtZW50fT5cbiAgICAgIHtjdXN0b21UcmlnZ2VyID8gY3VzdG9tVHJpZ2dlcih7IHByb3BzIH0pIDogPFRyaWdnZXIgey4uLnByb3BzfSAvPn1cbiAgICA8L0xpc3Rib3guQnV0dG9uPlxuICApO1xufTtcblxuY29uc3QgUm9vdCA9IHN0eWxlZC5kaXZgXG4gIG1pbi13aWR0aDogMHB4O1xuYDtcblxuY29uc3QgU2VsZWN0TWVudU9wdGlvbiA9ICh7XG4gIG9wdGlvbixcbiAgaXNTZWxlY3RlZCxcbiAgbWVudUl0ZW1zSGF2ZUVsbGlwc2lzLFxuICBhbGxvd011bHRpcGxlLFxuICAuLi5wcm9wc1xufSkgPT4gKFxuICA8TGlzdGJveC5PcHRpb24gdmFsdWU9e29wdGlvbn0gYXM9e0ZyYWdtZW50fSBkaXNhYmxlZD17b3B0aW9uLmlzRGlzYWJsZWR9IHsuLi5wcm9wc30+XG4gICAgeyh7IGFjdGl2ZSB9KSA9PiB7XG4gICAgICByZXR1cm4gKFxuICAgICAgICA8TWVudUl0ZW1cbiAgICAgICAgICBpc1NlbGVjdGVkPXtpc1NlbGVjdGVkfVxuICAgICAgICAgIGlzQWN0aXZlPXthY3RpdmV9XG4gICAgICAgICAgaXNEaXNhYmxlZD17b3B0aW9uLmlzRGlzYWJsZWR9XG4gICAgICAgICAgaGFzRWxsaXBzaXM9e21lbnVJdGVtc0hhdmVFbGxpcHNpc31cbiAgICAgICAgICBjaGVja1ZhcmlhbnQ9e2FsbG93TXVsdGlwbGUgPyAnY2hlY2tib3gnIDogJ2NoZWNrJ31cbiAgICAgICAgPlxuICAgICAgICAgIHtvcHRpb24uY29udGVudH1cbiAgICAgICAgPC9NZW51SXRlbT5cbiAgICAgICk7XG4gICAgfX1cbiAgPC9MaXN0Ym94Lk9wdGlvbj5cbik7XG5cbmNvbnN0IFNlbGVjdEluZm8gPSAoeyBsYWJlbCwgaGludCB9KSA9PiAoXG4gIDw+XG4gICAgeyhsYWJlbCB8fCBoaW50KSAmJiAoXG4gICAgICA8Qm94IG1hcmdpbkJvdHRvbT1cImJldHdlZW5Gb3JtQ29udHJvbEFuZExhYmVsXCI+XG4gICAgICAgIHtsYWJlbCAmJiA8TGlzdGJveC5MYWJlbCBhcz17RmllbGRMYWJlbH0+e2xhYmVsfTwvTGlzdGJveC5MYWJlbD59XG4gICAgICAgIHtoaW50ICYmIDxGaWVsZEhpbnQ+e2hpbnR9PC9GaWVsZEhpbnQ+fVxuICAgICAgPC9Cb3g+XG4gICAgKX1cbiAgPC8+
|
|
41
|
+
})(styles.buttonReset, ";", styles.transitions, ";color:inherit;display:flex;align-items:center;gap:8px;border:none;background:", getColor('fade1'), ";padding:0 ", getSpace('fieldPaddingX'), ";min-height:", componentVars.textBoxHeight, ";border-radius:", getRadius('s'), ";width:100%;text-align:left;&:hover{background:", getColor('fade2'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/select.jsx"],"names":[],"mappings":"AAoBmC","file":"../../src/components/select.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { Box, Icon, Text, TextButton } from '.';\nimport { IconChevronDown } from '../icons';\nimport { Listbox } from '@headlessui/react';\nimport { getColor, getRadius, getSpace } from '../utilities';\nimport { defaultProps, types } from '../types';\nimport {\n  componentVars,\n  FieldHint,\n  FieldLabel,\n  MenuCard,\n  MenuCardTransition,\n  MenuItem,\n  styles,\n} from '../foundational';\nimport { useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\n\nconst TriggerButton = styled.button`\n  ${styles.buttonReset};\n  ${styles.transitions};\n\n  color: inherit;\n  display: flex;\n  align-items: center;\n  gap: 8px;\n  border: none;\n  background: ${getColor('fade1')};\n  padding: 0 ${getSpace('fieldPaddingX')};\n  min-height: ${componentVars.textBoxHeight};\n  border-radius: ${getRadius('s')};\n  width: 100%;\n  text-align: left;\n\n  &:hover {\n    background: ${getColor('fade2')};\n  }\n`;\n\nconst Trigger = forwardRef(({ children, variant = 'box', ...props }, ref) => (\n  <>\n    {variant === 'box' && (\n      <TriggerButton ref={ref} type=\"button\" {...props}>\n        <Text hasEllipsis>{children}</Text>\n        <Box marginLeft=\"auto\">\n          <Icon icon={<IconChevronDown />} />\n        </Box>\n      </TriggerButton>\n    )}\n    {variant === 'text' && (\n      <TextButton ref={ref} icon={<IconChevronDown />} iconPosition=\"right\" {...props}>\n        {children}\n      </TextButton>\n    )}\n  </>\n));\n\nconst getTriggerContent = (isMultiple, selected) => {\n  if (!selected) return null;\n  if (isMultiple) {\n    return selected.map(option => option.content).join(', ');\n  }\n  return selected.content;\n};\n\nconst SelectButton = ({ customTrigger, variant, isMultiple, selected }) => {\n  const props = {\n    variant,\n    children: getTriggerContent(isMultiple, selected),\n  };\n  return (\n    <Listbox.Button as={Fragment}>\n      {customTrigger ? customTrigger({ props }) : <Trigger {...props} />}\n    </Listbox.Button>\n  );\n};\n\nconst Root = styled.div`\n  min-width: 0px;\n`;\n\nconst SelectMenuOption = ({\n  option,\n  isSelected,\n  menuItemsHaveEllipsis,\n  allowMultiple,\n  ...props\n}) => (\n  <Listbox.Option value={option} as={Fragment} disabled={option.isDisabled} {...props}>\n    {({ active }) => {\n      return (\n        <MenuItem\n          isSelected={isSelected}\n          isActive={active}\n          isDisabled={option.isDisabled}\n          hasEllipsis={menuItemsHaveEllipsis}\n          checkVariant={allowMultiple ? 'checkbox' : 'check'}\n        >\n          {option.content}\n        </MenuItem>\n      );\n    }}\n  </Listbox.Option>\n);\n\nconst SelectInfo = ({ label, hint }) => (\n  <>\n    {(label || hint) && (\n      <Box marginBottom=\"betweenFormControlAndLabel\">\n        {label && <Listbox.Label as={FieldLabel}>{label}</Listbox.Label>}\n        {hint && <FieldHint>{hint}</FieldHint>}\n      </Box>\n    )}\n  </>\n);\n\nconst SelectMenu = ({ children, floating, hasPortal, isOpen, ...props }) => {\n  const Root = hasPortal ? FloatingPortal : Fragment;\n  return (\n    <Root>\n      <MenuCardTransition>\n        <Listbox.Options\n          static\n          className=\"grn-context\"\n          ref={floating}\n          as={MenuCard}\n          isOpen={isOpen}\n          {...props}\n        >\n          {children}\n        </Listbox.Options>\n      </MenuCardTransition>\n    </Root>\n  );\n};\n\nconst getIsSelected = (isMultiple, option, selected) => {\n  if (!selected) return null;\n  if (isMultiple) {\n    return selected.some(selectedOption => selectedOption.content === option.content);\n  }\n\n  return option.value === selected.value;\n};\n\nexport const Select = ({\n  options,\n  value,\n  onChange,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  trigger: customTrigger,\n  triggerVariant = 'box',\n  label,\n  hint,\n  menuZIndex,\n  menuItemsHaveEllipsis = true,\n  allowMultiple,\n  hasPortal = true,\n}) => {\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n  const rootWidth = triggerVariant === 'text' ? 'fit-content' : undefined;\n\n  const optionMap = useMemo(() => {\n    return new Map(options.map(option => [option.value, option]));\n  }, [options]);\n\n  const selected = useMemo(\n    () => (allowMultiple ? value.map(value => optionMap.get(value)) : optionMap.get(value)),\n    [allowMultiple, optionMap, value],\n  );\n\n  return (\n    <Listbox\n      as={Root}\n      value={selected}\n      onChange={onChange}\n      width={rootWidth}\n      multiple={allowMultiple}\n    >\n      {({ open }) => (\n        <>\n          <SelectInfo label={label} hint={hint} />\n\n          <Box position=\"relative\" ref={reference}>\n            <SelectButton\n              customTrigger={customTrigger}\n              variant={triggerVariant}\n              isMultiple={allowMultiple}\n              selected={selected}\n            />\n\n            <SelectMenu\n              placement={menuPlacement}\n              maxHeight={menuMaxHeight}\n              zIndex={menuZIndex}\n              floating={floating}\n              style={floatingStyles}\n              hasPortal={hasPortal}\n              isOpen={open}\n            >\n              {options.map((option, index) => {\n                const isSelected = getIsSelected(allowMultiple, option, selected);\n                return (\n                  <SelectMenuOption\n                    isSelected={isSelected}\n                    key={index}\n                    option={option}\n                    hasEllipsis={menuItemsHaveEllipsis}\n                    allowMultiple={allowMultiple}\n                  />\n                );\n              })}\n            </SelectMenu>\n          </Box>\n        </>\n      )}\n    </Listbox>\n  );\n};\n\nSelect.Label = FieldLabel;\nSelect.Hint = FieldHint;\nSelect.Trigger = Trigger;\n\nSelect.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.oneOfType([PropTypes.string, PropTypes.array]).isRequired,\n  onChange: PropTypes.func.isRequired,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuZIndex: types.zIndex,\n  menuMaxHeight: types.dimension,\n  trigger: PropTypes.func,\n  triggerVariant: PropTypes.oneOf(['box', 'text']),\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  allowMultiple: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n};\n"]} */"));
|
|
42
42
|
|
|
43
43
|
var Trigger = /*#__PURE__*/forwardRef(function (_ref, ref) {
|
|
44
44
|
var children = _ref.children,
|
|
@@ -101,7 +101,7 @@ var Root = /*#__PURE__*/_styled("div", process.env.NODE_ENV === "production" ? {
|
|
|
101
101
|
} : {
|
|
102
102
|
name: "wowqs1",
|
|
103
103
|
styles: "min-width:0px",
|
|
104
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb21wb25lbnRzL3NlbGVjdC5qc3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBK0V1QiIsImZpbGUiOiIuLi8uLi9zcmMvY29tcG9uZW50cy9zZWxlY3QuanN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFByb3BUeXBlcyBmcm9tICdwcm9wLXR5cGVzJztcbmltcG9ydCBSZWFjdCwgeyBmb3J3YXJkUmVmLCBGcmFnbWVudCwgdXNlTWVtbyB9IGZyb20gJ3JlYWN0JztcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJztcbmltcG9ydCB7IEJveCwgSWNvbiwgVGV4dCwgVGV4dEJ1dHRvbiB9IGZyb20gJy4nO1xuaW1wb3J0IHsgSWNvbkNoZXZyb25Eb3duIH0gZnJvbSAnLi4vaWNvbnMnO1xuaW1wb3J0IHsgTGlzdGJveCB9IGZyb20gJ0BoZWFkbGVzc3VpL3JlYWN0JztcbmltcG9ydCB7IGdldENvbG9yLCBnZXRSYWRpdXMsIGdldFNwYWNlIH0gZnJvbSAnLi4vdXRpbGl0aWVzJztcbmltcG9ydCB7IGRlZmF1bHRQcm9wcywgdHlwZXMgfSBmcm9tICcuLi90eXBlcyc7XG5pbXBvcnQge1xuICBjb21wb25lbnRWYXJzLFxuICBGaWVsZEhpbnQsXG4gIEZpZWxkTGFiZWwsXG4gIE1lbnVDYXJkLFxuICBNZW51Q2FyZFRyYW5zaXRpb24sXG4gIE1lbnVJdGVtLFxuICBzdHlsZXMsXG59IGZyb20gJy4uL2ZvdW5kYXRpb25hbCc7XG5pbXBvcnQgeyB1c2VNZW51UG9zaXRpb24gfSBmcm9tICcuLi9mb3VuZGF0aW9uYWwvbWVudSc7XG5pbXBvcnQgeyBGbG9hdGluZ1BvcnRhbCB9IGZyb20gJ0BmbG9hdGluZy11aS9yZWFjdC1kb20taW50ZXJhY3Rpb25zJztcblxuY29uc3QgVHJpZ2dlckJ1dHRvbiA9IHN0eWxlZC5idXR0b25gXG4gICR7c3R5bGVzLmJ1dHRvblJlc2V0fTtcbiAgJHtzdHlsZXMudHJhbnNpdGlvbnN9O1xuXG4gIGNvbG9yOiBpbmhlcml0O1xuICBkaXNwbGF5OiBmbGV4O1xuICBhbGlnbi1pdGVtczogY2VudGVyO1xuICBnYXA6IDhweDtcbiAgYm9yZGVyOiBub25lO1xuICBiYWNrZ3JvdW5kOiAke2dldENvbG9yKCdmYWRlMScpfTtcbiAgcGFkZGluZzogMCAke2dldFNwYWNlKCdmaWVsZFBhZGRpbmdYJyl9O1xuICBtaW4taGVpZ2h0OiAke2NvbXBvbmVudFZhcnMudGV4dEJveEhlaWdodH07XG4gIGJvcmRlci1yYWRpdXM6ICR7Z2V0UmFkaXVzKCdzJyl9O1xuICB3aWR0aDogMTAwJTtcbiAgdGV4dC1hbGlnbjogbGVmdDtcblxuICAmOmhvdmVyIHtcbiAgICBiYWNrZ3JvdW5kOiAke2dldENvbG9yKCdmYWRlMicpfTtcbiAgfVxuYDtcblxuY29uc3QgVHJpZ2dlciA9IGZvcndhcmRSZWYoKHsgY2hpbGRyZW4sIHZhcmlhbnQgPSAnYm94JywgLi4ucHJvcHMgfSwgcmVmKSA9PiAoXG4gIDw+XG4gICAge3ZhcmlhbnQgPT09ICdib3gnICYmIChcbiAgICAgIDxUcmlnZ2VyQnV0dG9uIHJlZj17cmVmfSB0eXBlPVwiYnV0dG9uXCIgey4uLnByb3BzfT5cbiAgICAgICAgPFRleHQgaGFzRWxsaXBzaXM+e2NoaWxkcmVufTwvVGV4dD5cbiAgICAgICAgPEJveCBtYXJnaW5MZWZ0PVwiYXV0b1wiPlxuICAgICAgICAgIDxJY29uIGljb249ezxJY29uQ2hldnJvbkRvd24gLz59IC8+XG4gICAgICAgIDwvQm94PlxuICAgICAgPC9UcmlnZ2VyQnV0dG9uPlxuICAgICl9XG4gICAge3ZhcmlhbnQgPT09ICd0ZXh0JyAmJiAoXG4gICAgICA8VGV4dEJ1dHRvbiByZWY9e3JlZn0gaWNvbj17PEljb25DaGV2cm9uRG93biAvPn0gaWNvblBvc2l0aW9uPVwicmlnaHRcIiB7Li4ucHJvcHN9PlxuICAgICAgICB7Y2hpbGRyZW59XG4gICAgICA8L1RleHRCdXR0b24+XG4gICAgKX1cbiAgPC8+XG4pKTtcblxuY29uc3QgZ2V0VHJpZ2dlckNvbnRlbnQgPSAoaXNNdWx0aXBsZSwgc2VsZWN0ZWQpID0+IHtcbiAgaWYgKCFzZWxlY3RlZCkgcmV0dXJuIG51bGw7XG4gIGlmIChpc011bHRpcGxlKSB7XG4gICAgcmV0dXJuIHNlbGVjdGVkLm1hcChvcHRpb24gPT4gb3B0aW9uLmNvbnRlbnQpLmpvaW4oJywgJyk7XG4gIH1cbiAgcmV0dXJuIHNlbGVjdGVkLmNvbnRlbnQ7XG59O1xuXG5jb25zdCBTZWxlY3RCdXR0b24gPSAoeyBjdXN0b21UcmlnZ2VyLCB2YXJpYW50LCBpc011bHRpcGxlLCBzZWxlY3RlZCB9KSA9PiB7XG4gIGNvbnN0IHByb3BzID0ge1xuICAgIHZhcmlhbnQsXG4gICAgY2hpbGRyZW46IGdldFRyaWdnZXJDb250ZW50KGlzTXVsdGlwbGUsIHNlbGVjdGVkKSxcbiAgfTtcbiAgcmV0dXJuIChcbiAgICA8TGlzdGJveC5CdXR0b24gYXM9e0ZyYWdtZW50fT5cbiAgICAgIHtjdXN0b21UcmlnZ2VyID8gY3VzdG9tVHJpZ2dlcih7IHByb3BzIH0pIDogPFRyaWdnZXIgey4uLnByb3BzfSAvPn1cbiAgICA8L0xpc3Rib3guQnV0dG9uPlxuICApO1xufTtcblxuY29uc3QgUm9vdCA9IHN0eWxlZC5kaXZgXG4gIG1pbi13aWR0aDogMHB4O1xuYDtcblxuY29uc3QgU2VsZWN0TWVudU9wdGlvbiA9ICh7XG4gIG9wdGlvbixcbiAgaXNTZWxlY3RlZCxcbiAgbWVudUl0ZW1zSGF2ZUVsbGlwc2lzLFxuICBhbGxvd011bHRpcGxlLFxuICAuLi5wcm9wc1xufSkgPT4gKFxuICA8TGlzdGJveC5PcHRpb24gdmFsdWU9e29wdGlvbn0gYXM9e0ZyYWdtZW50fSBkaXNhYmxlZD17b3B0aW9uLmlzRGlzYWJsZWR9IHsuLi5wcm9wc30+XG4gICAgeyh7IGFjdGl2ZSB9KSA9PiB7XG4gICAgICByZXR1cm4gKFxuICAgICAgICA8TWVudUl0ZW1cbiAgICAgICAgICBpc1NlbGVjdGVkPXtpc1NlbGVjdGVkfVxuICAgICAgICAgIGlzQWN0aXZlPXthY3RpdmV9XG4gICAgICAgICAgaXNEaXNhYmxlZD17b3B0aW9uLmlzRGlzYWJsZWR9XG4gICAgICAgICAgaGFzRWxsaXBzaXM9e21lbnVJdGVtc0hhdmVFbGxpcHNpc31cbiAgICAgICAgICBjaGVja1ZhcmlhbnQ9e2FsbG93TXVsdGlwbGUgPyAnY2hlY2tib3gnIDogJ2NoZWNrJ31cbiAgICAgICAgPlxuICAgICAgICAgIHtvcHRpb24uY29udGVudH1cbiAgICAgICAgPC9NZW51SXRlbT5cbiAgICAgICk7XG4gICAgfX1cbiAgPC9MaXN0Ym94Lk9wdGlvbj5cbik7XG5cbmNvbnN0IFNlbGVjdEluZm8gPSAoeyBsYWJlbCwgaGludCB9KSA9PiAoXG4gIDw+XG4gICAgeyhsYWJlbCB8fCBoaW50KSAmJiAoXG4gICAgICA8Qm94IG1hcmdpbkJvdHRvbT1cImJldHdlZW5Gb3JtQ29udHJvbEFuZExhYmVsXCI+XG4gICAgICAgIHtsYWJlbCAmJiA8TGlzdGJveC5MYWJlbCBhcz17RmllbGRMYWJlbH0+e2xhYmVsfTwvTGlzdGJveC5MYWJlbD59XG4gICAgICAgIHtoaW50ICYmIDxGaWVsZEhpbnQ+e2hpbnR9PC9GaWVsZEhpbnQ+fVxuICAgICAgPC9Cb3g+XG4gICAgKX1cbiAgPC8+
|
|
104
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/components/select.jsx"],"names":[],"mappings":"AA+EuB","file":"../../src/components/select.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo } from 'react';\nimport styled from '@emotion/styled';\nimport { Box, Icon, Text, TextButton } from '.';\nimport { IconChevronDown } from '../icons';\nimport { Listbox } from '@headlessui/react';\nimport { getColor, getRadius, getSpace } from '../utilities';\nimport { defaultProps, types } from '../types';\nimport {\n  componentVars,\n  FieldHint,\n  FieldLabel,\n  MenuCard,\n  MenuCardTransition,\n  MenuItem,\n  styles,\n} from '../foundational';\nimport { useMenuPosition } from '../foundational/menu';\nimport { FloatingPortal } from '@floating-ui/react-dom-interactions';\n\nconst TriggerButton = styled.button`\n  ${styles.buttonReset};\n  ${styles.transitions};\n\n  color: inherit;\n  display: flex;\n  align-items: center;\n  gap: 8px;\n  border: none;\n  background: ${getColor('fade1')};\n  padding: 0 ${getSpace('fieldPaddingX')};\n  min-height: ${componentVars.textBoxHeight};\n  border-radius: ${getRadius('s')};\n  width: 100%;\n  text-align: left;\n\n  &:hover {\n    background: ${getColor('fade2')};\n  }\n`;\n\nconst Trigger = forwardRef(({ children, variant = 'box', ...props }, ref) => (\n  <>\n    {variant === 'box' && (\n      <TriggerButton ref={ref} type=\"button\" {...props}>\n        <Text hasEllipsis>{children}</Text>\n        <Box marginLeft=\"auto\">\n          <Icon icon={<IconChevronDown />} />\n        </Box>\n      </TriggerButton>\n    )}\n    {variant === 'text' && (\n      <TextButton ref={ref} icon={<IconChevronDown />} iconPosition=\"right\" {...props}>\n        {children}\n      </TextButton>\n    )}\n  </>\n));\n\nconst getTriggerContent = (isMultiple, selected) => {\n  if (!selected) return null;\n  if (isMultiple) {\n    return selected.map(option => option.content).join(', ');\n  }\n  return selected.content;\n};\n\nconst SelectButton = ({ customTrigger, variant, isMultiple, selected }) => {\n  const props = {\n    variant,\n    children: getTriggerContent(isMultiple, selected),\n  };\n  return (\n    <Listbox.Button as={Fragment}>\n      {customTrigger ? customTrigger({ props }) : <Trigger {...props} />}\n    </Listbox.Button>\n  );\n};\n\nconst Root = styled.div`\n  min-width: 0px;\n`;\n\nconst SelectMenuOption = ({\n  option,\n  isSelected,\n  menuItemsHaveEllipsis,\n  allowMultiple,\n  ...props\n}) => (\n  <Listbox.Option value={option} as={Fragment} disabled={option.isDisabled} {...props}>\n    {({ active }) => {\n      return (\n        <MenuItem\n          isSelected={isSelected}\n          isActive={active}\n          isDisabled={option.isDisabled}\n          hasEllipsis={menuItemsHaveEllipsis}\n          checkVariant={allowMultiple ? 'checkbox' : 'check'}\n        >\n          {option.content}\n        </MenuItem>\n      );\n    }}\n  </Listbox.Option>\n);\n\nconst SelectInfo = ({ label, hint }) => (\n  <>\n    {(label || hint) && (\n      <Box marginBottom=\"betweenFormControlAndLabel\">\n        {label && <Listbox.Label as={FieldLabel}>{label}</Listbox.Label>}\n        {hint && <FieldHint>{hint}</FieldHint>}\n      </Box>\n    )}\n  </>\n);\n\nconst SelectMenu = ({ children, floating, hasPortal, isOpen, ...props }) => {\n  const Root = hasPortal ? FloatingPortal : Fragment;\n  return (\n    <Root>\n      <MenuCardTransition>\n        <Listbox.Options\n          static\n          className=\"grn-context\"\n          ref={floating}\n          as={MenuCard}\n          isOpen={isOpen}\n          {...props}\n        >\n          {children}\n        </Listbox.Options>\n      </MenuCardTransition>\n    </Root>\n  );\n};\n\nconst getIsSelected = (isMultiple, option, selected) => {\n  if (!selected) return null;\n  if (isMultiple) {\n    return selected.some(selectedOption => selectedOption.content === option.content);\n  }\n\n  return option.value === selected.value;\n};\n\nexport const Select = ({\n  options,\n  value,\n  onChange,\n  menuPlacement = defaultProps.menuPlacement,\n  menuWidth,\n  menuMaxHeight = defaultProps.menuMaxHeight,\n  trigger: customTrigger,\n  triggerVariant = 'box',\n  label,\n  hint,\n  menuZIndex,\n  menuItemsHaveEllipsis = true,\n  allowMultiple,\n  hasPortal = true,\n}) => {\n  const { reference, floating, floatingStyles } = useMenuPosition({ menuWidth, menuPlacement });\n  const rootWidth = triggerVariant === 'text' ? 'fit-content' : undefined;\n\n  const optionMap = useMemo(() => {\n    return new Map(options.map(option => [option.value, option]));\n  }, [options]);\n\n  const selected = useMemo(\n    () => (allowMultiple ? value.map(value => optionMap.get(value)) : optionMap.get(value)),\n    [allowMultiple, optionMap, value],\n  );\n\n  return (\n    <Listbox\n      as={Root}\n      value={selected}\n      onChange={onChange}\n      width={rootWidth}\n      multiple={allowMultiple}\n    >\n      {({ open }) => (\n        <>\n          <SelectInfo label={label} hint={hint} />\n\n          <Box position=\"relative\" ref={reference}>\n            <SelectButton\n              customTrigger={customTrigger}\n              variant={triggerVariant}\n              isMultiple={allowMultiple}\n              selected={selected}\n            />\n\n            <SelectMenu\n              placement={menuPlacement}\n              maxHeight={menuMaxHeight}\n              zIndex={menuZIndex}\n              floating={floating}\n              style={floatingStyles}\n              hasPortal={hasPortal}\n              isOpen={open}\n            >\n              {options.map((option, index) => {\n                const isSelected = getIsSelected(allowMultiple, option, selected);\n                return (\n                  <SelectMenuOption\n                    isSelected={isSelected}\n                    key={index}\n                    option={option}\n                    hasEllipsis={menuItemsHaveEllipsis}\n                    allowMultiple={allowMultiple}\n                  />\n                );\n              })}\n            </SelectMenu>\n          </Box>\n        </>\n      )}\n    </Listbox>\n  );\n};\n\nSelect.Label = FieldLabel;\nSelect.Hint = FieldHint;\nSelect.Trigger = Trigger;\n\nSelect.propTypes = {\n  options: PropTypes.array,\n  value: PropTypes.oneOfType([PropTypes.string, PropTypes.array]).isRequired,\n  onChange: PropTypes.func.isRequired,\n  menuPlacement: types.menuPlacement,\n  menuWidth: types.dimension,\n  menuZIndex: types.zIndex,\n  menuMaxHeight: types.dimension,\n  trigger: PropTypes.func,\n  triggerVariant: PropTypes.oneOf(['box', 'text']),\n  label: types.label,\n  hint: types.hint,\n  menuItemsHaveEllipsis: PropTypes.bool,\n  allowMultiple: PropTypes.bool,\n  hasPortal: PropTypes.bool,\n};\n"]} */",
|
|
105
105
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
106
106
|
});
|
|
107
107
|
|
|
@@ -142,6 +142,7 @@ var SelectMenu = function SelectMenu(_ref6) {
|
|
|
142
142
|
var children = _ref6.children,
|
|
143
143
|
floating = _ref6.floating,
|
|
144
144
|
hasPortal = _ref6.hasPortal,
|
|
145
|
+
isOpen = _ref6.isOpen,
|
|
145
146
|
props = _objectWithoutProperties(_ref6, _excluded3);
|
|
146
147
|
|
|
147
148
|
var Root = hasPortal ? FloatingPortal : Fragment;
|
|
@@ -149,7 +150,8 @@ var SelectMenu = function SelectMenu(_ref6) {
|
|
|
149
150
|
static: true,
|
|
150
151
|
className: "grn-context",
|
|
151
152
|
ref: floating,
|
|
152
|
-
as: MenuCard
|
|
153
|
+
as: MenuCard,
|
|
154
|
+
isOpen: isOpen
|
|
153
155
|
}, props), children)));
|
|
154
156
|
};
|
|
155
157
|
|
|
@@ -211,34 +213,38 @@ export var Select = function Select(_ref7) {
|
|
|
211
213
|
onChange: onChange,
|
|
212
214
|
width: rootWidth,
|
|
213
215
|
multiple: allowMultiple
|
|
214
|
-
},
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
216
|
+
}, function (_ref8) {
|
|
217
|
+
var open = _ref8.open;
|
|
218
|
+
return ___EmotionJSX(React.Fragment, null, ___EmotionJSX(SelectInfo, {
|
|
219
|
+
label: label,
|
|
220
|
+
hint: hint
|
|
221
|
+
}), ___EmotionJSX(Box, {
|
|
222
|
+
position: "relative",
|
|
223
|
+
ref: reference
|
|
224
|
+
}, ___EmotionJSX(SelectButton, {
|
|
225
|
+
customTrigger: customTrigger,
|
|
226
|
+
variant: triggerVariant,
|
|
227
|
+
isMultiple: allowMultiple,
|
|
228
|
+
selected: selected
|
|
229
|
+
}), ___EmotionJSX(SelectMenu, {
|
|
230
|
+
placement: menuPlacement,
|
|
231
|
+
maxHeight: menuMaxHeight,
|
|
232
|
+
zIndex: menuZIndex,
|
|
233
|
+
floating: floating,
|
|
234
|
+
style: floatingStyles,
|
|
235
|
+
hasPortal: hasPortal,
|
|
236
|
+
isOpen: open
|
|
237
|
+
}, options.map(function (option, index) {
|
|
238
|
+
var isSelected = getIsSelected(allowMultiple, option, selected);
|
|
239
|
+
return ___EmotionJSX(SelectMenuOption, {
|
|
240
|
+
isSelected: isSelected,
|
|
241
|
+
key: index,
|
|
242
|
+
option: option,
|
|
243
|
+
hasEllipsis: menuItemsHaveEllipsis,
|
|
244
|
+
allowMultiple: allowMultiple
|
|
245
|
+
});
|
|
246
|
+
}))));
|
|
247
|
+
});
|
|
242
248
|
};
|
|
243
249
|
Select.Label = FieldLabel;
|
|
244
250
|
Select.Hint = FieldHint;
|
package/es/foundational/menu.js
CHANGED
|
@@ -2,7 +2,7 @@ import "core-js/modules/es.object.keys.js";
|
|
|
2
2
|
import "core-js/modules/es.array.index-of.js";
|
|
3
3
|
import "core-js/modules/es.symbol.js";
|
|
4
4
|
var _excluded = ["children"],
|
|
5
|
-
_excluded2 = ["children", "zIndex", "placement", "maxHeight"],
|
|
5
|
+
_excluded2 = ["children", "zIndex", "placement", "maxHeight", "isOpen"],
|
|
6
6
|
_excluded3 = ["children", "isDisabled", "hasIcon", "isSelected", "color", "columns"],
|
|
7
7
|
_excluded4 = ["children", "isSelected", "isActive", "isDisabled", "icon", "variant", "hasEllipsis", "checkVariant"];
|
|
8
8
|
|
|
@@ -83,7 +83,7 @@ export var useMenuPosition = function useMenuPosition(_ref) {
|
|
|
83
83
|
floatingStyles: floatingStyles
|
|
84
84
|
};
|
|
85
85
|
};
|
|
86
|
-
var menuCardStyles = /*#__PURE__*/css(".menuCardLeaveTo,.menuCardEnterFrom{opacity:0;}.menuCardEnterTo{transition-duration:", getTransition('xxFast'), ";}.menuCardLeaveTo{transition-duration:", getTransition('xFast'), ";}" + (process.env.NODE_ENV === "production" ? "" : ";label:menuCardStyles;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9mb3VuZGF0aW9uYWwvbWVudS5qc3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBK0QwQiIsImZpbGUiOiIuLi8uLi9zcmMvZm91bmRhdGlvbmFsL21lbnUuanN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFJlYWN0LCB7IEZyYWdtZW50LCBmb3J3YXJkUmVmIH0gZnJvbSAncmVhY3QnO1xuaW1wb3J0IFByb3BUeXBlcyBmcm9tICdwcm9wLXR5cGVzJztcbmltcG9ydCB7IEdsb2JhbCwgY3NzIH0gZnJvbSAnQGVtb3Rpb24vcmVhY3QnO1xuaW1wb3J0IHsgZ2V0Q29sb3IsIGdldERpbWVuc2lvbiwgZ2V0UmFkaXVzLCBnZXRTaGFkb3csIGdldFRyYW5zaXRpb24gfSBmcm9tICcuLi91dGlsaXRpZXMnO1xuaW1wb3J0IHsgQm94LCBDaGVja2JveCwgSWNvbiwgVGV4dCB9IGZyb20gJy4uL2NvbXBvbmVudHMnO1xuaW1wb3J0IHsgSWNvbkNoZWNrIH0gZnJvbSAnLi4vaWNvbnMnO1xuaW1wb3J0IHsgY29tcG9uZW50VmFycyB9IGZyb20gJy4vc3R5bGVzJztcbmltcG9ydCB7IGF1dG9VcGRhdGUsIGZsaXAsIHNpemUsIHVzZUZsb2F0aW5nIH0gZnJvbSAnQGZsb2F0aW5nLXVpL3JlYWN0LWRvbS1pbnRlcmFjdGlvbnMnO1xuaW1wb3J0IHsgVHJhbnNpdGlvbiB9IGZyb20gJ0BoZWFkbGVzc3VpL3JlYWN0JztcblxuY29uc3QgY2FyZFBhZGRpbmcgPSA4O1xuY29uc3QgY2FyZFBhZGRpbmdQeCA9IGAke2NhcmRQYWRkaW5nfXB4YDtcbmNvbnN0IGl0ZW1ZUGFkZGluZyA9IDEwO1xuY29uc3QgaXRlbVhQYWRkaW5nID0gMTI7XG5cbmV4cG9ydCBjb25zdCBwbGFjZW1lbnRzTWFwID0ge1xuICB0b3A6ICd0b3AnLFxuICB0b3BTdGFydDogJ3RvcC1zdGFydCcsXG4gIHRvcEVuZDogJ3RvcC1lbmQnLFxuICBib3R0b206ICdib3R0b20nLFxuICBib3R0b21TdGFydDogJ2JvdHRvbS1zdGFydCcsXG4gIGJvdHRvbUVuZDogJ2JvdHRvbS1lbmQnLFxuICByaWdodDogJ3JpZ2h0JyxcbiAgcmlnaHRTdGFydDogJ3JpZ2h0LXN0YXJ0JyxcbiAgcmlnaHRFbmQ6ICdyaWdodC1lbmQnLFxuICBsZWZ0OiAnbGVmdCcsXG4gIGxlZnRTdGFydDogJ2xlZnQtc3RhcnQnLFxuICBsZWZ0RW5kOiAnbGVmdC1lbmQnLFxufTtcblxuZXhwb3J0IGNvbnN0IHVzZU1lbnVQb3NpdGlvbiA9ICh7IG1lbnVXaWR0aCwgbWVudVBsYWNlbWVudCB9KSA9PiB7XG4gIGNvbnN0IHsgeCwgeSwgcmVmZXJlbmNlLCBmbG9hdGluZywgc3RyYXRlZ3ksIHJlZnMgfSA9IHVzZUZsb2F0aW5nKHtcbiAgICBwbGFjZW1lbnQ6IHBsYWNlbWVudHNNYXBbbWVudVBsYWNlbWVudF0sXG4gICAgd2hpbGVFbGVtZW50c01vdW50ZWQ6IGF1dG9VcGRhdGUsXG4gICAgbWlkZGxld2FyZTogW1xuICAgICAgZmxpcCh7IHBhZGRpbmc6IDI0LCBmbGlwQWxpZ25tZW50OiBmYWxzZSB9KSxcbiAgICAgIHNpemUoe1xuICAgICAgICBhcHBseSh7IHJlY3RzLCBlbGVtZW50cyB9KSB7XG4gICAgICAgICAgY29uc3Qgd2lkdGggPSBnZXREaW1lbnNpb24obWVudVdpZHRoKSB8fCBgJHtyZWN0cy5yZWZlcmVuY2Uud2lkdGh9cHhgO1xuICAgICAgICAgIE9iamVjdC5hc3NpZ24oZWxlbWVudHMuZmxvYXRpbmcuc3R5bGUsIHtcbiAgICAgICAgICAgIHdpZHRoOiB3aWR0aCxcbiAgICAgICAgICB9KTtcbiAgICAgICAgfSxcbiAgICAgICAgcGFkZGluZzogMjQsXG4gICAgICB9KSxcbiAgICBdLFxuICB9KTtcblxuICBjb25zdCBmbG9hdGluZ1N0eWxlcyA9IHtcbiAgICBwb3NpdGlvbjogc3RyYXRlZ3ksXG4gICAgdG9wOiB5ID8/IDAsXG4gICAgbGVmdDogeCA/PyAwLFxuICAgIHpJbmRleDogMTEwMCxcbiAgfTtcblxuICByZXR1cm4ge1xuICAgIHJlZnMsXG4gICAgcmVmZXJlbmNlLFxuICAgIGZsb2F0aW5nLFxuICAgIGZsb2F0aW5nU3R5bGVzLFxuICB9O1xufTtcblxuY29uc3QgbWVudUNhcmRTdHlsZXMgPSBjc3NgXG4gIC5tZW51Q2FyZExlYXZlVG8sXG4gIC5tZW51Q2FyZEVudGVyRnJvbSB7XG4gICAgb3BhY2l0eTogMDtcbiAgfVxuICAubWVudUNhcmRFbnRlclRvIHtcbiAgICB0cmFuc2l0aW9uLWR1cmF0aW9uOiAke2dldFRyYW5zaXRpb24oJ3h4RmFzdCcpfTtcbiAgfVxuICAubWVudUNhcmRMZWF2ZVRvIHtcbiAgICB0cmFuc2l0aW9uLWR1cmF0aW9uOiAke2dldFRyYW5zaXRpb24oJ3hGYXN0Jyl9O1xuICB9XG5gO1xuXG5leHBvcnQgY29uc3QgTWVudUNhcmRUcmFuc2l0aW9uID0gZm9yd2FyZFJlZigoeyBjaGlsZHJlbiwgLi4ucHJvcHMgfSwgcmVmKSA9PiAoXG4gIDxUcmFuc2l0aW9uXG4gICAgYXM9e0ZyYWdtZW50fVxuICAgIGVudGVyRnJvbT1cIm1lbnVDYXJkRW50ZXJGcm9tXCJcbiAgICBlbnRlclRvPVwibWVudUNhcmRFbnRlclRvXCJcbiAgICBsZWF2ZUZyb209XCJtZW51Q2FyZExlYXZlRnJvbVwiXG4gICAgbGVhdmVUbz1cIm1lbnVDYXJkTGVhdmVUb1wiXG4gICAgcmVmPXtyZWZ9XG4gICAgey4uLnByb3BzfVxuICA+XG4gICAge2NoaWxkcmVufVxuICA8L1RyYW5zaXRpb24+
|
|
86
|
+
var menuCardStyles = /*#__PURE__*/css(".menuCardLeaveTo,.menuCardEnterFrom{opacity:0;}.menuCardEnterTo{transition-duration:", getTransition('xxFast'), ";}.menuCardLeaveTo{transition-duration:", getTransition('xFast'), ";}" + (process.env.NODE_ENV === "production" ? "" : ";label:menuCardStyles;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AA+D0B","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, { Fragment, forwardRef } from 'react';\nimport PropTypes from 'prop-types';\nimport { Global, css } from '@emotion/react';\nimport { getColor, getDimension, getRadius, getShadow, getTransition } from '../utilities';\nimport { Box, Checkbox, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { componentVars } from './styles';\nimport { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemYPadding = 10;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = css`\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(({ children, ...props }, ref) => (\n  <Transition\n    as={Fragment}\n    enterFrom=\"menuCardEnterFrom\"\n    enterTo=\"menuCardEnterTo\"\n    leaveFrom=\"menuCardLeaveFrom\"\n    leaveTo=\"menuCardLeaveTo\"\n    ref={ref}\n    {...props}\n  >\n    {children}\n  </Transition>\n));\n\nexport const MenuCard = forwardRef(\n  ({ children, zIndex = 1, placement, maxHeight, isOpen, ...props }, ref) => (\n    <>\n      <Global styles={menuCardStyles} />\n      <ul\n        ref={ref}\n        css={css`\n          padding: ${cardPaddingPx} 0;\n          margin: 0;\n          background: ${getColor('floatingBackground')};\n          z-index: ${zIndex};\n          box-shadow: ${getShadow('m')};\n          border-radius: ${getRadius('m')};\n          overflow: auto;\n          max-height: ${maxHeight};\n          /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n          ${!isOpen && `pointer-events: none;`}\n\n          &:focus {\n            outline: none;\n          }\n        `}\n        {...props}\n      >\n        {children}\n      </ul>\n    </>\n  ),\n);\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('fade1'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('danger'),\n    iconColor: componentVars.dangerIconColor,\n    iconColorActive: getColor('danger'),\n    backgroundColorActive: componentVars.clearButtonDangerBackgroundHover,\n  },\n};\n\nconst MenuItemRoot = forwardRef(\n  ({ children, isDisabled, hasIcon, isSelected, color, columns, ...props }, ref) => (\n    <li\n      ref={ref}\n      css={css`\n        display: grid;\n        grid-auto-flow: column;\n        gap: 8px;\n        ${isDisabled && `color: ${getColor('disabledContent')}`};\n        ${!isDisabled && `cursor: pointer;`};\n        align-items: center;\n        justify-content: start;\n        list-style: none;\n        padding: ${itemYPadding}px ${itemXPadding}px;\n        margin: 0 ${cardPadding}px;\n        min-height: ${componentVars.textBoxHeight};\n        border-radius: ${getRadius('s')};\n        appearance: none;\n        color: ${color};\n      `}\n      {...props}\n    >\n      {children}\n    </li>\n  ),\n);\n\nexport const MenuGroupTitle = ({ children, hasDivider }) => (\n  <Box\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingBottom={cardPaddingPx}\n    marginTop={cardPaddingPx}\n    paddingTop={hasDivider ? `${cardPadding * 2}px` : undefined}\n    borderSide={hasDivider ? 'top' : undefined}\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height={componentVars.dividerStrokeSize} marginY={1} backgroundColor=\"border\" />\n);\n\nexport const MenuItem = forwardRef(\n  (\n    {\n      children,\n      isSelected,\n      isActive,\n      isDisabled,\n      icon,\n      variant = 'neutral',\n      hasEllipsis,\n      checkVariant = 'check',\n      ...props\n    },\n    ref,\n  ) => {\n    const hasCheck = checkVariant === 'check';\n    const hasCheckbox = checkVariant === 'checkbox';\n\n    const getColumns = () => {\n      if (Boolean(icon) || hasCheckbox) return 'auto 1fr';\n      if (hasCheck) return '1fr auto';\n    };\n\n    return (\n      <MenuItemRoot\n        ref={ref}\n        isSelected={isSelected}\n        isDisabled={isDisabled}\n        color={variantStyles[variant].color}\n        hasIcon={Boolean(icon)}\n        style={{\n          background: isActive && variantStyles[variant].backgroundColorActive,\n          gridTemplateColumns: getColumns(),\n        }}\n        {...props}\n      >\n        {hasCheckbox && <Checkbox.Box checked={isSelected} disabled={isDisabled} />}\n        {icon && (\n          <Icon\n            icon={icon}\n            hasEvenBoundary\n            color={\n              isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n            }\n          />\n        )}\n        <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n          <Text hasEllipsis={hasEllipsis}>{children}</Text>\n        </Box>\n        {hasCheck && (\n          <Box opacity={isSelected ? '1' : '0'}>\n            <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n          </Box>\n        )}\n      </MenuItemRoot>\n    );\n  },\n);\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'checkbox', 'none']),\n};\n"]} */");
|
|
87
87
|
export var MenuCardTransition = /*#__PURE__*/forwardRef(function (_ref3, ref) {
|
|
88
88
|
var children = _ref3.children,
|
|
89
89
|
props = _objectWithoutProperties(_ref3, _excluded);
|
|
@@ -103,13 +103,14 @@ export var MenuCard = /*#__PURE__*/forwardRef(function (_ref4, ref) {
|
|
|
103
103
|
zIndex = _ref4$zIndex === void 0 ? 1 : _ref4$zIndex,
|
|
104
104
|
placement = _ref4.placement,
|
|
105
105
|
maxHeight = _ref4.maxHeight,
|
|
106
|
+
isOpen = _ref4.isOpen,
|
|
106
107
|
props = _objectWithoutProperties(_ref4, _excluded2);
|
|
107
108
|
|
|
108
109
|
return ___EmotionJSX(React.Fragment, null, ___EmotionJSX(Global, {
|
|
109
110
|
styles: menuCardStyles
|
|
110
111
|
}), ___EmotionJSX("ul", _extends({
|
|
111
112
|
ref: ref,
|
|
112
|
-
css: /*#__PURE__*/css("padding:", cardPaddingPx, " 0;margin:0;background:", getColor('floatingBackground'), ";z-index:", zIndex, ";box-shadow:", getShadow('m'), ";border-radius:", getRadius('m'), ";overflow:auto;max-height:", maxHeight, "
|
|
113
|
+
css: /*#__PURE__*/css("padding:", cardPaddingPx, " 0;margin:0;background:", getColor('floatingBackground'), ";z-index:", zIndex, ";box-shadow:", getShadow('m'), ";border-radius:", getRadius('m'), ";overflow:auto;max-height:", maxHeight, ";", !isOpen && "pointer-events: none;", " &:focus{outline:none;}" + (process.env.NODE_ENV === "production" ? "" : ";label:MenuCard;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AAgGgB","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, { Fragment, forwardRef } from 'react';\nimport PropTypes from 'prop-types';\nimport { Global, css } from '@emotion/react';\nimport { getColor, getDimension, getRadius, getShadow, getTransition } from '../utilities';\nimport { Box, Checkbox, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { componentVars } from './styles';\nimport { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemYPadding = 10;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = css`\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(({ children, ...props }, ref) => (\n  <Transition\n    as={Fragment}\n    enterFrom=\"menuCardEnterFrom\"\n    enterTo=\"menuCardEnterTo\"\n    leaveFrom=\"menuCardLeaveFrom\"\n    leaveTo=\"menuCardLeaveTo\"\n    ref={ref}\n    {...props}\n  >\n    {children}\n  </Transition>\n));\n\nexport const MenuCard = forwardRef(\n  ({ children, zIndex = 1, placement, maxHeight, isOpen, ...props }, ref) => (\n    <>\n      <Global styles={menuCardStyles} />\n      <ul\n        ref={ref}\n        css={css`\n          padding: ${cardPaddingPx} 0;\n          margin: 0;\n          background: ${getColor('floatingBackground')};\n          z-index: ${zIndex};\n          box-shadow: ${getShadow('m')};\n          border-radius: ${getRadius('m')};\n          overflow: auto;\n          max-height: ${maxHeight};\n          /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n          ${!isOpen && `pointer-events: none;`}\n\n          &:focus {\n            outline: none;\n          }\n        `}\n        {...props}\n      >\n        {children}\n      </ul>\n    </>\n  ),\n);\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('fade1'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('danger'),\n    iconColor: componentVars.dangerIconColor,\n    iconColorActive: getColor('danger'),\n    backgroundColorActive: componentVars.clearButtonDangerBackgroundHover,\n  },\n};\n\nconst MenuItemRoot = forwardRef(\n  ({ children, isDisabled, hasIcon, isSelected, color, columns, ...props }, ref) => (\n    <li\n      ref={ref}\n      css={css`\n        display: grid;\n        grid-auto-flow: column;\n        gap: 8px;\n        ${isDisabled && `color: ${getColor('disabledContent')}`};\n        ${!isDisabled && `cursor: pointer;`};\n        align-items: center;\n        justify-content: start;\n        list-style: none;\n        padding: ${itemYPadding}px ${itemXPadding}px;\n        margin: 0 ${cardPadding}px;\n        min-height: ${componentVars.textBoxHeight};\n        border-radius: ${getRadius('s')};\n        appearance: none;\n        color: ${color};\n      `}\n      {...props}\n    >\n      {children}\n    </li>\n  ),\n);\n\nexport const MenuGroupTitle = ({ children, hasDivider }) => (\n  <Box\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingBottom={cardPaddingPx}\n    marginTop={cardPaddingPx}\n    paddingTop={hasDivider ? `${cardPadding * 2}px` : undefined}\n    borderSide={hasDivider ? 'top' : undefined}\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height={componentVars.dividerStrokeSize} marginY={1} backgroundColor=\"border\" />\n);\n\nexport const MenuItem = forwardRef(\n  (\n    {\n      children,\n      isSelected,\n      isActive,\n      isDisabled,\n      icon,\n      variant = 'neutral',\n      hasEllipsis,\n      checkVariant = 'check',\n      ...props\n    },\n    ref,\n  ) => {\n    const hasCheck = checkVariant === 'check';\n    const hasCheckbox = checkVariant === 'checkbox';\n\n    const getColumns = () => {\n      if (Boolean(icon) || hasCheckbox) return 'auto 1fr';\n      if (hasCheck) return '1fr auto';\n    };\n\n    return (\n      <MenuItemRoot\n        ref={ref}\n        isSelected={isSelected}\n        isDisabled={isDisabled}\n        color={variantStyles[variant].color}\n        hasIcon={Boolean(icon)}\n        style={{\n          background: isActive && variantStyles[variant].backgroundColorActive,\n          gridTemplateColumns: getColumns(),\n        }}\n        {...props}\n      >\n        {hasCheckbox && <Checkbox.Box checked={isSelected} disabled={isDisabled} />}\n        {icon && (\n          <Icon\n            icon={icon}\n            hasEvenBoundary\n            color={\n              isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n            }\n          />\n        )}\n        <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n          <Text hasEllipsis={hasEllipsis}>{children}</Text>\n        </Box>\n        {hasCheck && (\n          <Box opacity={isSelected ? '1' : '0'}>\n            <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n          </Box>\n        )}\n      </MenuItemRoot>\n    );\n  },\n);\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'checkbox', 'none']),\n};\n"]} */")
|
|
113
114
|
}, props), children));
|
|
114
115
|
});
|
|
115
116
|
var variantStyles = {
|
|
@@ -135,7 +136,7 @@ var MenuItemRoot = /*#__PURE__*/forwardRef(function (_ref5, ref) {
|
|
|
135
136
|
|
|
136
137
|
return ___EmotionJSX("li", _extends({
|
|
137
138
|
ref: ref,
|
|
138
|
-
css: /*#__PURE__*/css("display:grid;grid-auto-flow:column;gap:8px;", isDisabled && "color: ".concat(getColor('disabledContent')), ";", !isDisabled && "cursor: pointer;", ";align-items:center;justify-content:start;list-style:none;padding:", itemYPadding, "px ", itemXPadding, "px;margin:0 ", cardPadding, "px;min-height:", componentVars.textBoxHeight, ";border-radius:", getRadius('s'), ";appearance:none;color:", color, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MenuItemRoot;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
139
|
+
css: /*#__PURE__*/css("display:grid;grid-auto-flow:column;gap:8px;", isDisabled && "color: ".concat(getColor('disabledContent')), ";", !isDisabled && "cursor: pointer;", ";align-items:center;justify-content:start;list-style:none;padding:", itemYPadding, "px ", itemXPadding, "px;margin:0 ", cardPadding, "px;min-height:", componentVars.textBoxHeight, ";border-radius:", getRadius('s'), ";appearance:none;color:", color, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:MenuItemRoot;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/foundational/menu.jsx"],"names":[],"mappings":"AAyIc","file":"../../src/foundational/menu.jsx","sourcesContent":["import React, { Fragment, forwardRef } from 'react';\nimport PropTypes from 'prop-types';\nimport { Global, css } from '@emotion/react';\nimport { getColor, getDimension, getRadius, getShadow, getTransition } from '../utilities';\nimport { Box, Checkbox, Icon, Text } from '../components';\nimport { IconCheck } from '../icons';\nimport { componentVars } from './styles';\nimport { autoUpdate, flip, size, useFloating } from '@floating-ui/react-dom-interactions';\nimport { Transition } from '@headlessui/react';\n\nconst cardPadding = 8;\nconst cardPaddingPx = `${cardPadding}px`;\nconst itemYPadding = 10;\nconst itemXPadding = 12;\n\nexport const placementsMap = {\n  top: 'top',\n  topStart: 'top-start',\n  topEnd: 'top-end',\n  bottom: 'bottom',\n  bottomStart: 'bottom-start',\n  bottomEnd: 'bottom-end',\n  right: 'right',\n  rightStart: 'right-start',\n  rightEnd: 'right-end',\n  left: 'left',\n  leftStart: 'left-start',\n  leftEnd: 'left-end',\n};\n\nexport const useMenuPosition = ({ menuWidth, menuPlacement }) => {\n  const { x, y, reference, floating, strategy, refs } = useFloating({\n    placement: placementsMap[menuPlacement],\n    whileElementsMounted: autoUpdate,\n    middleware: [\n      flip({ padding: 24, flipAlignment: false }),\n      size({\n        apply({ rects, elements }) {\n          const width = getDimension(menuWidth) || `${rects.reference.width}px`;\n          Object.assign(elements.floating.style, {\n            width: width,\n          });\n        },\n        padding: 24,\n      }),\n    ],\n  });\n\n  const floatingStyles = {\n    position: strategy,\n    top: y ?? 0,\n    left: x ?? 0,\n    zIndex: 1100,\n  };\n\n  return {\n    refs,\n    reference,\n    floating,\n    floatingStyles,\n  };\n};\n\nconst menuCardStyles = css`\n  .menuCardLeaveTo,\n  .menuCardEnterFrom {\n    opacity: 0;\n  }\n  .menuCardEnterTo {\n    transition-duration: ${getTransition('xxFast')};\n  }\n  .menuCardLeaveTo {\n    transition-duration: ${getTransition('xFast')};\n  }\n`;\n\nexport const MenuCardTransition = forwardRef(({ children, ...props }, ref) => (\n  <Transition\n    as={Fragment}\n    enterFrom=\"menuCardEnterFrom\"\n    enterTo=\"menuCardEnterTo\"\n    leaveFrom=\"menuCardLeaveFrom\"\n    leaveTo=\"menuCardLeaveTo\"\n    ref={ref}\n    {...props}\n  >\n    {children}\n  </Transition>\n));\n\nexport const MenuCard = forwardRef(\n  ({ children, zIndex = 1, placement, maxHeight, isOpen, ...props }, ref) => (\n    <>\n      <Global styles={menuCardStyles} />\n      <ul\n        ref={ref}\n        css={css`\n          padding: ${cardPaddingPx} 0;\n          margin: 0;\n          background: ${getColor('floatingBackground')};\n          z-index: ${zIndex};\n          box-shadow: ${getShadow('m')};\n          border-radius: ${getRadius('m')};\n          overflow: auto;\n          max-height: ${maxHeight};\n          /* to prevent a bug: the menu is not always rmoeved from the DOM, in a animated container */\n          ${!isOpen && `pointer-events: none;`}\n\n          &:focus {\n            outline: none;\n          }\n        `}\n        {...props}\n      >\n        {children}\n      </ul>\n    </>\n  ),\n);\n\nconst variantStyles = {\n  neutral: {\n    backgroundColorActive: getColor('fade1'),\n    iconColorActive: getColor('content'),\n  },\n  danger: {\n    color: getColor('danger'),\n    iconColor: componentVars.dangerIconColor,\n    iconColorActive: getColor('danger'),\n    backgroundColorActive: componentVars.clearButtonDangerBackgroundHover,\n  },\n};\n\nconst MenuItemRoot = forwardRef(\n  ({ children, isDisabled, hasIcon, isSelected, color, columns, ...props }, ref) => (\n    <li\n      ref={ref}\n      css={css`\n        display: grid;\n        grid-auto-flow: column;\n        gap: 8px;\n        ${isDisabled && `color: ${getColor('disabledContent')}`};\n        ${!isDisabled && `cursor: pointer;`};\n        align-items: center;\n        justify-content: start;\n        list-style: none;\n        padding: ${itemYPadding}px ${itemXPadding}px;\n        margin: 0 ${cardPadding}px;\n        min-height: ${componentVars.textBoxHeight};\n        border-radius: ${getRadius('s')};\n        appearance: none;\n        color: ${color};\n      `}\n      {...props}\n    >\n      {children}\n    </li>\n  ),\n);\n\nexport const MenuGroupTitle = ({ children, hasDivider }) => (\n  <Box\n    paddingX={`${cardPadding + itemXPadding}px`}\n    paddingBottom={cardPaddingPx}\n    marginTop={cardPaddingPx}\n    paddingTop={hasDivider ? `${cardPadding * 2}px` : undefined}\n    borderSide={hasDivider ? 'top' : undefined}\n  >\n    <Text variant=\"caps\">{children}</Text>\n  </Box>\n);\n\nexport const MenuItemDivider = () => (\n  <Box height={componentVars.dividerStrokeSize} marginY={1} backgroundColor=\"border\" />\n);\n\nexport const MenuItem = forwardRef(\n  (\n    {\n      children,\n      isSelected,\n      isActive,\n      isDisabled,\n      icon,\n      variant = 'neutral',\n      hasEllipsis,\n      checkVariant = 'check',\n      ...props\n    },\n    ref,\n  ) => {\n    const hasCheck = checkVariant === 'check';\n    const hasCheckbox = checkVariant === 'checkbox';\n\n    const getColumns = () => {\n      if (Boolean(icon) || hasCheckbox) return 'auto 1fr';\n      if (hasCheck) return '1fr auto';\n    };\n\n    return (\n      <MenuItemRoot\n        ref={ref}\n        isSelected={isSelected}\n        isDisabled={isDisabled}\n        color={variantStyles[variant].color}\n        hasIcon={Boolean(icon)}\n        style={{\n          background: isActive && variantStyles[variant].backgroundColorActive,\n          gridTemplateColumns: getColumns(),\n        }}\n        {...props}\n      >\n        {hasCheckbox && <Checkbox.Box checked={isSelected} disabled={isDisabled} />}\n        {icon && (\n          <Icon\n            icon={icon}\n            hasEvenBoundary\n            color={\n              isActive ? variantStyles[variant].iconColorActive : variantStyles[variant].iconColor\n            }\n          />\n        )}\n        <Box width=\"100%\" minWidth=\"0px\" tag=\"span\">\n          <Text hasEllipsis={hasEllipsis}>{children}</Text>\n        </Box>\n        {hasCheck && (\n          <Box opacity={isSelected ? '1' : '0'}>\n            <Icon icon={<IconCheck />} color={isActive ? 'content' : 'icon'} />\n          </Box>\n        )}\n      </MenuItemRoot>\n    );\n  },\n);\n\nMenuItem.propTypes = {\n  isSelected: PropTypes.bool,\n  isActive: PropTypes.bool,\n  isDisabled: PropTypes.bool,\n  icon: PropTypes.node,\n  variant: PropTypes.oneOf(['neutral', 'danger']),\n  hasEllipsis: PropTypes.bool,\n  checkVariant: PropTypes.oneOf(['check', 'checkbox', 'none']),\n};\n"]} */")
|
|
139
140
|
}, props), children);
|
|
140
141
|
});
|
|
141
142
|
export var MenuGroupTitle = function MenuGroupTitle(_ref6) {
|