@flodesk/grain 10.4.0 → 10.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -37,7 +37,7 @@ import "core-js/modules/es.string.includes.js";
|
|
|
37
37
|
import "core-js/modules/es.array.find.js";
|
|
38
38
|
import "core-js/modules/es.array.map.js";
|
|
39
39
|
import PropTypes from 'prop-types';
|
|
40
|
-
import React, { forwardRef, Fragment, useMemo, useState } from 'react';
|
|
40
|
+
import React, { forwardRef, Fragment, useMemo, useRef, useState } from 'react';
|
|
41
41
|
import { Icon, Box, Text } from '.';
|
|
42
42
|
import { IconCheck, IconChevronDown } from '../icons';
|
|
43
43
|
import { Combobox } from '@headlessui/react';
|
|
@@ -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:grid;grid-template-columns:1fr auto;gap:8px;align-items:center;list-style:none;padding:10px 12px;margin:0 8px;min-height:", componentVars.textBoxHeight, ";border-radius:", getRadius('s'), ";appearance:none;}.autocompleteMenuItemText.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":"AAqD0B","file":"../../src/components/autocomplete.jsx","sourcesContent":["import PropTypes from 'prop-types';\nimport React, { forwardRef, Fragment, useMemo, 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  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: grid;\n    grid-template-columns: 1fr auto;\n    gap: 8px;\n    align-items: center;\n    list-style: none;\n    padding: 10px 12px;\n    margin: 0 8px;\n    min-height: ${componentVars.textBoxHeight};\n    border-radius: ${getRadius('s')};\n    appearance: none;\n  }\n\n  .autocompleteMenuItemText.hasEllipsis {\n    white-space: nowrap;\n    overflow: hidden;\n    text-overflow: ellipsis;\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  return (\n    <>\n      <Global styles={menuItemStyles} />\n      <Combobox as=\"div\" value={selectedOption || ''} onChange={handleChange} {...props}>\n        {({ open }) => (\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                <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                  hasError={hasError}\n                />\n                <ExpandIcon />\n              </Combobox.Button>\n\n              <OptionsRoot>\n                <Combobox.Options\n                  static\n                  isOpen={open}\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 value={option} as={Fragment} disabled={option.isDisabled}>\n                          {({ active }) => {\n                            const isDisabled = option.isDisabled;\n                            const isSelected = option.value === value;\n                            const isActive = !option.isDisabled && active;\n                            const hasEllipsis = menuItemsHaveEllipsis;\n                            return (\n                              <li\n                                className=\"autocompleteMenuItem\"\n                                style={{\n                                  backgroundColor: isActive && getColor('fade1'),\n                                  color: isDisabled && getColor('disabledContent'),\n                                  cursor: !isDisabled && 'pointer',\n                                }}\n                              >\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              </OptionsRoot>\n            </Box>\n            {errorMessage && (\n              <Box marginTop=\"betweenFormControlAndLabel\" color=\"danger\">\n                {errorMessage}\n              </Box>\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":"AAqD0B","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  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                  <Combobox.Options\n                    static\n                    isOpen={open}\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                </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,
|
|
@@ -144,6 +144,18 @@ export var Autocomplete = function Autocomplete(_ref2) {
|
|
|
144
144
|
floatingStyles = _useMenuPosition.floatingStyles;
|
|
145
145
|
|
|
146
146
|
var OptionsRoot = hasPortal ? FloatingPortal : Fragment;
|
|
147
|
+
var preContentRef = useRef(null);
|
|
148
|
+
|
|
149
|
+
var _useState3 = useState(0),
|
|
150
|
+
_useState4 = _slicedToArray(_useState3, 2),
|
|
151
|
+
preContentWidth = _useState4[0],
|
|
152
|
+
setPreContentWidth = _useState4[1];
|
|
153
|
+
|
|
154
|
+
React.useEffect(function () {
|
|
155
|
+
if (preContentRef.current) {
|
|
156
|
+
setPreContentWidth(preContentRef.current.offsetWidth);
|
|
157
|
+
}
|
|
158
|
+
}, [selectedOption, preContentRef]);
|
|
147
159
|
return ___EmotionJSX(React.Fragment, null, ___EmotionJSX(Global, {
|
|
148
160
|
styles: menuItemStyles
|
|
149
161
|
}), ___EmotionJSX(Combobox, _extends({
|
|
@@ -152,6 +164,7 @@ export var Autocomplete = function Autocomplete(_ref2) {
|
|
|
152
164
|
onChange: handleChange
|
|
153
165
|
}, props), function (_ref3) {
|
|
154
166
|
var open = _ref3.open;
|
|
167
|
+
var hasPreContent = selectedOption === null || selectedOption === void 0 ? void 0 : selectedOption.preContent;
|
|
155
168
|
return ___EmotionJSX(React.Fragment, null, label && ___EmotionJSX(Combobox.Label, {
|
|
156
169
|
as: FieldLabel
|
|
157
170
|
}, label), hint && ___EmotionJSX(FieldHint, null, hint), ___EmotionJSX(Box, {
|
|
@@ -160,7 +173,16 @@ export var Autocomplete = function Autocomplete(_ref2) {
|
|
|
160
173
|
ref: reference
|
|
161
174
|
}, ___EmotionJSX(Combobox.Button, {
|
|
162
175
|
as: Trigger
|
|
163
|
-
}, ___EmotionJSX(
|
|
176
|
+
}, hasPreContent && !open && ___EmotionJSX(Box, {
|
|
177
|
+
ref: preContentRef,
|
|
178
|
+
width: "fit-content",
|
|
179
|
+
position: "absolute",
|
|
180
|
+
top: "0px",
|
|
181
|
+
bottom: "0px",
|
|
182
|
+
margin: "auto",
|
|
183
|
+
height: "fit-content",
|
|
184
|
+
left: "0"
|
|
185
|
+
}, selectedOption.preContent), ___EmotionJSX(Combobox.Input, {
|
|
164
186
|
as: InputField,
|
|
165
187
|
autoComplete: "off",
|
|
166
188
|
onChange: function onChange(event) {
|
|
@@ -171,6 +193,7 @@ export var Autocomplete = function Autocomplete(_ref2) {
|
|
|
171
193
|
return option && option.content;
|
|
172
194
|
},
|
|
173
195
|
paddingRight: "32px",
|
|
196
|
+
paddingLeft: !open && hasPreContent ? "".concat(preContentWidth + 8, "px") : undefined,
|
|
174
197
|
hasError: hasError
|
|
175
198
|
}), ___EmotionJSX(ExpandIcon, null)), ___EmotionJSX(OptionsRoot, null, ___EmotionJSX(Combobox.Options, {
|
|
176
199
|
static: true,
|
|
@@ -198,14 +221,15 @@ export var Autocomplete = function Autocomplete(_ref2) {
|
|
|
198
221
|
var isSelected = option.value === value;
|
|
199
222
|
var isActive = !option.isDisabled && active;
|
|
200
223
|
var hasEllipsis = menuItemsHaveEllipsis;
|
|
224
|
+
var hasPreContent = option.preContent;
|
|
201
225
|
return ___EmotionJSX("li", {
|
|
202
|
-
className:
|
|
226
|
+
className: hasPreContent ? 'autocompleteMenuItem hasPreContent' : 'autocompleteMenuItem',
|
|
203
227
|
style: {
|
|
204
228
|
backgroundColor: isActive && getColor('fade1'),
|
|
205
229
|
color: isDisabled && getColor('disabledContent'),
|
|
206
230
|
cursor: !isDisabled && 'pointer'
|
|
207
231
|
}
|
|
208
|
-
}, ___EmotionJSX("span", {
|
|
232
|
+
}, hasPreContent && option.preContent, ___EmotionJSX("span", {
|
|
209
233
|
className: hasEllipsis ? 'autocompleteMenuItemText hasEllipsis' : 'autocompleteMenuItemText'
|
|
210
234
|
}, option.content), ___EmotionJSX(Icon, {
|
|
211
235
|
style: {
|