@ndla/ui 14.0.0 → 15.0.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.
Files changed (189) hide show
  1. package/es/Article/Article.js +22 -3
  2. package/es/Article/ArticleFavoritesButton.js +38 -0
  3. package/es/Article/index.js +2 -1
  4. package/es/Breadcrumb/ActionBreadcrumb.js +57 -0
  5. package/es/Breadcrumb/index.js +1 -0
  6. package/es/InfoBlock/InfoBlock.js +55 -0
  7. package/es/InfoBlock/index.js +1 -0
  8. package/es/MyNdla/Navigation/VerticalNavigation.js +51 -0
  9. package/es/MyNdla/Navigation/index.js +2 -0
  10. package/es/MyNdla/Resource/Folder.js +86 -0
  11. package/{lib/MyNdla/ResourceDash/ResourcesView.d.ts → es/MyNdla/Resource/index.js} +2 -3
  12. package/es/MyNdla/index.js +3 -4
  13. package/es/Resource/BlockResource.js +73 -0
  14. package/es/Resource/ListResource.js +66 -0
  15. package/es/Resource/index.js +10 -0
  16. package/es/Resource/resourceComponents.js +97 -0
  17. package/es/ResourceGroup/ResourceGroup.js +7 -5
  18. package/es/ResourceGroup/ResourceItem.js +25 -24
  19. package/es/ResourceGroup/ResourceList.js +18 -6
  20. package/es/SnackBar/SnackBar.js +117 -0
  21. package/es/SnackBar/index.js +9 -0
  22. package/es/TagSelector/SuggestionInput.js +240 -0
  23. package/es/TagSelector/Suggestions.js +93 -0
  24. package/es/TagSelector/TagSelector.js +137 -0
  25. package/es/TagSelector/index.js +9 -0
  26. package/es/TreeStructure/FolderItem.js +130 -0
  27. package/es/TreeStructure/FolderItems.js +123 -0
  28. package/es/TreeStructure/FolderNameInput.js +112 -0
  29. package/es/TreeStructure/TreeStructure.js +254 -0
  30. package/es/TreeStructure/TreeStructure.types.js +0 -0
  31. package/es/TreeStructure/TreeStructureWrapper.js +13 -0
  32. package/es/TreeStructure/helperFunctions.js +92 -0
  33. package/es/TreeStructure/index.js +9 -0
  34. package/es/TreeStructure/keyboardNavigation/keyboardNavigation.js +182 -0
  35. package/es/TreeStructure/keyboardNavigation/keyboardNavigation.types.js +0 -0
  36. package/es/all.css +72 -0
  37. package/es/index.js +8 -3
  38. package/es/locale/messages-en.js +62 -4
  39. package/es/locale/messages-nb.js +61 -3
  40. package/es/locale/messages-nn.js +61 -3
  41. package/es/locale/messages-se.js +61 -3
  42. package/es/locale/messages-sma.js +61 -3
  43. package/lib/Article/Article.d.ts +3 -1
  44. package/lib/Article/Article.js +43 -23
  45. package/lib/Article/ArticleFavoritesButton.d.ts +15 -0
  46. package/lib/Article/ArticleFavoritesButton.js +56 -0
  47. package/lib/Article/index.d.ts +2 -1
  48. package/lib/Article/index.js +8 -0
  49. package/lib/Breadcrumb/ActionBreadcrumb.d.ts +16 -0
  50. package/lib/Breadcrumb/ActionBreadcrumb.js +72 -0
  51. package/lib/Breadcrumb/index.d.ts +1 -0
  52. package/lib/Breadcrumb/index.js +8 -0
  53. package/lib/InfoBlock/InfoBlock.d.ts +8 -0
  54. package/lib/InfoBlock/InfoBlock.js +58 -0
  55. package/lib/InfoBlock/index.d.ts +1 -0
  56. package/lib/InfoBlock/index.js +13 -0
  57. package/lib/MyNdla/Navigation/VerticalNavigation.d.ts +10 -0
  58. package/lib/MyNdla/Navigation/VerticalNavigation.js +61 -0
  59. package/lib/MyNdla/Navigation/index.d.ts +2 -0
  60. package/lib/MyNdla/Navigation/index.js +15 -0
  61. package/lib/MyNdla/Resource/Folder.d.ts +20 -0
  62. package/lib/MyNdla/Resource/Folder.js +100 -0
  63. package/lib/MyNdla/Resource/index.d.ts +9 -0
  64. package/lib/MyNdla/Resource/index.js +15 -0
  65. package/lib/MyNdla/index.d.ts +3 -4
  66. package/lib/MyNdla/index.js +9 -11
  67. package/lib/Resource/BlockResource.d.ts +20 -0
  68. package/lib/Resource/BlockResource.js +84 -0
  69. package/lib/Resource/ListResource.d.ts +20 -0
  70. package/lib/Resource/ListResource.js +78 -0
  71. package/lib/Resource/index.d.ts +11 -0
  72. package/lib/Resource/index.js +29 -0
  73. package/lib/Resource/resourceComponents.d.ts +24 -0
  74. package/lib/Resource/resourceComponents.js +106 -0
  75. package/lib/ResourceGroup/ResourceGroup.d.ts +2 -1
  76. package/lib/ResourceGroup/ResourceGroup.js +7 -5
  77. package/lib/ResourceGroup/ResourceItem.d.ts +5 -1
  78. package/lib/ResourceGroup/ResourceItem.js +26 -24
  79. package/lib/ResourceGroup/ResourceList.d.ts +3 -1
  80. package/lib/ResourceGroup/ResourceList.js +18 -6
  81. package/lib/SnackBar/SnackBar.d.ts +23 -0
  82. package/lib/SnackBar/SnackBar.js +127 -0
  83. package/lib/SnackBar/index.d.ts +10 -0
  84. package/lib/SnackBar/index.js +15 -0
  85. package/lib/TagSelector/SuggestionInput.d.ts +19 -0
  86. package/lib/TagSelector/SuggestionInput.js +255 -0
  87. package/lib/TagSelector/Suggestions.d.ts +12 -0
  88. package/lib/TagSelector/Suggestions.js +96 -0
  89. package/lib/TagSelector/TagSelector.d.ts +16 -0
  90. package/lib/TagSelector/TagSelector.js +150 -0
  91. package/lib/TagSelector/index.d.ts +10 -0
  92. package/lib/TagSelector/index.js +19 -0
  93. package/lib/TreeStructure/FolderItem.d.ts +27 -0
  94. package/lib/TreeStructure/FolderItem.js +140 -0
  95. package/lib/TreeStructure/FolderItems.d.ts +11 -0
  96. package/lib/TreeStructure/FolderItems.js +130 -0
  97. package/lib/TreeStructure/FolderNameInput.d.ts +15 -0
  98. package/lib/TreeStructure/FolderNameInput.js +125 -0
  99. package/lib/TreeStructure/TreeStructure.d.ts +12 -0
  100. package/lib/TreeStructure/TreeStructure.js +273 -0
  101. package/lib/TreeStructure/TreeStructure.types.d.ts +63 -0
  102. package/lib/TreeStructure/TreeStructure.types.js +1 -0
  103. package/lib/TreeStructure/TreeStructureWrapper.d.ts +12 -0
  104. package/lib/TreeStructure/TreeStructureWrapper.js +24 -0
  105. package/lib/TreeStructure/helperFunctions.d.ts +5 -0
  106. package/lib/TreeStructure/helperFunctions.js +103 -0
  107. package/lib/TreeStructure/index.d.ts +10 -0
  108. package/lib/TreeStructure/index.js +15 -0
  109. package/lib/TreeStructure/keyboardNavigation/keyboardNavigation.d.ts +11 -0
  110. package/lib/TreeStructure/keyboardNavigation/keyboardNavigation.js +186 -0
  111. package/lib/TreeStructure/keyboardNavigation/keyboardNavigation.types.d.ts +26 -0
  112. package/lib/TreeStructure/keyboardNavigation/keyboardNavigation.types.js +1 -0
  113. package/lib/User/apiTypes.d.ts +1 -1
  114. package/lib/User/index.d.ts +2 -2
  115. package/lib/all.css +72 -0
  116. package/lib/index.d.ts +13 -4
  117. package/lib/index.js +68 -9
  118. package/lib/locale/messages-en.d.ts +58 -0
  119. package/lib/locale/messages-en.js +62 -4
  120. package/lib/locale/messages-nb.d.ts +58 -0
  121. package/lib/locale/messages-nb.js +61 -3
  122. package/lib/locale/messages-nn.d.ts +58 -0
  123. package/lib/locale/messages-nn.js +61 -3
  124. package/lib/locale/messages-se.d.ts +58 -0
  125. package/lib/locale/messages-se.js +61 -3
  126. package/lib/locale/messages-sma.d.ts +58 -0
  127. package/lib/locale/messages-sma.js +61 -3
  128. package/lib/types.d.ts +1 -1
  129. package/package.json +11 -11
  130. package/src/Article/Article.tsx +31 -0
  131. package/src/Article/ArticleFavoritesButton.tsx +40 -0
  132. package/src/Article/index.ts +2 -0
  133. package/src/Breadcrumb/ActionBreadcrumb.tsx +68 -0
  134. package/src/Breadcrumb/index.ts +2 -0
  135. package/src/InfoBlock/InfoBlock.tsx +61 -0
  136. package/src/InfoBlock/index.ts +1 -0
  137. package/src/MyNdla/Navigation/VerticalNavigation.tsx +93 -0
  138. package/src/MyNdla/Navigation/index.ts +2 -0
  139. package/src/MyNdla/Resource/Folder.tsx +143 -0
  140. package/src/MyNdla/Resource/index.ts +10 -0
  141. package/src/MyNdla/index.ts +3 -5
  142. package/src/Resource/BlockResource.tsx +101 -0
  143. package/src/Resource/ListResource.tsx +111 -0
  144. package/src/Resource/index.ts +12 -0
  145. package/src/Resource/resourceComponents.tsx +143 -0
  146. package/src/ResourceGroup/ResourceGroup.tsx +3 -0
  147. package/src/ResourceGroup/ResourceItem.tsx +17 -0
  148. package/src/ResourceGroup/ResourceList.tsx +16 -3
  149. package/src/SnackBar/SnackBar.tsx +183 -0
  150. package/src/SnackBar/index.ts +13 -0
  151. package/src/TagSelector/SuggestionInput.tsx +230 -0
  152. package/src/TagSelector/Suggestions.tsx +125 -0
  153. package/src/TagSelector/TagSelector.tsx +111 -0
  154. package/src/TagSelector/index.ts +13 -0
  155. package/src/TreeStructure/FolderItem.tsx +160 -0
  156. package/src/TreeStructure/FolderItems.tsx +109 -0
  157. package/src/TreeStructure/FolderNameInput.tsx +109 -0
  158. package/src/TreeStructure/TreeStructure.tsx +184 -0
  159. package/src/TreeStructure/TreeStructure.types.ts +69 -0
  160. package/src/TreeStructure/TreeStructureWrapper.tsx +34 -0
  161. package/src/TreeStructure/helperFunctions.ts +52 -0
  162. package/src/TreeStructure/index.ts +11 -0
  163. package/src/TreeStructure/keyboardNavigation/keyboardNavigation.ts +161 -0
  164. package/src/TreeStructure/keyboardNavigation/keyboardNavigation.types.ts +28 -0
  165. package/src/User/apiTypes.ts +1 -1
  166. package/src/User/index.ts +2 -2
  167. package/src/all.scss +1 -0
  168. package/src/index.ts +14 -5
  169. package/src/locale/messages-en.ts +56 -3
  170. package/src/locale/messages-nb.ts +55 -2
  171. package/src/locale/messages-nn.ts +55 -2
  172. package/src/locale/messages-se.ts +55 -2
  173. package/src/locale/messages-sma.ts +55 -2
  174. package/src/types.ts +1 -1
  175. package/es/MyNdla/ResourceDash/Breadcrumbs.js +0 -22
  176. package/es/MyNdla/ResourceDash/ResourceElement.js +0 -27
  177. package/es/MyNdla/ResourceDash/ResourcesView.js +0 -43
  178. package/es/MyNdla/ResourceDash/index.js +0 -4
  179. package/lib/MyNdla/ResourceDash/Breadcrumbs.d.ts +0 -15
  180. package/lib/MyNdla/ResourceDash/Breadcrumbs.js +0 -35
  181. package/lib/MyNdla/ResourceDash/ResourceElement.d.ts +0 -18
  182. package/lib/MyNdla/ResourceDash/ResourceElement.js +0 -38
  183. package/lib/MyNdla/ResourceDash/ResourcesView.js +0 -57
  184. package/lib/MyNdla/ResourceDash/index.d.ts +0 -4
  185. package/lib/MyNdla/ResourceDash/index.js +0 -31
  186. package/src/MyNdla/ResourceDash/Breadcrumbs.tsx +0 -31
  187. package/src/MyNdla/ResourceDash/ResourceElement.tsx +0 -50
  188. package/src/MyNdla/ResourceDash/ResourcesView.tsx +0 -42
  189. package/src/MyNdla/ResourceDash/index.ts +0 -5
@@ -0,0 +1,240 @@
1
+ import _styled from "@emotion/styled-base";
2
+
3
+ function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); }
4
+
5
+ function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); }
6
+
7
+ function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }
8
+
9
+ function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; }
10
+
11
+ function _iterableToArrayLimit(arr, i) { if (typeof Symbol === "undefined" || !(Symbol.iterator in Object(arr))) return; var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"] != null) _i["return"](); } finally { if (_d) throw _e; } } return _arr; }
12
+
13
+ function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
14
+
15
+ function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
16
+
17
+ /*
18
+ * Copyright (c) 2022-present, NDLA.
19
+ *
20
+ * This source code is licensed under the GPLv3 license found in the
21
+ * LICENSE file in the root directory of this source tree.
22
+ *
23
+ */
24
+ import React, { useState, useRef, useEffect } from 'react';
25
+ import { isMobile } from 'react-device-detect';
26
+ import { useTranslation } from 'react-i18next';
27
+ import Button, { IconButtonDualStates } from '@ndla/button';
28
+ import { ChevronDown, ChevronUp } from '@ndla/icons/common';
29
+ import { Cross as CrossRaw } from '@ndla/icons/action';
30
+ import { spacing, colors, misc, animations, fonts } from '@ndla/core';
31
+ import Tooltip from '@ndla/tooltip';
32
+ import { uuid } from '@ndla/util';
33
+ import Suggestions from './Suggestions';
34
+ import { jsx as ___EmotionJSX } from "@emotion/core";
35
+
36
+ var Cross = /*#__PURE__*/_styled(CrossRaw, {
37
+ target: "e55qeml0",
38
+ label: "Cross"
39
+ })("margin-left:", spacing.xxsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AAqB8B","file":"SuggestionInput.tsx","sourcesContent":["/*\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useState, useRef, useEffect, ReactNode, RefObject, ChangeEvent, KeyboardEvent } from 'react';\nimport { isMobile } from 'react-device-detect';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport Button, { IconButtonDualStates } from '@ndla/button';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport { Cross as CrossRaw } from '@ndla/icons/action';\nimport { spacing, colors, misc, animations, fonts } from '@ndla/core';\nimport Tooltip from '@ndla/tooltip';\nimport { uuid } from '@ndla/util';\nimport Suggestions from './Suggestions';\nimport type { TagType } from './TagSelector';\n\nconst Cross = styled(CrossRaw)`\n  margin-left: ${spacing.xxsmall};\n`;\n\nconst SuggestionInputContainer = styled.div`\n  margin-bottom: ${spacing.large};\n`;\n\nconst StyledInput = styled.input`\n  flex-grow: 1;\n  border: 0;\n  outline: none;\n  background: transparent;\n  ${fonts.sizes(18)};\n`;\nconst StyledInputWrapper = styled.div`\n  display: flex;\n  flex-wrap: wrap;\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.greyLighter};\n  transition: border-color ${animations.durations.normal} ease;\n  border-radius: ${misc.borderRadius};\n  &:focus-within {\n    border-color: ${colors.brand.primary};\n  }\n`;\n\nconst CombinedInputAndDropdownWrapper = styled.div`\n  display: flex;\n  flex-grow: 1;\n`;\n\ninterface SuggestionInputProps {\n  suggestions: TagType[];\n  value: string;\n  onChange: (e: ChangeEvent<HTMLInputElement>) => void;\n  setExpanded: (expanded: boolean) => void;\n  expanded: boolean;\n  onToggleTag: (id: string) => void;\n  setInputValue: (value: string) => void;\n  onCreateTag: (tagName: string) => void;\n  addedTags: TagType[];\n  dropdownMaxHeight: string;\n  prefix?: string | ReactNode;\n  inline?: boolean;\n  scrollAnchorElement: RefObject<HTMLDivElement>;\n}\n\nconst SuggestionInput = ({\n  suggestions,\n  value,\n  setInputValue,\n  onCreateTag,\n  onChange,\n  onToggleTag,\n  addedTags,\n  setExpanded,\n  expanded,\n  dropdownMaxHeight,\n  prefix,\n  inline,\n  scrollAnchorElement,\n}: SuggestionInputProps) => {\n  const { t } = useTranslation();\n  const [currentHighlightedIndex, setCurrentHighlightedIndex] = useState(0);\n  const [hasFocus, setHasFocus] = useState(false);\n  const initalRender = useRef(true);\n  const inputRef = useRef<HTMLInputElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n  const suggestionIdRef = useRef<string>(uuid());\n\n  useEffect(() => {\n    setCurrentHighlightedIndex(0);\n  }, [suggestions]);\n\n  useEffect(() => {\n    if (!initalRender.current) {\n      inputRef.current?.focus();\n    } else {\n      initalRender.current = false;\n    }\n  }, [addedTags]);\n\n  useEffect(() => {\n    const selectedSuggestionElement = document\n      .getElementById(suggestionIdRef.current)\n      ?.querySelector('[aria-selected=\"true\"]');\n    if (selectedSuggestionElement) {\n      // Do we need to scroll this into view?\n      selectedSuggestionElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [currentHighlightedIndex]);\n\n  const hasBeenAdded = (id: string) => addedTags.some(({ id: idAdded }) => idAdded === id);\n\n  return (\n    <SuggestionInputContainer ref={containerRef}>\n      <StyledInputWrapper>\n        {addedTags.map(({ id, name }) => (\n          <Button\n            aria-label={t('tagSelector.removeTag', { name })}\n            onClick={() => onToggleTag(id)}\n            light\n            borderShape=\"rounded\"\n            key={id}\n            size=\"small\">\n            {prefix}\n            {name}\n            <Cross />\n          </Button>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          <StyledInput\n            placeholder={t('tagSelector.placeholder')}\n            value={value}\n            autoComplete=\"off\"\n            onBlur={(e) => {\n              const relatedTarget = e.relatedTarget as HTMLElement;\n              if (!relatedTarget?.dataset?.suggestionbutton) {\n                setExpanded(false);\n                setHasFocus(false);\n              }\n            }}\n            onChange={onChange}\n            onFocus={() => {\n              if (isMobile && scrollAnchorElement?.current) {\n                scrollAnchorElement.current.scrollIntoView({\n                  behavior: 'smooth',\n                });\n              }\n              setHasFocus(true);\n            }}\n            ref={inputRef}\n            onKeyDown={(e: KeyboardEvent<HTMLInputElement>) => {\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n              } else if (e.key === 'Enter' || e.key === 'Tab') {\n                if (value !== '' || expanded) {\n                  if (suggestions.length > 0) {\n                    if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                      onToggleTag(suggestions[currentHighlightedIndex].id);\n                    }\n                    setInputValue('');\n                    if (e.key === 'Enter') {\n                      e.preventDefault();\n                    }\n                  } else {\n                    onCreateTag(value);\n                    setInputValue('');\n                    e.preventDefault();\n                  }\n                } else if (e.key === 'Enter') {\n                  e.preventDefault();\n                }\n              } else if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n              } else if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n              }\n            }}\n          />\n          <Tooltip tooltip={expanded ? t('tagSelector.hideAllTags') : t('tagSelector.showAllTags')}>\n            <IconButtonDualStates\n              data-suggestionbutton\n              ariaLabelActive={t('tagSelector.showAllTags')}\n              ariaLabelInActive={t('tagSelector.hideAllTags')}\n              active={expanded}\n              greyLighter\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              aria-controls={suggestionIdRef.current}\n              onClick={() => {\n                setInputValue('');\n                setExpanded(!expanded);\n                inputRef.current?.focus();\n              }}\n            />\n          </Tooltip>\n        </CombinedInputAndDropdownWrapper>\n      </StyledInputWrapper>\n      <div id={suggestionIdRef.current} aria-live=\"polite\">\n        {(hasFocus || expanded) && suggestions.length > 0 ? (\n          <Suggestions\n            inline={inline}\n            dropdownMaxHeight={dropdownMaxHeight}\n            suggestions={suggestions}\n            currentHighlightedIndex={currentHighlightedIndex}\n            onToggleTag={onToggleTag}\n            hasBeenAdded={hasBeenAdded}\n          />\n        ) : null}\n      </div>\n    </SuggestionInputContainer>\n  );\n};\n\nexport default SuggestionInput;\n"]} */"));
40
+
41
+ var SuggestionInputContainer = _styled("div", {
42
+ target: "e55qeml1",
43
+ label: "SuggestionInputContainer"
44
+ })("margin-bottom:", spacing.large, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AAyB2C","file":"SuggestionInput.tsx","sourcesContent":["/*\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useState, useRef, useEffect, ReactNode, RefObject, ChangeEvent, KeyboardEvent } from 'react';\nimport { isMobile } from 'react-device-detect';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport Button, { IconButtonDualStates } from '@ndla/button';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport { Cross as CrossRaw } from '@ndla/icons/action';\nimport { spacing, colors, misc, animations, fonts } from '@ndla/core';\nimport Tooltip from '@ndla/tooltip';\nimport { uuid } from '@ndla/util';\nimport Suggestions from './Suggestions';\nimport type { TagType } from './TagSelector';\n\nconst Cross = styled(CrossRaw)`\n  margin-left: ${spacing.xxsmall};\n`;\n\nconst SuggestionInputContainer = styled.div`\n  margin-bottom: ${spacing.large};\n`;\n\nconst StyledInput = styled.input`\n  flex-grow: 1;\n  border: 0;\n  outline: none;\n  background: transparent;\n  ${fonts.sizes(18)};\n`;\nconst StyledInputWrapper = styled.div`\n  display: flex;\n  flex-wrap: wrap;\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.greyLighter};\n  transition: border-color ${animations.durations.normal} ease;\n  border-radius: ${misc.borderRadius};\n  &:focus-within {\n    border-color: ${colors.brand.primary};\n  }\n`;\n\nconst CombinedInputAndDropdownWrapper = styled.div`\n  display: flex;\n  flex-grow: 1;\n`;\n\ninterface SuggestionInputProps {\n  suggestions: TagType[];\n  value: string;\n  onChange: (e: ChangeEvent<HTMLInputElement>) => void;\n  setExpanded: (expanded: boolean) => void;\n  expanded: boolean;\n  onToggleTag: (id: string) => void;\n  setInputValue: (value: string) => void;\n  onCreateTag: (tagName: string) => void;\n  addedTags: TagType[];\n  dropdownMaxHeight: string;\n  prefix?: string | ReactNode;\n  inline?: boolean;\n  scrollAnchorElement: RefObject<HTMLDivElement>;\n}\n\nconst SuggestionInput = ({\n  suggestions,\n  value,\n  setInputValue,\n  onCreateTag,\n  onChange,\n  onToggleTag,\n  addedTags,\n  setExpanded,\n  expanded,\n  dropdownMaxHeight,\n  prefix,\n  inline,\n  scrollAnchorElement,\n}: SuggestionInputProps) => {\n  const { t } = useTranslation();\n  const [currentHighlightedIndex, setCurrentHighlightedIndex] = useState(0);\n  const [hasFocus, setHasFocus] = useState(false);\n  const initalRender = useRef(true);\n  const inputRef = useRef<HTMLInputElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n  const suggestionIdRef = useRef<string>(uuid());\n\n  useEffect(() => {\n    setCurrentHighlightedIndex(0);\n  }, [suggestions]);\n\n  useEffect(() => {\n    if (!initalRender.current) {\n      inputRef.current?.focus();\n    } else {\n      initalRender.current = false;\n    }\n  }, [addedTags]);\n\n  useEffect(() => {\n    const selectedSuggestionElement = document\n      .getElementById(suggestionIdRef.current)\n      ?.querySelector('[aria-selected=\"true\"]');\n    if (selectedSuggestionElement) {\n      // Do we need to scroll this into view?\n      selectedSuggestionElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [currentHighlightedIndex]);\n\n  const hasBeenAdded = (id: string) => addedTags.some(({ id: idAdded }) => idAdded === id);\n\n  return (\n    <SuggestionInputContainer ref={containerRef}>\n      <StyledInputWrapper>\n        {addedTags.map(({ id, name }) => (\n          <Button\n            aria-label={t('tagSelector.removeTag', { name })}\n            onClick={() => onToggleTag(id)}\n            light\n            borderShape=\"rounded\"\n            key={id}\n            size=\"small\">\n            {prefix}\n            {name}\n            <Cross />\n          </Button>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          <StyledInput\n            placeholder={t('tagSelector.placeholder')}\n            value={value}\n            autoComplete=\"off\"\n            onBlur={(e) => {\n              const relatedTarget = e.relatedTarget as HTMLElement;\n              if (!relatedTarget?.dataset?.suggestionbutton) {\n                setExpanded(false);\n                setHasFocus(false);\n              }\n            }}\n            onChange={onChange}\n            onFocus={() => {\n              if (isMobile && scrollAnchorElement?.current) {\n                scrollAnchorElement.current.scrollIntoView({\n                  behavior: 'smooth',\n                });\n              }\n              setHasFocus(true);\n            }}\n            ref={inputRef}\n            onKeyDown={(e: KeyboardEvent<HTMLInputElement>) => {\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n              } else if (e.key === 'Enter' || e.key === 'Tab') {\n                if (value !== '' || expanded) {\n                  if (suggestions.length > 0) {\n                    if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                      onToggleTag(suggestions[currentHighlightedIndex].id);\n                    }\n                    setInputValue('');\n                    if (e.key === 'Enter') {\n                      e.preventDefault();\n                    }\n                  } else {\n                    onCreateTag(value);\n                    setInputValue('');\n                    e.preventDefault();\n                  }\n                } else if (e.key === 'Enter') {\n                  e.preventDefault();\n                }\n              } else if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n              } else if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n              }\n            }}\n          />\n          <Tooltip tooltip={expanded ? t('tagSelector.hideAllTags') : t('tagSelector.showAllTags')}>\n            <IconButtonDualStates\n              data-suggestionbutton\n              ariaLabelActive={t('tagSelector.showAllTags')}\n              ariaLabelInActive={t('tagSelector.hideAllTags')}\n              active={expanded}\n              greyLighter\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              aria-controls={suggestionIdRef.current}\n              onClick={() => {\n                setInputValue('');\n                setExpanded(!expanded);\n                inputRef.current?.focus();\n              }}\n            />\n          </Tooltip>\n        </CombinedInputAndDropdownWrapper>\n      </StyledInputWrapper>\n      <div id={suggestionIdRef.current} aria-live=\"polite\">\n        {(hasFocus || expanded) && suggestions.length > 0 ? (\n          <Suggestions\n            inline={inline}\n            dropdownMaxHeight={dropdownMaxHeight}\n            suggestions={suggestions}\n            currentHighlightedIndex={currentHighlightedIndex}\n            onToggleTag={onToggleTag}\n            hasBeenAdded={hasBeenAdded}\n          />\n        ) : null}\n      </div>\n    </SuggestionInputContainer>\n  );\n};\n\nexport default SuggestionInput;\n"]} */"));
45
+
46
+ var StyledInput = _styled("input", {
47
+ target: "e55qeml2",
48
+ label: "StyledInput"
49
+ })("flex-grow:1;border:0;outline:none;background:transparent;", fonts.sizes(18), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AA6BgC","file":"SuggestionInput.tsx","sourcesContent":["/*\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useState, useRef, useEffect, ReactNode, RefObject, ChangeEvent, KeyboardEvent } from 'react';\nimport { isMobile } from 'react-device-detect';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport Button, { IconButtonDualStates } from '@ndla/button';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport { Cross as CrossRaw } from '@ndla/icons/action';\nimport { spacing, colors, misc, animations, fonts } from '@ndla/core';\nimport Tooltip from '@ndla/tooltip';\nimport { uuid } from '@ndla/util';\nimport Suggestions from './Suggestions';\nimport type { TagType } from './TagSelector';\n\nconst Cross = styled(CrossRaw)`\n  margin-left: ${spacing.xxsmall};\n`;\n\nconst SuggestionInputContainer = styled.div`\n  margin-bottom: ${spacing.large};\n`;\n\nconst StyledInput = styled.input`\n  flex-grow: 1;\n  border: 0;\n  outline: none;\n  background: transparent;\n  ${fonts.sizes(18)};\n`;\nconst StyledInputWrapper = styled.div`\n  display: flex;\n  flex-wrap: wrap;\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.greyLighter};\n  transition: border-color ${animations.durations.normal} ease;\n  border-radius: ${misc.borderRadius};\n  &:focus-within {\n    border-color: ${colors.brand.primary};\n  }\n`;\n\nconst CombinedInputAndDropdownWrapper = styled.div`\n  display: flex;\n  flex-grow: 1;\n`;\n\ninterface SuggestionInputProps {\n  suggestions: TagType[];\n  value: string;\n  onChange: (e: ChangeEvent<HTMLInputElement>) => void;\n  setExpanded: (expanded: boolean) => void;\n  expanded: boolean;\n  onToggleTag: (id: string) => void;\n  setInputValue: (value: string) => void;\n  onCreateTag: (tagName: string) => void;\n  addedTags: TagType[];\n  dropdownMaxHeight: string;\n  prefix?: string | ReactNode;\n  inline?: boolean;\n  scrollAnchorElement: RefObject<HTMLDivElement>;\n}\n\nconst SuggestionInput = ({\n  suggestions,\n  value,\n  setInputValue,\n  onCreateTag,\n  onChange,\n  onToggleTag,\n  addedTags,\n  setExpanded,\n  expanded,\n  dropdownMaxHeight,\n  prefix,\n  inline,\n  scrollAnchorElement,\n}: SuggestionInputProps) => {\n  const { t } = useTranslation();\n  const [currentHighlightedIndex, setCurrentHighlightedIndex] = useState(0);\n  const [hasFocus, setHasFocus] = useState(false);\n  const initalRender = useRef(true);\n  const inputRef = useRef<HTMLInputElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n  const suggestionIdRef = useRef<string>(uuid());\n\n  useEffect(() => {\n    setCurrentHighlightedIndex(0);\n  }, [suggestions]);\n\n  useEffect(() => {\n    if (!initalRender.current) {\n      inputRef.current?.focus();\n    } else {\n      initalRender.current = false;\n    }\n  }, [addedTags]);\n\n  useEffect(() => {\n    const selectedSuggestionElement = document\n      .getElementById(suggestionIdRef.current)\n      ?.querySelector('[aria-selected=\"true\"]');\n    if (selectedSuggestionElement) {\n      // Do we need to scroll this into view?\n      selectedSuggestionElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [currentHighlightedIndex]);\n\n  const hasBeenAdded = (id: string) => addedTags.some(({ id: idAdded }) => idAdded === id);\n\n  return (\n    <SuggestionInputContainer ref={containerRef}>\n      <StyledInputWrapper>\n        {addedTags.map(({ id, name }) => (\n          <Button\n            aria-label={t('tagSelector.removeTag', { name })}\n            onClick={() => onToggleTag(id)}\n            light\n            borderShape=\"rounded\"\n            key={id}\n            size=\"small\">\n            {prefix}\n            {name}\n            <Cross />\n          </Button>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          <StyledInput\n            placeholder={t('tagSelector.placeholder')}\n            value={value}\n            autoComplete=\"off\"\n            onBlur={(e) => {\n              const relatedTarget = e.relatedTarget as HTMLElement;\n              if (!relatedTarget?.dataset?.suggestionbutton) {\n                setExpanded(false);\n                setHasFocus(false);\n              }\n            }}\n            onChange={onChange}\n            onFocus={() => {\n              if (isMobile && scrollAnchorElement?.current) {\n                scrollAnchorElement.current.scrollIntoView({\n                  behavior: 'smooth',\n                });\n              }\n              setHasFocus(true);\n            }}\n            ref={inputRef}\n            onKeyDown={(e: KeyboardEvent<HTMLInputElement>) => {\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n              } else if (e.key === 'Enter' || e.key === 'Tab') {\n                if (value !== '' || expanded) {\n                  if (suggestions.length > 0) {\n                    if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                      onToggleTag(suggestions[currentHighlightedIndex].id);\n                    }\n                    setInputValue('');\n                    if (e.key === 'Enter') {\n                      e.preventDefault();\n                    }\n                  } else {\n                    onCreateTag(value);\n                    setInputValue('');\n                    e.preventDefault();\n                  }\n                } else if (e.key === 'Enter') {\n                  e.preventDefault();\n                }\n              } else if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n              } else if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n              }\n            }}\n          />\n          <Tooltip tooltip={expanded ? t('tagSelector.hideAllTags') : t('tagSelector.showAllTags')}>\n            <IconButtonDualStates\n              data-suggestionbutton\n              ariaLabelActive={t('tagSelector.showAllTags')}\n              ariaLabelInActive={t('tagSelector.hideAllTags')}\n              active={expanded}\n              greyLighter\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              aria-controls={suggestionIdRef.current}\n              onClick={() => {\n                setInputValue('');\n                setExpanded(!expanded);\n                inputRef.current?.focus();\n              }}\n            />\n          </Tooltip>\n        </CombinedInputAndDropdownWrapper>\n      </StyledInputWrapper>\n      <div id={suggestionIdRef.current} aria-live=\"polite\">\n        {(hasFocus || expanded) && suggestions.length > 0 ? (\n          <Suggestions\n            inline={inline}\n            dropdownMaxHeight={dropdownMaxHeight}\n            suggestions={suggestions}\n            currentHighlightedIndex={currentHighlightedIndex}\n            onToggleTag={onToggleTag}\n            hasBeenAdded={hasBeenAdded}\n          />\n        ) : null}\n      </div>\n    </SuggestionInputContainer>\n  );\n};\n\nexport default SuggestionInput;\n"]} */"));
50
+
51
+ var StyledInputWrapper = _styled("div", {
52
+ target: "e55qeml3",
53
+ label: "StyledInputWrapper"
54
+ })("display:flex;flex-wrap:wrap;gap:", spacing.xsmall, ";padding:", spacing.small, ";border:1px solid ", colors.brand.greyLighter, ";transition:border-color ", animations.durations.normal, " ease;border-radius:", misc.borderRadius, ";&:focus-within{border-color:", colors.brand.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AAoCqC","file":"SuggestionInput.tsx","sourcesContent":["/*\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useState, useRef, useEffect, ReactNode, RefObject, ChangeEvent, KeyboardEvent } from 'react';\nimport { isMobile } from 'react-device-detect';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport Button, { IconButtonDualStates } from '@ndla/button';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport { Cross as CrossRaw } from '@ndla/icons/action';\nimport { spacing, colors, misc, animations, fonts } from '@ndla/core';\nimport Tooltip from '@ndla/tooltip';\nimport { uuid } from '@ndla/util';\nimport Suggestions from './Suggestions';\nimport type { TagType } from './TagSelector';\n\nconst Cross = styled(CrossRaw)`\n  margin-left: ${spacing.xxsmall};\n`;\n\nconst SuggestionInputContainer = styled.div`\n  margin-bottom: ${spacing.large};\n`;\n\nconst StyledInput = styled.input`\n  flex-grow: 1;\n  border: 0;\n  outline: none;\n  background: transparent;\n  ${fonts.sizes(18)};\n`;\nconst StyledInputWrapper = styled.div`\n  display: flex;\n  flex-wrap: wrap;\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.greyLighter};\n  transition: border-color ${animations.durations.normal} ease;\n  border-radius: ${misc.borderRadius};\n  &:focus-within {\n    border-color: ${colors.brand.primary};\n  }\n`;\n\nconst CombinedInputAndDropdownWrapper = styled.div`\n  display: flex;\n  flex-grow: 1;\n`;\n\ninterface SuggestionInputProps {\n  suggestions: TagType[];\n  value: string;\n  onChange: (e: ChangeEvent<HTMLInputElement>) => void;\n  setExpanded: (expanded: boolean) => void;\n  expanded: boolean;\n  onToggleTag: (id: string) => void;\n  setInputValue: (value: string) => void;\n  onCreateTag: (tagName: string) => void;\n  addedTags: TagType[];\n  dropdownMaxHeight: string;\n  prefix?: string | ReactNode;\n  inline?: boolean;\n  scrollAnchorElement: RefObject<HTMLDivElement>;\n}\n\nconst SuggestionInput = ({\n  suggestions,\n  value,\n  setInputValue,\n  onCreateTag,\n  onChange,\n  onToggleTag,\n  addedTags,\n  setExpanded,\n  expanded,\n  dropdownMaxHeight,\n  prefix,\n  inline,\n  scrollAnchorElement,\n}: SuggestionInputProps) => {\n  const { t } = useTranslation();\n  const [currentHighlightedIndex, setCurrentHighlightedIndex] = useState(0);\n  const [hasFocus, setHasFocus] = useState(false);\n  const initalRender = useRef(true);\n  const inputRef = useRef<HTMLInputElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n  const suggestionIdRef = useRef<string>(uuid());\n\n  useEffect(() => {\n    setCurrentHighlightedIndex(0);\n  }, [suggestions]);\n\n  useEffect(() => {\n    if (!initalRender.current) {\n      inputRef.current?.focus();\n    } else {\n      initalRender.current = false;\n    }\n  }, [addedTags]);\n\n  useEffect(() => {\n    const selectedSuggestionElement = document\n      .getElementById(suggestionIdRef.current)\n      ?.querySelector('[aria-selected=\"true\"]');\n    if (selectedSuggestionElement) {\n      // Do we need to scroll this into view?\n      selectedSuggestionElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [currentHighlightedIndex]);\n\n  const hasBeenAdded = (id: string) => addedTags.some(({ id: idAdded }) => idAdded === id);\n\n  return (\n    <SuggestionInputContainer ref={containerRef}>\n      <StyledInputWrapper>\n        {addedTags.map(({ id, name }) => (\n          <Button\n            aria-label={t('tagSelector.removeTag', { name })}\n            onClick={() => onToggleTag(id)}\n            light\n            borderShape=\"rounded\"\n            key={id}\n            size=\"small\">\n            {prefix}\n            {name}\n            <Cross />\n          </Button>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          <StyledInput\n            placeholder={t('tagSelector.placeholder')}\n            value={value}\n            autoComplete=\"off\"\n            onBlur={(e) => {\n              const relatedTarget = e.relatedTarget as HTMLElement;\n              if (!relatedTarget?.dataset?.suggestionbutton) {\n                setExpanded(false);\n                setHasFocus(false);\n              }\n            }}\n            onChange={onChange}\n            onFocus={() => {\n              if (isMobile && scrollAnchorElement?.current) {\n                scrollAnchorElement.current.scrollIntoView({\n                  behavior: 'smooth',\n                });\n              }\n              setHasFocus(true);\n            }}\n            ref={inputRef}\n            onKeyDown={(e: KeyboardEvent<HTMLInputElement>) => {\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n              } else if (e.key === 'Enter' || e.key === 'Tab') {\n                if (value !== '' || expanded) {\n                  if (suggestions.length > 0) {\n                    if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                      onToggleTag(suggestions[currentHighlightedIndex].id);\n                    }\n                    setInputValue('');\n                    if (e.key === 'Enter') {\n                      e.preventDefault();\n                    }\n                  } else {\n                    onCreateTag(value);\n                    setInputValue('');\n                    e.preventDefault();\n                  }\n                } else if (e.key === 'Enter') {\n                  e.preventDefault();\n                }\n              } else if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n              } else if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n              }\n            }}\n          />\n          <Tooltip tooltip={expanded ? t('tagSelector.hideAllTags') : t('tagSelector.showAllTags')}>\n            <IconButtonDualStates\n              data-suggestionbutton\n              ariaLabelActive={t('tagSelector.showAllTags')}\n              ariaLabelInActive={t('tagSelector.hideAllTags')}\n              active={expanded}\n              greyLighter\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              aria-controls={suggestionIdRef.current}\n              onClick={() => {\n                setInputValue('');\n                setExpanded(!expanded);\n                inputRef.current?.focus();\n              }}\n            />\n          </Tooltip>\n        </CombinedInputAndDropdownWrapper>\n      </StyledInputWrapper>\n      <div id={suggestionIdRef.current} aria-live=\"polite\">\n        {(hasFocus || expanded) && suggestions.length > 0 ? (\n          <Suggestions\n            inline={inline}\n            dropdownMaxHeight={dropdownMaxHeight}\n            suggestions={suggestions}\n            currentHighlightedIndex={currentHighlightedIndex}\n            onToggleTag={onToggleTag}\n            hasBeenAdded={hasBeenAdded}\n          />\n        ) : null}\n      </div>\n    </SuggestionInputContainer>\n  );\n};\n\nexport default SuggestionInput;\n"]} */"));
55
+
56
+ var CombinedInputAndDropdownWrapper = _styled("div", {
57
+ target: "e55qeml4",
58
+ label: "CombinedInputAndDropdownWrapper"
59
+ })(process.env.NODE_ENV === "production" ? {
60
+ name: "1ilyui9",
61
+ styles: "display:flex;flex-grow:1;"
62
+ } : {
63
+ name: "1ilyui9",
64
+ styles: "display:flex;flex-grow:1;",
65
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AAiDkD","file":"SuggestionInput.tsx","sourcesContent":["/*\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useState, useRef, useEffect, ReactNode, RefObject, ChangeEvent, KeyboardEvent } from 'react';\nimport { isMobile } from 'react-device-detect';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport Button, { IconButtonDualStates } from '@ndla/button';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport { Cross as CrossRaw } from '@ndla/icons/action';\nimport { spacing, colors, misc, animations, fonts } from '@ndla/core';\nimport Tooltip from '@ndla/tooltip';\nimport { uuid } from '@ndla/util';\nimport Suggestions from './Suggestions';\nimport type { TagType } from './TagSelector';\n\nconst Cross = styled(CrossRaw)`\n  margin-left: ${spacing.xxsmall};\n`;\n\nconst SuggestionInputContainer = styled.div`\n  margin-bottom: ${spacing.large};\n`;\n\nconst StyledInput = styled.input`\n  flex-grow: 1;\n  border: 0;\n  outline: none;\n  background: transparent;\n  ${fonts.sizes(18)};\n`;\nconst StyledInputWrapper = styled.div`\n  display: flex;\n  flex-wrap: wrap;\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.greyLighter};\n  transition: border-color ${animations.durations.normal} ease;\n  border-radius: ${misc.borderRadius};\n  &:focus-within {\n    border-color: ${colors.brand.primary};\n  }\n`;\n\nconst CombinedInputAndDropdownWrapper = styled.div`\n  display: flex;\n  flex-grow: 1;\n`;\n\ninterface SuggestionInputProps {\n  suggestions: TagType[];\n  value: string;\n  onChange: (e: ChangeEvent<HTMLInputElement>) => void;\n  setExpanded: (expanded: boolean) => void;\n  expanded: boolean;\n  onToggleTag: (id: string) => void;\n  setInputValue: (value: string) => void;\n  onCreateTag: (tagName: string) => void;\n  addedTags: TagType[];\n  dropdownMaxHeight: string;\n  prefix?: string | ReactNode;\n  inline?: boolean;\n  scrollAnchorElement: RefObject<HTMLDivElement>;\n}\n\nconst SuggestionInput = ({\n  suggestions,\n  value,\n  setInputValue,\n  onCreateTag,\n  onChange,\n  onToggleTag,\n  addedTags,\n  setExpanded,\n  expanded,\n  dropdownMaxHeight,\n  prefix,\n  inline,\n  scrollAnchorElement,\n}: SuggestionInputProps) => {\n  const { t } = useTranslation();\n  const [currentHighlightedIndex, setCurrentHighlightedIndex] = useState(0);\n  const [hasFocus, setHasFocus] = useState(false);\n  const initalRender = useRef(true);\n  const inputRef = useRef<HTMLInputElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n  const suggestionIdRef = useRef<string>(uuid());\n\n  useEffect(() => {\n    setCurrentHighlightedIndex(0);\n  }, [suggestions]);\n\n  useEffect(() => {\n    if (!initalRender.current) {\n      inputRef.current?.focus();\n    } else {\n      initalRender.current = false;\n    }\n  }, [addedTags]);\n\n  useEffect(() => {\n    const selectedSuggestionElement = document\n      .getElementById(suggestionIdRef.current)\n      ?.querySelector('[aria-selected=\"true\"]');\n    if (selectedSuggestionElement) {\n      // Do we need to scroll this into view?\n      selectedSuggestionElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [currentHighlightedIndex]);\n\n  const hasBeenAdded = (id: string) => addedTags.some(({ id: idAdded }) => idAdded === id);\n\n  return (\n    <SuggestionInputContainer ref={containerRef}>\n      <StyledInputWrapper>\n        {addedTags.map(({ id, name }) => (\n          <Button\n            aria-label={t('tagSelector.removeTag', { name })}\n            onClick={() => onToggleTag(id)}\n            light\n            borderShape=\"rounded\"\n            key={id}\n            size=\"small\">\n            {prefix}\n            {name}\n            <Cross />\n          </Button>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          <StyledInput\n            placeholder={t('tagSelector.placeholder')}\n            value={value}\n            autoComplete=\"off\"\n            onBlur={(e) => {\n              const relatedTarget = e.relatedTarget as HTMLElement;\n              if (!relatedTarget?.dataset?.suggestionbutton) {\n                setExpanded(false);\n                setHasFocus(false);\n              }\n            }}\n            onChange={onChange}\n            onFocus={() => {\n              if (isMobile && scrollAnchorElement?.current) {\n                scrollAnchorElement.current.scrollIntoView({\n                  behavior: 'smooth',\n                });\n              }\n              setHasFocus(true);\n            }}\n            ref={inputRef}\n            onKeyDown={(e: KeyboardEvent<HTMLInputElement>) => {\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n              } else if (e.key === 'Enter' || e.key === 'Tab') {\n                if (value !== '' || expanded) {\n                  if (suggestions.length > 0) {\n                    if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                      onToggleTag(suggestions[currentHighlightedIndex].id);\n                    }\n                    setInputValue('');\n                    if (e.key === 'Enter') {\n                      e.preventDefault();\n                    }\n                  } else {\n                    onCreateTag(value);\n                    setInputValue('');\n                    e.preventDefault();\n                  }\n                } else if (e.key === 'Enter') {\n                  e.preventDefault();\n                }\n              } else if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n              } else if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n              }\n            }}\n          />\n          <Tooltip tooltip={expanded ? t('tagSelector.hideAllTags') : t('tagSelector.showAllTags')}>\n            <IconButtonDualStates\n              data-suggestionbutton\n              ariaLabelActive={t('tagSelector.showAllTags')}\n              ariaLabelInActive={t('tagSelector.hideAllTags')}\n              active={expanded}\n              greyLighter\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              aria-controls={suggestionIdRef.current}\n              onClick={() => {\n                setInputValue('');\n                setExpanded(!expanded);\n                inputRef.current?.focus();\n              }}\n            />\n          </Tooltip>\n        </CombinedInputAndDropdownWrapper>\n      </StyledInputWrapper>\n      <div id={suggestionIdRef.current} aria-live=\"polite\">\n        {(hasFocus || expanded) && suggestions.length > 0 ? (\n          <Suggestions\n            inline={inline}\n            dropdownMaxHeight={dropdownMaxHeight}\n            suggestions={suggestions}\n            currentHighlightedIndex={currentHighlightedIndex}\n            onToggleTag={onToggleTag}\n            hasBeenAdded={hasBeenAdded}\n          />\n        ) : null}\n      </div>\n    </SuggestionInputContainer>\n  );\n};\n\nexport default SuggestionInput;\n"]} */",
66
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
67
+ });
68
+
69
+ var SuggestionInput = function SuggestionInput(_ref) {
70
+ var suggestions = _ref.suggestions,
71
+ value = _ref.value,
72
+ setInputValue = _ref.setInputValue,
73
+ onCreateTag = _ref.onCreateTag,
74
+ onChange = _ref.onChange,
75
+ onToggleTag = _ref.onToggleTag,
76
+ addedTags = _ref.addedTags,
77
+ setExpanded = _ref.setExpanded,
78
+ expanded = _ref.expanded,
79
+ dropdownMaxHeight = _ref.dropdownMaxHeight,
80
+ prefix = _ref.prefix,
81
+ inline = _ref.inline,
82
+ scrollAnchorElement = _ref.scrollAnchorElement;
83
+
84
+ var _useTranslation = useTranslation(),
85
+ t = _useTranslation.t;
86
+
87
+ var _useState = useState(0),
88
+ _useState2 = _slicedToArray(_useState, 2),
89
+ currentHighlightedIndex = _useState2[0],
90
+ setCurrentHighlightedIndex = _useState2[1];
91
+
92
+ var _useState3 = useState(false),
93
+ _useState4 = _slicedToArray(_useState3, 2),
94
+ hasFocus = _useState4[0],
95
+ setHasFocus = _useState4[1];
96
+
97
+ var initalRender = useRef(true);
98
+ var inputRef = useRef(null);
99
+ var containerRef = useRef(null);
100
+ var suggestionIdRef = useRef(uuid());
101
+ useEffect(function () {
102
+ setCurrentHighlightedIndex(0);
103
+ }, [suggestions]);
104
+ useEffect(function () {
105
+ if (!initalRender.current) {
106
+ var _inputRef$current;
107
+
108
+ (_inputRef$current = inputRef.current) === null || _inputRef$current === void 0 ? void 0 : _inputRef$current.focus();
109
+ } else {
110
+ initalRender.current = false;
111
+ }
112
+ }, [addedTags]);
113
+ useEffect(function () {
114
+ var _document$getElementB;
115
+
116
+ var selectedSuggestionElement = (_document$getElementB = document.getElementById(suggestionIdRef.current)) === null || _document$getElementB === void 0 ? void 0 : _document$getElementB.querySelector('[aria-selected="true"]');
117
+
118
+ if (selectedSuggestionElement) {
119
+ // Do we need to scroll this into view?
120
+ selectedSuggestionElement.scrollIntoView({
121
+ behavior: 'smooth',
122
+ block: 'nearest'
123
+ });
124
+ }
125
+ }, [currentHighlightedIndex]);
126
+
127
+ var hasBeenAdded = function hasBeenAdded(id) {
128
+ return addedTags.some(function (_ref2) {
129
+ var idAdded = _ref2.id;
130
+ return idAdded === id;
131
+ });
132
+ };
133
+
134
+ return ___EmotionJSX(SuggestionInputContainer, {
135
+ ref: containerRef
136
+ }, ___EmotionJSX(StyledInputWrapper, null, addedTags.map(function (_ref3) {
137
+ var id = _ref3.id,
138
+ name = _ref3.name;
139
+ return ___EmotionJSX(Button, {
140
+ "aria-label": t('tagSelector.removeTag', {
141
+ name: name
142
+ }),
143
+ onClick: function onClick() {
144
+ return onToggleTag(id);
145
+ },
146
+ light: true,
147
+ borderShape: "rounded",
148
+ key: id,
149
+ size: "small"
150
+ }, prefix, name, ___EmotionJSX(Cross, null));
151
+ }), ___EmotionJSX(CombinedInputAndDropdownWrapper, null, ___EmotionJSX(StyledInput, {
152
+ placeholder: t('tagSelector.placeholder'),
153
+ value: value,
154
+ autoComplete: "off",
155
+ onBlur: function onBlur(e) {
156
+ var _relatedTarget$datase;
157
+
158
+ var relatedTarget = e.relatedTarget;
159
+
160
+ if (!(relatedTarget === null || relatedTarget === void 0 ? void 0 : (_relatedTarget$datase = relatedTarget.dataset) === null || _relatedTarget$datase === void 0 ? void 0 : _relatedTarget$datase.suggestionbutton)) {
161
+ setExpanded(false);
162
+ setHasFocus(false);
163
+ }
164
+ },
165
+ onChange: onChange,
166
+ onFocus: function onFocus() {
167
+ if (isMobile && (scrollAnchorElement === null || scrollAnchorElement === void 0 ? void 0 : scrollAnchorElement.current)) {
168
+ scrollAnchorElement.current.scrollIntoView({
169
+ behavior: 'smooth'
170
+ });
171
+ }
172
+
173
+ setHasFocus(true);
174
+ },
175
+ ref: inputRef,
176
+ onKeyDown: function onKeyDown(e) {
177
+ if (e.key === 'Escape') {
178
+ setExpanded(false);
179
+ e.preventDefault();
180
+ } else if (e.key === 'Enter' || e.key === 'Tab') {
181
+ if (value !== '' || expanded) {
182
+ if (suggestions.length > 0) {
183
+ if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {
184
+ onToggleTag(suggestions[currentHighlightedIndex].id);
185
+ }
186
+
187
+ setInputValue('');
188
+
189
+ if (e.key === 'Enter') {
190
+ e.preventDefault();
191
+ }
192
+ } else {
193
+ onCreateTag(value);
194
+ setInputValue('');
195
+ e.preventDefault();
196
+ }
197
+ } else if (e.key === 'Enter') {
198
+ e.preventDefault();
199
+ }
200
+ } else if (e.key === 'ArrowUp') {
201
+ setCurrentHighlightedIndex(currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1);
202
+ e.preventDefault();
203
+ } else if (e.key === 'ArrowDown') {
204
+ setCurrentHighlightedIndex(currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1);
205
+ e.preventDefault();
206
+ }
207
+ }
208
+ }), ___EmotionJSX(Tooltip, {
209
+ tooltip: expanded ? t('tagSelector.hideAllTags') : t('tagSelector.showAllTags')
210
+ }, ___EmotionJSX(IconButtonDualStates, {
211
+ "data-suggestionbutton": true,
212
+ ariaLabelActive: t('tagSelector.showAllTags'),
213
+ ariaLabelInActive: t('tagSelector.hideAllTags'),
214
+ active: expanded,
215
+ greyLighter: true,
216
+ inactiveIcon: ___EmotionJSX(ChevronDown, null),
217
+ activeIcon: ___EmotionJSX(ChevronUp, null),
218
+ size: "small",
219
+ "aria-controls": suggestionIdRef.current,
220
+ onClick: function onClick() {
221
+ var _inputRef$current2;
222
+
223
+ setInputValue('');
224
+ setExpanded(!expanded);
225
+ (_inputRef$current2 = inputRef.current) === null || _inputRef$current2 === void 0 ? void 0 : _inputRef$current2.focus();
226
+ }
227
+ })))), ___EmotionJSX("div", {
228
+ id: suggestionIdRef.current,
229
+ "aria-live": "polite"
230
+ }, (hasFocus || expanded) && suggestions.length > 0 ? ___EmotionJSX(Suggestions, {
231
+ inline: inline,
232
+ dropdownMaxHeight: dropdownMaxHeight,
233
+ suggestions: suggestions,
234
+ currentHighlightedIndex: currentHighlightedIndex,
235
+ onToggleTag: onToggleTag,
236
+ hasBeenAdded: hasBeenAdded
237
+ }) : null));
238
+ };
239
+
240
+ export default SuggestionInput;
@@ -0,0 +1,93 @@
1
+ import _styled from "@emotion/styled-base";
2
+
3
+ function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
4
+
5
+ /*
6
+ * Copyright (c) 2022-present, NDLA.
7
+ *
8
+ * This source code is licensed under the GPLv3 license found in the
9
+ * LICENSE file in the root directory of this source tree.
10
+ *
11
+ */
12
+ import React from 'react';
13
+ import { Check } from '@ndla/icons/editor';
14
+ import { spacing, colors, misc, animations, fonts, shadows } from '@ndla/core';
15
+ import Button from '@ndla/button';
16
+ import { jsx as ___EmotionJSX } from "@emotion/core";
17
+ var ABSOLUTE_DROPDOWN_MAXHEIGHT = '360px';
18
+
19
+ var CheckedIcon = /*#__PURE__*/_styled(Check, {
20
+ target: "e1jg8lzr0",
21
+ label: "CheckedIcon"
22
+ })("width:", spacing.normal, ";height:", spacing.normal, ";fill:", colors.brand.light, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlN1Z2dlc3Rpb25zLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFpQmlDIiwiZmlsZSI6IlN1Z2dlc3Rpb25zLnRzeCIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMjItcHJlc2VudCwgTkRMQS5cbiAqXG4gKiBUaGlzIHNvdXJjZSBjb2RlIGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBHUEx2MyBsaWNlbnNlIGZvdW5kIGluIHRoZVxuICogTElDRU5TRSBmaWxlIGluIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGlzIHNvdXJjZSB0cmVlLlxuICpcbiAqL1xuXG5pbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQ2hlY2sgfSBmcm9tICdAbmRsYS9pY29ucy9lZGl0b3InO1xuaW1wb3J0IHsgc3BhY2luZywgY29sb3JzLCBtaXNjLCBhbmltYXRpb25zLCBmb250cywgc2hhZG93cyB9IGZyb20gJ0BuZGxhL2NvcmUnO1xuaW1wb3J0IEJ1dHRvbiBmcm9tICdAbmRsYS9idXR0b24nO1xuaW1wb3J0IHR5cGUgeyBUYWdUeXBlIH0gZnJvbSAnLi9UYWdTZWxlY3Rvcic7XG5cbmNvbnN0IEFCU09MVVRFX0RST1BET1dOX01BWEhFSUdIVCA9ICczNjBweCc7XG5cbmNvbnN0IENoZWNrZWRJY29uID0gc3R5bGVkKENoZWNrKWBcbiAgd2lkdGg6ICR7c3BhY2luZy5ub3JtYWx9O1xuICBoZWlnaHQ6ICR7c3BhY2luZy5ub3JtYWx9O1xuICBmaWxsOiAke2NvbG9ycy5icmFuZC5saWdodH07XG5gO1xuXG5pbnRlcmZhY2UgU3VnZ2VzdGlvbnNXcmFwcGVyUHJvcHMge1xuICBkcm9wZG93bk1heEhlaWdodDogc3RyaW5nO1xuICBpbmxpbmU/OiBib29sZWFuO1xufVxuXG5jb25zdCBTdWdnZXN0aW9uc1dyYXBwZXIgPSBzdHlsZWQuZGl2YFxuICBwb3NpdGlvbjogcmVsYXRpdmU7XG5gO1xuXG5jb25zdCBTdWdnZXN0aW9ucyA9IHN0eWxlZC5kaXY8U3VnZ2VzdGlvbnNXcmFwcGVyUHJvcHM+YFxuICBwb3NpdGlvbjogJHsoeyBpbmxpbmUgfSkgPT4gKGlubGluZSA/ICdzdGF0aWMnIDogJ2Fic29sdXRlJyl9O1xuICB6LWluZGV4OiA5OTk5OTtcbiAgcmlnaHQ6IDA7XG4gIGxlZnQ6IDA7XG4gIGJveC1zaGFkb3c6ICR7c2hhZG93cy5sZXZpdGF0ZTF9O1xuICBtYXJnaW46IDAgJHtzcGFjaW5nLnNtYWxsfTtcbiAgcGFkZGluZzogJHtzcGFjaW5nLnNtYWxsfSAwO1xuICBvdmVyZmxvdy15OiBzY3JvbGw7XG4gIHNjcm9sbC1iZWhhdmlvcjogc21vb3RoO1xuICBtYXgtaGVpZ2h0OiBtaW4oJHsoeyBkcm9wZG93bk1heEhlaWdodCB9KSA9PiBkcm9wZG93bk1heEhlaWdodH0sICR7QUJTT0xVVEVfRFJPUERPV05fTUFYSEVJR0hUfSk7XG4gIGJvcmRlci1yYWRpdXM6ICR7bWlzYy5ib3JkZXJSYWRpdXN9O1xuICBiYWNrZ3JvdW5kOiAke2NvbG9ycy53aGl0ZX07XG4gICR7YW5pbWF0aW9ucy5mYWRlSW4oYW5pbWF0aW9ucy5kdXJhdGlvbnMuZmFzdCl9XG5gO1xuXG5jb25zdCBTdWdnZXN0aW9uTGlzdCA9IHN0eWxlZC5kaXZgXG4gIG9wYWNpdHk6IDA7XG4gICR7YW5pbWF0aW9ucy5mYWRlSW5Cb3R0b20oKX1cbiAgYW5pbWF0aW9uLWRlbGF5OiAke2FuaW1hdGlvbnMuZHVyYXRpb25zLmZhc3R9O1xuICBhbmltYXRpb24tZmlsbC1tb2RlOiBmb3J3YXJkcztcbmA7XG5cbmludGVyZmFjZSBTdWdnZXN0aW9uQnV0dG9uUHJvcHMge1xuICBpc0hpZ2hsaWdodGVkOiBib29sZWFuO1xufVxuXG5jb25zdCBTdWdnZXN0aW9uQnV0dG9uID0gc3R5bGVkKEJ1dHRvbik8U3VnZ2VzdGlvbkJ1dHRvblByb3BzPmBcbiAgZGlzcGxheTogZmxleDtcbiAganVzdGlmeS1jb250ZW50OiBzcGFjZS1iZXR3ZWVuO1xuICAke2ZvbnRzLnNpemVzKDE4KX07XG4gIHRyYW5zaXRpb246ICR7bWlzYy50cmFuc2l0aW9uLmRlZmF1bHR9O1xuICBmb250LXdlaWdodDogNDAwO1xuXG4gICY6ZGlzYWJsZWQge1xuICAgIGNvbG9yOiAke2NvbG9ycy5icmFuZC5ncmV5TWVkaXVtfTtcbiAgICAmOmhvdmVyIHtcbiAgICAgIHN2ZyB7XG4gICAgICAgIGZpbGw6ICR7Y29sb3JzLmJyYW5kLmdyZXlMaWdodH07XG4gICAgICB9XG4gICAgfVxuICB9XG5gO1xuXG5pbnRlcmZhY2UgUHJvcHMge1xuICBpbmxpbmU/OiBib29sZWFuO1xuICBkcm9wZG93bk1heEhlaWdodDogc3RyaW5nO1xuICBzdWdnZXN0aW9uczogVGFnVHlwZVtdO1xuICBjdXJyZW50SGlnaGxpZ2h0ZWRJbmRleDogbnVtYmVyO1xuICBvblRvZ2dsZVRhZzogKGlkOiBzdHJpbmcpID0+IHZvaWQ7XG4gIGhhc0JlZW5BZGRlZDogKGlkOiBzdHJpbmcpID0+IGJvb2xlYW47XG59XG5cbmNvbnN0IFRhZ1N1Z2dlc3Rpb25zID0gKHtcbiAgaW5saW5lLFxuICBkcm9wZG93bk1heEhlaWdodCxcbiAgc3VnZ2VzdGlvbnMsXG4gIGN1cnJlbnRIaWdobGlnaHRlZEluZGV4LFxuICBvblRvZ2dsZVRhZyxcbiAgaGFzQmVlbkFkZGVkLFxufTogUHJvcHMpID0+IChcbiAgPFN1Z2dlc3Rpb25zV3JhcHBlcj5cbiAgICA8U3VnZ2VzdGlvbnMgaW5saW5lPXtpbmxpbmV9IGRyb3Bkb3duTWF4SGVpZ2h0PXtkcm9wZG93bk1heEhlaWdodH0+XG4gICAgICA8U3VnZ2VzdGlvbkxpc3Qgcm9sZT1cImxpc3Rib3hcIj5cbiAgICAgICAge3N1Z2dlc3Rpb25zLm1hcCgoeyBpZCwgbmFtZSB9LCBpbmRleDogbnVtYmVyKSA9PiB7XG4gICAgICAgICAgY29uc3QgYWxyZWFkeUFkZGVkID0gaGFzQmVlbkFkZGVkKGlkKTtcbiAgICAgICAgICBjb25zdCBzZWxlY3RlZCA9IGluZGV4ID09PSBjdXJyZW50SGlnaGxpZ2h0ZWRJbmRleDtcbiAgICAgICAgICByZXR1cm4gKFxuICAgICAgICAgICAgPFN1Z2dlc3Rpb25CdXR0b25cbiAgICAgICAgICAgICAgYm9yZGVyU2hhcGU9XCJzaGFycGVuZWRcIlxuICAgICAgICAgICAgICBnaG9zdFBpbGxcbiAgICAgICAgICAgICAgd2lkdGg9XCJmdWxsXCJcbiAgICAgICAgICAgICAgdGV4dEFsaWduPVwibGVmdFwiXG4gICAgICAgICAgICAgIGRhdGEtc3VnZ2VzdGlvbmJ1dHRvblxuICAgICAgICAgICAgICByb2xlPVwib3B0aW9uXCJcbiAgICAgICAgICAgICAgYXJpYS1zZWxlY3RlZD17c2VsZWN0ZWR9XG4gICAgICAgICAgICAgIGRpc2FibGVkPXthbHJlYWR5QWRkZWR9XG4gICAgICAgICAgICAgIGlzSGlnaGxpZ2h0ZWQ9e3NlbGVjdGVkfVxuICAgICAgICAgICAgICBvbk1vdXNlRG93bj17KCkgPT4ge1xuICAgICAgICAgICAgICAgIG9uVG9nZ2xlVGFnKGlkKTtcbiAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAga2V5PXtpZH0+XG4gICAgICAgICAgICAgIDxzcGFuPntuYW1lfTwvc3Bhbj5cbiAgICAgICAgICAgICAge2FscmVhZHlBZGRlZCAmJiA8Q2hlY2tlZEljb24gLz59XG4gICAgICAgICAgICA8L1N1Z2dlc3Rpb25CdXR0b24+XG4gICAgICAgICAgKTtcbiAgICAgICAgfSl9XG4gICAgICA8L1N1Z2dlc3Rpb25MaXN0PlxuICAgIDwvU3VnZ2VzdGlvbnM+XG4gIDwvU3VnZ2VzdGlvbnNXcmFwcGVyPlxuKTtcblxuZXhwb3J0IGRlZmF1bHQgVGFnU3VnZ2VzdGlvbnM7XG4iXX0= */"));
23
+
24
+ var SuggestionsWrapper = _styled("div", {
25
+ target: "e1jg8lzr1",
26
+ label: "SuggestionsWrapper"
27
+ })(process.env.NODE_ENV === "production" ? {
28
+ name: "79elbk",
29
+ styles: "position:relative;"
30
+ } : {
31
+ name: "79elbk",
32
+ styles: "position:relative;",
33
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlN1Z2dlc3Rpb25zLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUE0QnFDIiwiZmlsZSI6IlN1Z2dlc3Rpb25zLnRzeCIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMjItcHJlc2VudCwgTkRMQS5cbiAqXG4gKiBUaGlzIHNvdXJjZSBjb2RlIGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBHUEx2MyBsaWNlbnNlIGZvdW5kIGluIHRoZVxuICogTElDRU5TRSBmaWxlIGluIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGlzIHNvdXJjZSB0cmVlLlxuICpcbiAqL1xuXG5pbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQ2hlY2sgfSBmcm9tICdAbmRsYS9pY29ucy9lZGl0b3InO1xuaW1wb3J0IHsgc3BhY2luZywgY29sb3JzLCBtaXNjLCBhbmltYXRpb25zLCBmb250cywgc2hhZG93cyB9IGZyb20gJ0BuZGxhL2NvcmUnO1xuaW1wb3J0IEJ1dHRvbiBmcm9tICdAbmRsYS9idXR0b24nO1xuaW1wb3J0IHR5cGUgeyBUYWdUeXBlIH0gZnJvbSAnLi9UYWdTZWxlY3Rvcic7XG5cbmNvbnN0IEFCU09MVVRFX0RST1BET1dOX01BWEhFSUdIVCA9ICczNjBweCc7XG5cbmNvbnN0IENoZWNrZWRJY29uID0gc3R5bGVkKENoZWNrKWBcbiAgd2lkdGg6ICR7c3BhY2luZy5ub3JtYWx9O1xuICBoZWlnaHQ6ICR7c3BhY2luZy5ub3JtYWx9O1xuICBmaWxsOiAke2NvbG9ycy5icmFuZC5saWdodH07XG5gO1xuXG5pbnRlcmZhY2UgU3VnZ2VzdGlvbnNXcmFwcGVyUHJvcHMge1xuICBkcm9wZG93bk1heEhlaWdodDogc3RyaW5nO1xuICBpbmxpbmU/OiBib29sZWFuO1xufVxuXG5jb25zdCBTdWdnZXN0aW9uc1dyYXBwZXIgPSBzdHlsZWQuZGl2YFxuICBwb3NpdGlvbjogcmVsYXRpdmU7XG5gO1xuXG5jb25zdCBTdWdnZXN0aW9ucyA9IHN0eWxlZC5kaXY8U3VnZ2VzdGlvbnNXcmFwcGVyUHJvcHM+YFxuICBwb3NpdGlvbjogJHsoeyBpbmxpbmUgfSkgPT4gKGlubGluZSA/ICdzdGF0aWMnIDogJ2Fic29sdXRlJyl9O1xuICB6LWluZGV4OiA5OTk5OTtcbiAgcmlnaHQ6IDA7XG4gIGxlZnQ6IDA7XG4gIGJveC1zaGFkb3c6ICR7c2hhZG93cy5sZXZpdGF0ZTF9O1xuICBtYXJnaW46IDAgJHtzcGFjaW5nLnNtYWxsfTtcbiAgcGFkZGluZzogJHtzcGFjaW5nLnNtYWxsfSAwO1xuICBvdmVyZmxvdy15OiBzY3JvbGw7XG4gIHNjcm9sbC1iZWhhdmlvcjogc21vb3RoO1xuICBtYXgtaGVpZ2h0OiBtaW4oJHsoeyBkcm9wZG93bk1heEhlaWdodCB9KSA9PiBkcm9wZG93bk1heEhlaWdodH0sICR7QUJTT0xVVEVfRFJPUERPV05fTUFYSEVJR0hUfSk7XG4gIGJvcmRlci1yYWRpdXM6ICR7bWlzYy5ib3JkZXJSYWRpdXN9O1xuICBiYWNrZ3JvdW5kOiAke2NvbG9ycy53aGl0ZX07XG4gICR7YW5pbWF0aW9ucy5mYWRlSW4oYW5pbWF0aW9ucy5kdXJhdGlvbnMuZmFzdCl9XG5gO1xuXG5jb25zdCBTdWdnZXN0aW9uTGlzdCA9IHN0eWxlZC5kaXZgXG4gIG9wYWNpdHk6IDA7XG4gICR7YW5pbWF0aW9ucy5mYWRlSW5Cb3R0b20oKX1cbiAgYW5pbWF0aW9uLWRlbGF5OiAke2FuaW1hdGlvbnMuZHVyYXRpb25zLmZhc3R9O1xuICBhbmltYXRpb24tZmlsbC1tb2RlOiBmb3J3YXJkcztcbmA7XG5cbmludGVyZmFjZSBTdWdnZXN0aW9uQnV0dG9uUHJvcHMge1xuICBpc0hpZ2hsaWdodGVkOiBib29sZWFuO1xufVxuXG5jb25zdCBTdWdnZXN0aW9uQnV0dG9uID0gc3R5bGVkKEJ1dHRvbik8U3VnZ2VzdGlvbkJ1dHRvblByb3BzPmBcbiAgZGlzcGxheTogZmxleDtcbiAganVzdGlmeS1jb250ZW50OiBzcGFjZS1iZXR3ZWVuO1xuICAke2ZvbnRzLnNpemVzKDE4KX07XG4gIHRyYW5zaXRpb246ICR7bWlzYy50cmFuc2l0aW9uLmRlZmF1bHR9O1xuICBmb250LXdlaWdodDogNDAwO1xuXG4gICY6ZGlzYWJsZWQge1xuICAgIGNvbG9yOiAke2NvbG9ycy5icmFuZC5ncmV5TWVkaXVtfTtcbiAgICAmOmhvdmVyIHtcbiAgICAgIHN2ZyB7XG4gICAgICAgIGZpbGw6ICR7Y29sb3JzLmJyYW5kLmdyZXlMaWdodH07XG4gICAgICB9XG4gICAgfVxuICB9XG5gO1xuXG5pbnRlcmZhY2UgUHJvcHMge1xuICBpbmxpbmU/OiBib29sZWFuO1xuICBkcm9wZG93bk1heEhlaWdodDogc3RyaW5nO1xuICBzdWdnZXN0aW9uczogVGFnVHlwZVtdO1xuICBjdXJyZW50SGlnaGxpZ2h0ZWRJbmRleDogbnVtYmVyO1xuICBvblRvZ2dsZVRhZzogKGlkOiBzdHJpbmcpID0+IHZvaWQ7XG4gIGhhc0JlZW5BZGRlZDogKGlkOiBzdHJpbmcpID0+IGJvb2xlYW47XG59XG5cbmNvbnN0IFRhZ1N1Z2dlc3Rpb25zID0gKHtcbiAgaW5saW5lLFxuICBkcm9wZG93bk1heEhlaWdodCxcbiAgc3VnZ2VzdGlvbnMsXG4gIGN1cnJlbnRIaWdobGlnaHRlZEluZGV4LFxuICBvblRvZ2dsZVRhZyxcbiAgaGFzQmVlbkFkZGVkLFxufTogUHJvcHMpID0+IChcbiAgPFN1Z2dlc3Rpb25zV3JhcHBlcj5cbiAgICA8U3VnZ2VzdGlvbnMgaW5saW5lPXtpbmxpbmV9IGRyb3Bkb3duTWF4SGVpZ2h0PXtkcm9wZG93bk1heEhlaWdodH0+XG4gICAgICA8U3VnZ2VzdGlvbkxpc3Qgcm9sZT1cImxpc3Rib3hcIj5cbiAgICAgICAge3N1Z2dlc3Rpb25zLm1hcCgoeyBpZCwgbmFtZSB9LCBpbmRleDogbnVtYmVyKSA9PiB7XG4gICAgICAgICAgY29uc3QgYWxyZWFkeUFkZGVkID0gaGFzQmVlbkFkZGVkKGlkKTtcbiAgICAgICAgICBjb25zdCBzZWxlY3RlZCA9IGluZGV4ID09PSBjdXJyZW50SGlnaGxpZ2h0ZWRJbmRleDtcbiAgICAgICAgICByZXR1cm4gKFxuICAgICAgICAgICAgPFN1Z2dlc3Rpb25CdXR0b25cbiAgICAgICAgICAgICAgYm9yZGVyU2hhcGU9XCJzaGFycGVuZWRcIlxuICAgICAgICAgICAgICBnaG9zdFBpbGxcbiAgICAgICAgICAgICAgd2lkdGg9XCJmdWxsXCJcbiAgICAgICAgICAgICAgdGV4dEFsaWduPVwibGVmdFwiXG4gICAgICAgICAgICAgIGRhdGEtc3VnZ2VzdGlvbmJ1dHRvblxuICAgICAgICAgICAgICByb2xlPVwib3B0aW9uXCJcbiAgICAgICAgICAgICAgYXJpYS1zZWxlY3RlZD17c2VsZWN0ZWR9XG4gICAgICAgICAgICAgIGRpc2FibGVkPXthbHJlYWR5QWRkZWR9XG4gICAgICAgICAgICAgIGlzSGlnaGxpZ2h0ZWQ9e3NlbGVjdGVkfVxuICAgICAgICAgICAgICBvbk1vdXNlRG93bj17KCkgPT4ge1xuICAgICAgICAgICAgICAgIG9uVG9nZ2xlVGFnKGlkKTtcbiAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAga2V5PXtpZH0+XG4gICAgICAgICAgICAgIDxzcGFuPntuYW1lfTwvc3Bhbj5cbiAgICAgICAgICAgICAge2FscmVhZHlBZGRlZCAmJiA8Q2hlY2tlZEljb24gLz59XG4gICAgICAgICAgICA8L1N1Z2dlc3Rpb25CdXR0b24+XG4gICAgICAgICAgKTtcbiAgICAgICAgfSl9XG4gICAgICA8L1N1Z2dlc3Rpb25MaXN0PlxuICAgIDwvU3VnZ2VzdGlvbnM+XG4gIDwvU3VnZ2VzdGlvbnNXcmFwcGVyPlxuKTtcblxuZXhwb3J0IGRlZmF1bHQgVGFnU3VnZ2VzdGlvbnM7XG4iXX0= */",
34
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
35
+ });
36
+
37
+ var Suggestions = _styled("div", {
38
+ target: "e1jg8lzr2",
39
+ label: "Suggestions"
40
+ })("position:", function (_ref) {
41
+ var inline = _ref.inline;
42
+ return inline ? 'static' : 'absolute';
43
+ }, ";z-index:99999;right:0;left:0;box-shadow:", shadows.levitate1, ";margin:0 ", spacing.small, ";padding:", spacing.small, " 0;overflow-y:scroll;scroll-behavior:smooth;max-height:min(", function (_ref2) {
44
+ var dropdownMaxHeight = _ref2.dropdownMaxHeight;
45
+ return dropdownMaxHeight;
46
+ }, ",", ABSOLUTE_DROPDOWN_MAXHEIGHT, ");border-radius:", misc.borderRadius, ";background:", colors.white, ";", animations.fadeIn(animations.durations.fast), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlN1Z2dlc3Rpb25zLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFnQ3VEIiwiZmlsZSI6IlN1Z2dlc3Rpb25zLnRzeCIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMjItcHJlc2VudCwgTkRMQS5cbiAqXG4gKiBUaGlzIHNvdXJjZSBjb2RlIGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBHUEx2MyBsaWNlbnNlIGZvdW5kIGluIHRoZVxuICogTElDRU5TRSBmaWxlIGluIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGlzIHNvdXJjZSB0cmVlLlxuICpcbiAqL1xuXG5pbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQ2hlY2sgfSBmcm9tICdAbmRsYS9pY29ucy9lZGl0b3InO1xuaW1wb3J0IHsgc3BhY2luZywgY29sb3JzLCBtaXNjLCBhbmltYXRpb25zLCBmb250cywgc2hhZG93cyB9IGZyb20gJ0BuZGxhL2NvcmUnO1xuaW1wb3J0IEJ1dHRvbiBmcm9tICdAbmRsYS9idXR0b24nO1xuaW1wb3J0IHR5cGUgeyBUYWdUeXBlIH0gZnJvbSAnLi9UYWdTZWxlY3Rvcic7XG5cbmNvbnN0IEFCU09MVVRFX0RST1BET1dOX01BWEhFSUdIVCA9ICczNjBweCc7XG5cbmNvbnN0IENoZWNrZWRJY29uID0gc3R5bGVkKENoZWNrKWBcbiAgd2lkdGg6ICR7c3BhY2luZy5ub3JtYWx9O1xuICBoZWlnaHQ6ICR7c3BhY2luZy5ub3JtYWx9O1xuICBmaWxsOiAke2NvbG9ycy5icmFuZC5saWdodH07XG5gO1xuXG5pbnRlcmZhY2UgU3VnZ2VzdGlvbnNXcmFwcGVyUHJvcHMge1xuICBkcm9wZG93bk1heEhlaWdodDogc3RyaW5nO1xuICBpbmxpbmU/OiBib29sZWFuO1xufVxuXG5jb25zdCBTdWdnZXN0aW9uc1dyYXBwZXIgPSBzdHlsZWQuZGl2YFxuICBwb3NpdGlvbjogcmVsYXRpdmU7XG5gO1xuXG5jb25zdCBTdWdnZXN0aW9ucyA9IHN0eWxlZC5kaXY8U3VnZ2VzdGlvbnNXcmFwcGVyUHJvcHM+YFxuICBwb3NpdGlvbjogJHsoeyBpbmxpbmUgfSkgPT4gKGlubGluZSA/ICdzdGF0aWMnIDogJ2Fic29sdXRlJyl9O1xuICB6LWluZGV4OiA5OTk5OTtcbiAgcmlnaHQ6IDA7XG4gIGxlZnQ6IDA7XG4gIGJveC1zaGFkb3c6ICR7c2hhZG93cy5sZXZpdGF0ZTF9O1xuICBtYXJnaW46IDAgJHtzcGFjaW5nLnNtYWxsfTtcbiAgcGFkZGluZzogJHtzcGFjaW5nLnNtYWxsfSAwO1xuICBvdmVyZmxvdy15OiBzY3JvbGw7XG4gIHNjcm9sbC1iZWhhdmlvcjogc21vb3RoO1xuICBtYXgtaGVpZ2h0OiBtaW4oJHsoeyBkcm9wZG93bk1heEhlaWdodCB9KSA9PiBkcm9wZG93bk1heEhlaWdodH0sICR7QUJTT0xVVEVfRFJPUERPV05fTUFYSEVJR0hUfSk7XG4gIGJvcmRlci1yYWRpdXM6ICR7bWlzYy5ib3JkZXJSYWRpdXN9O1xuICBiYWNrZ3JvdW5kOiAke2NvbG9ycy53aGl0ZX07XG4gICR7YW5pbWF0aW9ucy5mYWRlSW4oYW5pbWF0aW9ucy5kdXJhdGlvbnMuZmFzdCl9XG5gO1xuXG5jb25zdCBTdWdnZXN0aW9uTGlzdCA9IHN0eWxlZC5kaXZgXG4gIG9wYWNpdHk6IDA7XG4gICR7YW5pbWF0aW9ucy5mYWRlSW5Cb3R0b20oKX1cbiAgYW5pbWF0aW9uLWRlbGF5OiAke2FuaW1hdGlvbnMuZHVyYXRpb25zLmZhc3R9O1xuICBhbmltYXRpb24tZmlsbC1tb2RlOiBmb3J3YXJkcztcbmA7XG5cbmludGVyZmFjZSBTdWdnZXN0aW9uQnV0dG9uUHJvcHMge1xuICBpc0hpZ2hsaWdodGVkOiBib29sZWFuO1xufVxuXG5jb25zdCBTdWdnZXN0aW9uQnV0dG9uID0gc3R5bGVkKEJ1dHRvbik8U3VnZ2VzdGlvbkJ1dHRvblByb3BzPmBcbiAgZGlzcGxheTogZmxleDtcbiAganVzdGlmeS1jb250ZW50OiBzcGFjZS1iZXR3ZWVuO1xuICAke2ZvbnRzLnNpemVzKDE4KX07XG4gIHRyYW5zaXRpb246ICR7bWlzYy50cmFuc2l0aW9uLmRlZmF1bHR9O1xuICBmb250LXdlaWdodDogNDAwO1xuXG4gICY6ZGlzYWJsZWQge1xuICAgIGNvbG9yOiAke2NvbG9ycy5icmFuZC5ncmV5TWVkaXVtfTtcbiAgICAmOmhvdmVyIHtcbiAgICAgIHN2ZyB7XG4gICAgICAgIGZpbGw6ICR7Y29sb3JzLmJyYW5kLmdyZXlMaWdodH07XG4gICAgICB9XG4gICAgfVxuICB9XG5gO1xuXG5pbnRlcmZhY2UgUHJvcHMge1xuICBpbmxpbmU/OiBib29sZWFuO1xuICBkcm9wZG93bk1heEhlaWdodDogc3RyaW5nO1xuICBzdWdnZXN0aW9uczogVGFnVHlwZVtdO1xuICBjdXJyZW50SGlnaGxpZ2h0ZWRJbmRleDogbnVtYmVyO1xuICBvblRvZ2dsZVRhZzogKGlkOiBzdHJpbmcpID0+IHZvaWQ7XG4gIGhhc0JlZW5BZGRlZDogKGlkOiBzdHJpbmcpID0+IGJvb2xlYW47XG59XG5cbmNvbnN0IFRhZ1N1Z2dlc3Rpb25zID0gKHtcbiAgaW5saW5lLFxuICBkcm9wZG93bk1heEhlaWdodCxcbiAgc3VnZ2VzdGlvbnMsXG4gIGN1cnJlbnRIaWdobGlnaHRlZEluZGV4LFxuICBvblRvZ2dsZVRhZyxcbiAgaGFzQmVlbkFkZGVkLFxufTogUHJvcHMpID0+IChcbiAgPFN1Z2dlc3Rpb25zV3JhcHBlcj5cbiAgICA8U3VnZ2VzdGlvbnMgaW5saW5lPXtpbmxpbmV9IGRyb3Bkb3duTWF4SGVpZ2h0PXtkcm9wZG93bk1heEhlaWdodH0+XG4gICAgICA8U3VnZ2VzdGlvbkxpc3Qgcm9sZT1cImxpc3Rib3hcIj5cbiAgICAgICAge3N1Z2dlc3Rpb25zLm1hcCgoeyBpZCwgbmFtZSB9LCBpbmRleDogbnVtYmVyKSA9PiB7XG4gICAgICAgICAgY29uc3QgYWxyZWFkeUFkZGVkID0gaGFzQmVlbkFkZGVkKGlkKTtcbiAgICAgICAgICBjb25zdCBzZWxlY3RlZCA9IGluZGV4ID09PSBjdXJyZW50SGlnaGxpZ2h0ZWRJbmRleDtcbiAgICAgICAgICByZXR1cm4gKFxuICAgICAgICAgICAgPFN1Z2dlc3Rpb25CdXR0b25cbiAgICAgICAgICAgICAgYm9yZGVyU2hhcGU9XCJzaGFycGVuZWRcIlxuICAgICAgICAgICAgICBnaG9zdFBpbGxcbiAgICAgICAgICAgICAgd2lkdGg9XCJmdWxsXCJcbiAgICAgICAgICAgICAgdGV4dEFsaWduPVwibGVmdFwiXG4gICAgICAgICAgICAgIGRhdGEtc3VnZ2VzdGlvbmJ1dHRvblxuICAgICAgICAgICAgICByb2xlPVwib3B0aW9uXCJcbiAgICAgICAgICAgICAgYXJpYS1zZWxlY3RlZD17c2VsZWN0ZWR9XG4gICAgICAgICAgICAgIGRpc2FibGVkPXthbHJlYWR5QWRkZWR9XG4gICAgICAgICAgICAgIGlzSGlnaGxpZ2h0ZWQ9e3NlbGVjdGVkfVxuICAgICAgICAgICAgICBvbk1vdXNlRG93bj17KCkgPT4ge1xuICAgICAgICAgICAgICAgIG9uVG9nZ2xlVGFnKGlkKTtcbiAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAga2V5PXtpZH0+XG4gICAgICAgICAgICAgIDxzcGFuPntuYW1lfTwvc3Bhbj5cbiAgICAgICAgICAgICAge2FscmVhZHlBZGRlZCAmJiA8Q2hlY2tlZEljb24gLz59XG4gICAgICAgICAgICA8L1N1Z2dlc3Rpb25CdXR0b24+XG4gICAgICAgICAgKTtcbiAgICAgICAgfSl9XG4gICAgICA8L1N1Z2dlc3Rpb25MaXN0PlxuICAgIDwvU3VnZ2VzdGlvbnM+XG4gIDwvU3VnZ2VzdGlvbnNXcmFwcGVyPlxuKTtcblxuZXhwb3J0IGRlZmF1bHQgVGFnU3VnZ2VzdGlvbnM7XG4iXX0= */");
47
+
48
+ var SuggestionList = _styled("div", {
49
+ target: "e1jg8lzr3",
50
+ label: "SuggestionList"
51
+ })("opacity:0;", animations.fadeInBottom(), " animation-delay:", animations.durations.fast, ";animation-fill-mode:forwards;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlN1Z2dlc3Rpb25zLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFnRGlDIiwiZmlsZSI6IlN1Z2dlc3Rpb25zLnRzeCIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMjItcHJlc2VudCwgTkRMQS5cbiAqXG4gKiBUaGlzIHNvdXJjZSBjb2RlIGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBHUEx2MyBsaWNlbnNlIGZvdW5kIGluIHRoZVxuICogTElDRU5TRSBmaWxlIGluIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGlzIHNvdXJjZSB0cmVlLlxuICpcbiAqL1xuXG5pbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQ2hlY2sgfSBmcm9tICdAbmRsYS9pY29ucy9lZGl0b3InO1xuaW1wb3J0IHsgc3BhY2luZywgY29sb3JzLCBtaXNjLCBhbmltYXRpb25zLCBmb250cywgc2hhZG93cyB9IGZyb20gJ0BuZGxhL2NvcmUnO1xuaW1wb3J0IEJ1dHRvbiBmcm9tICdAbmRsYS9idXR0b24nO1xuaW1wb3J0IHR5cGUgeyBUYWdUeXBlIH0gZnJvbSAnLi9UYWdTZWxlY3Rvcic7XG5cbmNvbnN0IEFCU09MVVRFX0RST1BET1dOX01BWEhFSUdIVCA9ICczNjBweCc7XG5cbmNvbnN0IENoZWNrZWRJY29uID0gc3R5bGVkKENoZWNrKWBcbiAgd2lkdGg6ICR7c3BhY2luZy5ub3JtYWx9O1xuICBoZWlnaHQ6ICR7c3BhY2luZy5ub3JtYWx9O1xuICBmaWxsOiAke2NvbG9ycy5icmFuZC5saWdodH07XG5gO1xuXG5pbnRlcmZhY2UgU3VnZ2VzdGlvbnNXcmFwcGVyUHJvcHMge1xuICBkcm9wZG93bk1heEhlaWdodDogc3RyaW5nO1xuICBpbmxpbmU/OiBib29sZWFuO1xufVxuXG5jb25zdCBTdWdnZXN0aW9uc1dyYXBwZXIgPSBzdHlsZWQuZGl2YFxuICBwb3NpdGlvbjogcmVsYXRpdmU7XG5gO1xuXG5jb25zdCBTdWdnZXN0aW9ucyA9IHN0eWxlZC5kaXY8U3VnZ2VzdGlvbnNXcmFwcGVyUHJvcHM+YFxuICBwb3NpdGlvbjogJHsoeyBpbmxpbmUgfSkgPT4gKGlubGluZSA/ICdzdGF0aWMnIDogJ2Fic29sdXRlJyl9O1xuICB6LWluZGV4OiA5OTk5OTtcbiAgcmlnaHQ6IDA7XG4gIGxlZnQ6IDA7XG4gIGJveC1zaGFkb3c6ICR7c2hhZG93cy5sZXZpdGF0ZTF9O1xuICBtYXJnaW46IDAgJHtzcGFjaW5nLnNtYWxsfTtcbiAgcGFkZGluZzogJHtzcGFjaW5nLnNtYWxsfSAwO1xuICBvdmVyZmxvdy15OiBzY3JvbGw7XG4gIHNjcm9sbC1iZWhhdmlvcjogc21vb3RoO1xuICBtYXgtaGVpZ2h0OiBtaW4oJHsoeyBkcm9wZG93bk1heEhlaWdodCB9KSA9PiBkcm9wZG93bk1heEhlaWdodH0sICR7QUJTT0xVVEVfRFJPUERPV05fTUFYSEVJR0hUfSk7XG4gIGJvcmRlci1yYWRpdXM6ICR7bWlzYy5ib3JkZXJSYWRpdXN9O1xuICBiYWNrZ3JvdW5kOiAke2NvbG9ycy53aGl0ZX07XG4gICR7YW5pbWF0aW9ucy5mYWRlSW4oYW5pbWF0aW9ucy5kdXJhdGlvbnMuZmFzdCl9XG5gO1xuXG5jb25zdCBTdWdnZXN0aW9uTGlzdCA9IHN0eWxlZC5kaXZgXG4gIG9wYWNpdHk6IDA7XG4gICR7YW5pbWF0aW9ucy5mYWRlSW5Cb3R0b20oKX1cbiAgYW5pbWF0aW9uLWRlbGF5OiAke2FuaW1hdGlvbnMuZHVyYXRpb25zLmZhc3R9O1xuICBhbmltYXRpb24tZmlsbC1tb2RlOiBmb3J3YXJkcztcbmA7XG5cbmludGVyZmFjZSBTdWdnZXN0aW9uQnV0dG9uUHJvcHMge1xuICBpc0hpZ2hsaWdodGVkOiBib29sZWFuO1xufVxuXG5jb25zdCBTdWdnZXN0aW9uQnV0dG9uID0gc3R5bGVkKEJ1dHRvbik8U3VnZ2VzdGlvbkJ1dHRvblByb3BzPmBcbiAgZGlzcGxheTogZmxleDtcbiAganVzdGlmeS1jb250ZW50OiBzcGFjZS1iZXR3ZWVuO1xuICAke2ZvbnRzLnNpemVzKDE4KX07XG4gIHRyYW5zaXRpb246ICR7bWlzYy50cmFuc2l0aW9uLmRlZmF1bHR9O1xuICBmb250LXdlaWdodDogNDAwO1xuXG4gICY6ZGlzYWJsZWQge1xuICAgIGNvbG9yOiAke2NvbG9ycy5icmFuZC5ncmV5TWVkaXVtfTtcbiAgICAmOmhvdmVyIHtcbiAgICAgIHN2ZyB7XG4gICAgICAgIGZpbGw6ICR7Y29sb3JzLmJyYW5kLmdyZXlMaWdodH07XG4gICAgICB9XG4gICAgfVxuICB9XG5gO1xuXG5pbnRlcmZhY2UgUHJvcHMge1xuICBpbmxpbmU/OiBib29sZWFuO1xuICBkcm9wZG93bk1heEhlaWdodDogc3RyaW5nO1xuICBzdWdnZXN0aW9uczogVGFnVHlwZVtdO1xuICBjdXJyZW50SGlnaGxpZ2h0ZWRJbmRleDogbnVtYmVyO1xuICBvblRvZ2dsZVRhZzogKGlkOiBzdHJpbmcpID0+IHZvaWQ7XG4gIGhhc0JlZW5BZGRlZDogKGlkOiBzdHJpbmcpID0+IGJvb2xlYW47XG59XG5cbmNvbnN0IFRhZ1N1Z2dlc3Rpb25zID0gKHtcbiAgaW5saW5lLFxuICBkcm9wZG93bk1heEhlaWdodCxcbiAgc3VnZ2VzdGlvbnMsXG4gIGN1cnJlbnRIaWdobGlnaHRlZEluZGV4LFxuICBvblRvZ2dsZVRhZyxcbiAgaGFzQmVlbkFkZGVkLFxufTogUHJvcHMpID0+IChcbiAgPFN1Z2dlc3Rpb25zV3JhcHBlcj5cbiAgICA8U3VnZ2VzdGlvbnMgaW5saW5lPXtpbmxpbmV9IGRyb3Bkb3duTWF4SGVpZ2h0PXtkcm9wZG93bk1heEhlaWdodH0+XG4gICAgICA8U3VnZ2VzdGlvbkxpc3Qgcm9sZT1cImxpc3Rib3hcIj5cbiAgICAgICAge3N1Z2dlc3Rpb25zLm1hcCgoeyBpZCwgbmFtZSB9LCBpbmRleDogbnVtYmVyKSA9PiB7XG4gICAgICAgICAgY29uc3QgYWxyZWFkeUFkZGVkID0gaGFzQmVlbkFkZGVkKGlkKTtcbiAgICAgICAgICBjb25zdCBzZWxlY3RlZCA9IGluZGV4ID09PSBjdXJyZW50SGlnaGxpZ2h0ZWRJbmRleDtcbiAgICAgICAgICByZXR1cm4gKFxuICAgICAgICAgICAgPFN1Z2dlc3Rpb25CdXR0b25cbiAgICAgICAgICAgICAgYm9yZGVyU2hhcGU9XCJzaGFycGVuZWRcIlxuICAgICAgICAgICAgICBnaG9zdFBpbGxcbiAgICAgICAgICAgICAgd2lkdGg9XCJmdWxsXCJcbiAgICAgICAgICAgICAgdGV4dEFsaWduPVwibGVmdFwiXG4gICAgICAgICAgICAgIGRhdGEtc3VnZ2VzdGlvbmJ1dHRvblxuICAgICAgICAgICAgICByb2xlPVwib3B0aW9uXCJcbiAgICAgICAgICAgICAgYXJpYS1zZWxlY3RlZD17c2VsZWN0ZWR9XG4gICAgICAgICAgICAgIGRpc2FibGVkPXthbHJlYWR5QWRkZWR9XG4gICAgICAgICAgICAgIGlzSGlnaGxpZ2h0ZWQ9e3NlbGVjdGVkfVxuICAgICAgICAgICAgICBvbk1vdXNlRG93bj17KCkgPT4ge1xuICAgICAgICAgICAgICAgIG9uVG9nZ2xlVGFnKGlkKTtcbiAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAga2V5PXtpZH0+XG4gICAgICAgICAgICAgIDxzcGFuPntuYW1lfTwvc3Bhbj5cbiAgICAgICAgICAgICAge2FscmVhZHlBZGRlZCAmJiA8Q2hlY2tlZEljb24gLz59XG4gICAgICAgICAgICA8L1N1Z2dlc3Rpb25CdXR0b24+XG4gICAgICAgICAgKTtcbiAgICAgICAgfSl9XG4gICAgICA8L1N1Z2dlc3Rpb25MaXN0PlxuICAgIDwvU3VnZ2VzdGlvbnM+XG4gIDwvU3VnZ2VzdGlvbnNXcmFwcGVyPlxuKTtcblxuZXhwb3J0IGRlZmF1bHQgVGFnU3VnZ2VzdGlvbnM7XG4iXX0= */"));
52
+
53
+ var SuggestionButton = /*#__PURE__*/_styled(Button, {
54
+ target: "e1jg8lzr4",
55
+ label: "SuggestionButton"
56
+ })("display:flex;justify-content:space-between;", fonts.sizes(18), ";transition:", misc.transition["default"], ";font-weight:400;&:disabled{color:", colors.brand.greyMedium, ";&:hover{svg{fill:", colors.brand.greyLight, ";}}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlN1Z2dlc3Rpb25zLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUEyRDhEIiwiZmlsZSI6IlN1Z2dlc3Rpb25zLnRzeCIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMjItcHJlc2VudCwgTkRMQS5cbiAqXG4gKiBUaGlzIHNvdXJjZSBjb2RlIGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBHUEx2MyBsaWNlbnNlIGZvdW5kIGluIHRoZVxuICogTElDRU5TRSBmaWxlIGluIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGlzIHNvdXJjZSB0cmVlLlxuICpcbiAqL1xuXG5pbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQ2hlY2sgfSBmcm9tICdAbmRsYS9pY29ucy9lZGl0b3InO1xuaW1wb3J0IHsgc3BhY2luZywgY29sb3JzLCBtaXNjLCBhbmltYXRpb25zLCBmb250cywgc2hhZG93cyB9IGZyb20gJ0BuZGxhL2NvcmUnO1xuaW1wb3J0IEJ1dHRvbiBmcm9tICdAbmRsYS9idXR0b24nO1xuaW1wb3J0IHR5cGUgeyBUYWdUeXBlIH0gZnJvbSAnLi9UYWdTZWxlY3Rvcic7XG5cbmNvbnN0IEFCU09MVVRFX0RST1BET1dOX01BWEhFSUdIVCA9ICczNjBweCc7XG5cbmNvbnN0IENoZWNrZWRJY29uID0gc3R5bGVkKENoZWNrKWBcbiAgd2lkdGg6ICR7c3BhY2luZy5ub3JtYWx9O1xuICBoZWlnaHQ6ICR7c3BhY2luZy5ub3JtYWx9O1xuICBmaWxsOiAke2NvbG9ycy5icmFuZC5saWdodH07XG5gO1xuXG5pbnRlcmZhY2UgU3VnZ2VzdGlvbnNXcmFwcGVyUHJvcHMge1xuICBkcm9wZG93bk1heEhlaWdodDogc3RyaW5nO1xuICBpbmxpbmU/OiBib29sZWFuO1xufVxuXG5jb25zdCBTdWdnZXN0aW9uc1dyYXBwZXIgPSBzdHlsZWQuZGl2YFxuICBwb3NpdGlvbjogcmVsYXRpdmU7XG5gO1xuXG5jb25zdCBTdWdnZXN0aW9ucyA9IHN0eWxlZC5kaXY8U3VnZ2VzdGlvbnNXcmFwcGVyUHJvcHM+YFxuICBwb3NpdGlvbjogJHsoeyBpbmxpbmUgfSkgPT4gKGlubGluZSA/ICdzdGF0aWMnIDogJ2Fic29sdXRlJyl9O1xuICB6LWluZGV4OiA5OTk5OTtcbiAgcmlnaHQ6IDA7XG4gIGxlZnQ6IDA7XG4gIGJveC1zaGFkb3c6ICR7c2hhZG93cy5sZXZpdGF0ZTF9O1xuICBtYXJnaW46IDAgJHtzcGFjaW5nLnNtYWxsfTtcbiAgcGFkZGluZzogJHtzcGFjaW5nLnNtYWxsfSAwO1xuICBvdmVyZmxvdy15OiBzY3JvbGw7XG4gIHNjcm9sbC1iZWhhdmlvcjogc21vb3RoO1xuICBtYXgtaGVpZ2h0OiBtaW4oJHsoeyBkcm9wZG93bk1heEhlaWdodCB9KSA9PiBkcm9wZG93bk1heEhlaWdodH0sICR7QUJTT0xVVEVfRFJPUERPV05fTUFYSEVJR0hUfSk7XG4gIGJvcmRlci1yYWRpdXM6ICR7bWlzYy5ib3JkZXJSYWRpdXN9O1xuICBiYWNrZ3JvdW5kOiAke2NvbG9ycy53aGl0ZX07XG4gICR7YW5pbWF0aW9ucy5mYWRlSW4oYW5pbWF0aW9ucy5kdXJhdGlvbnMuZmFzdCl9XG5gO1xuXG5jb25zdCBTdWdnZXN0aW9uTGlzdCA9IHN0eWxlZC5kaXZgXG4gIG9wYWNpdHk6IDA7XG4gICR7YW5pbWF0aW9ucy5mYWRlSW5Cb3R0b20oKX1cbiAgYW5pbWF0aW9uLWRlbGF5OiAke2FuaW1hdGlvbnMuZHVyYXRpb25zLmZhc3R9O1xuICBhbmltYXRpb24tZmlsbC1tb2RlOiBmb3J3YXJkcztcbmA7XG5cbmludGVyZmFjZSBTdWdnZXN0aW9uQnV0dG9uUHJvcHMge1xuICBpc0hpZ2hsaWdodGVkOiBib29sZWFuO1xufVxuXG5jb25zdCBTdWdnZXN0aW9uQnV0dG9uID0gc3R5bGVkKEJ1dHRvbik8U3VnZ2VzdGlvbkJ1dHRvblByb3BzPmBcbiAgZGlzcGxheTogZmxleDtcbiAganVzdGlmeS1jb250ZW50OiBzcGFjZS1iZXR3ZWVuO1xuICAke2ZvbnRzLnNpemVzKDE4KX07XG4gIHRyYW5zaXRpb246ICR7bWlzYy50cmFuc2l0aW9uLmRlZmF1bHR9O1xuICBmb250LXdlaWdodDogNDAwO1xuXG4gICY6ZGlzYWJsZWQge1xuICAgIGNvbG9yOiAke2NvbG9ycy5icmFuZC5ncmV5TWVkaXVtfTtcbiAgICAmOmhvdmVyIHtcbiAgICAgIHN2ZyB7XG4gICAgICAgIGZpbGw6ICR7Y29sb3JzLmJyYW5kLmdyZXlMaWdodH07XG4gICAgICB9XG4gICAgfVxuICB9XG5gO1xuXG5pbnRlcmZhY2UgUHJvcHMge1xuICBpbmxpbmU/OiBib29sZWFuO1xuICBkcm9wZG93bk1heEhlaWdodDogc3RyaW5nO1xuICBzdWdnZXN0aW9uczogVGFnVHlwZVtdO1xuICBjdXJyZW50SGlnaGxpZ2h0ZWRJbmRleDogbnVtYmVyO1xuICBvblRvZ2dsZVRhZzogKGlkOiBzdHJpbmcpID0+IHZvaWQ7XG4gIGhhc0JlZW5BZGRlZDogKGlkOiBzdHJpbmcpID0+IGJvb2xlYW47XG59XG5cbmNvbnN0IFRhZ1N1Z2dlc3Rpb25zID0gKHtcbiAgaW5saW5lLFxuICBkcm9wZG93bk1heEhlaWdodCxcbiAgc3VnZ2VzdGlvbnMsXG4gIGN1cnJlbnRIaWdobGlnaHRlZEluZGV4LFxuICBvblRvZ2dsZVRhZyxcbiAgaGFzQmVlbkFkZGVkLFxufTogUHJvcHMpID0+IChcbiAgPFN1Z2dlc3Rpb25zV3JhcHBlcj5cbiAgICA8U3VnZ2VzdGlvbnMgaW5saW5lPXtpbmxpbmV9IGRyb3Bkb3duTWF4SGVpZ2h0PXtkcm9wZG93bk1heEhlaWdodH0+XG4gICAgICA8U3VnZ2VzdGlvbkxpc3Qgcm9sZT1cImxpc3Rib3hcIj5cbiAgICAgICAge3N1Z2dlc3Rpb25zLm1hcCgoeyBpZCwgbmFtZSB9LCBpbmRleDogbnVtYmVyKSA9PiB7XG4gICAgICAgICAgY29uc3QgYWxyZWFkeUFkZGVkID0gaGFzQmVlbkFkZGVkKGlkKTtcbiAgICAgICAgICBjb25zdCBzZWxlY3RlZCA9IGluZGV4ID09PSBjdXJyZW50SGlnaGxpZ2h0ZWRJbmRleDtcbiAgICAgICAgICByZXR1cm4gKFxuICAgICAgICAgICAgPFN1Z2dlc3Rpb25CdXR0b25cbiAgICAgICAgICAgICAgYm9yZGVyU2hhcGU9XCJzaGFycGVuZWRcIlxuICAgICAgICAgICAgICBnaG9zdFBpbGxcbiAgICAgICAgICAgICAgd2lkdGg9XCJmdWxsXCJcbiAgICAgICAgICAgICAgdGV4dEFsaWduPVwibGVmdFwiXG4gICAgICAgICAgICAgIGRhdGEtc3VnZ2VzdGlvbmJ1dHRvblxuICAgICAgICAgICAgICByb2xlPVwib3B0aW9uXCJcbiAgICAgICAgICAgICAgYXJpYS1zZWxlY3RlZD17c2VsZWN0ZWR9XG4gICAgICAgICAgICAgIGRpc2FibGVkPXthbHJlYWR5QWRkZWR9XG4gICAgICAgICAgICAgIGlzSGlnaGxpZ2h0ZWQ9e3NlbGVjdGVkfVxuICAgICAgICAgICAgICBvbk1vdXNlRG93bj17KCkgPT4ge1xuICAgICAgICAgICAgICAgIG9uVG9nZ2xlVGFnKGlkKTtcbiAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAga2V5PXtpZH0+XG4gICAgICAgICAgICAgIDxzcGFuPntuYW1lfTwvc3Bhbj5cbiAgICAgICAgICAgICAge2FscmVhZHlBZGRlZCAmJiA8Q2hlY2tlZEljb24gLz59XG4gICAgICAgICAgICA8L1N1Z2dlc3Rpb25CdXR0b24+XG4gICAgICAgICAgKTtcbiAgICAgICAgfSl9XG4gICAgICA8L1N1Z2dlc3Rpb25MaXN0PlxuICAgIDwvU3VnZ2VzdGlvbnM+XG4gIDwvU3VnZ2VzdGlvbnNXcmFwcGVyPlxuKTtcblxuZXhwb3J0IGRlZmF1bHQgVGFnU3VnZ2VzdGlvbnM7XG4iXX0= */"));
57
+
58
+ var TagSuggestions = function TagSuggestions(_ref3) {
59
+ var inline = _ref3.inline,
60
+ dropdownMaxHeight = _ref3.dropdownMaxHeight,
61
+ suggestions = _ref3.suggestions,
62
+ currentHighlightedIndex = _ref3.currentHighlightedIndex,
63
+ onToggleTag = _ref3.onToggleTag,
64
+ hasBeenAdded = _ref3.hasBeenAdded;
65
+ return ___EmotionJSX(SuggestionsWrapper, null, ___EmotionJSX(Suggestions, {
66
+ inline: inline,
67
+ dropdownMaxHeight: dropdownMaxHeight
68
+ }, ___EmotionJSX(SuggestionList, {
69
+ role: "listbox"
70
+ }, suggestions.map(function (_ref4, index) {
71
+ var id = _ref4.id,
72
+ name = _ref4.name;
73
+ var alreadyAdded = hasBeenAdded(id);
74
+ var selected = index === currentHighlightedIndex;
75
+ return ___EmotionJSX(SuggestionButton, {
76
+ borderShape: "sharpened",
77
+ ghostPill: true,
78
+ width: "full",
79
+ textAlign: "left",
80
+ "data-suggestionbutton": true,
81
+ role: "option",
82
+ "aria-selected": selected,
83
+ disabled: alreadyAdded,
84
+ isHighlighted: selected,
85
+ onMouseDown: function onMouseDown() {
86
+ onToggleTag(id);
87
+ },
88
+ key: id
89
+ }, ___EmotionJSX("span", null, name), alreadyAdded && ___EmotionJSX(CheckedIcon, null));
90
+ }))));
91
+ };
92
+
93
+ export default TagSuggestions;
@@ -0,0 +1,137 @@
1
+ import _styled from "@emotion/styled-base";
2
+
3
+ function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); }
4
+
5
+ function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); }
6
+
7
+ function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }
8
+
9
+ function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; }
10
+
11
+ function _iterableToArrayLimit(arr, i) { if (typeof Symbol === "undefined" || !(Symbol.iterator in Object(arr))) return; var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"] != null) _i["return"](); } finally { if (_d) throw _e; } } return _arr; }
12
+
13
+ function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
14
+
15
+ /*
16
+ * Copyright (c) 2022-present, NDLA.
17
+ *
18
+ * This source code is licensed under the GPLv3 license found in the
19
+ * LICENSE file in the root directory of this source tree.
20
+ *
21
+ */
22
+ import React, { useState, useRef, useEffect } from 'react';
23
+ import { spacingUnit, fonts } from '@ndla/core';
24
+ import { uuid } from '@ndla/util';
25
+ import SuggestionInput from './SuggestionInput';
26
+ import { jsx as ___EmotionJSX } from "@emotion/core";
27
+ var DEFAULT_DROPDOWN_MAXHEIGHT = '240px';
28
+
29
+ var StyledLabel = _styled("label", {
30
+ target: "eor9swf0",
31
+ label: "StyledLabel"
32
+ })("font-weight:", fonts.weight.semibold, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIlRhZ1NlbGVjdG9yLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFnQmdDIiwiZmlsZSI6IlRhZ1NlbGVjdG9yLnRzeCIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMjItcHJlc2VudCwgTkRMQS5cbiAqXG4gKiBUaGlzIHNvdXJjZSBjb2RlIGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBHUEx2MyBsaWNlbnNlIGZvdW5kIGluIHRoZVxuICogTElDRU5TRSBmaWxlIGluIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGlzIHNvdXJjZSB0cmVlLlxuICpcbiAqL1xuXG5pbXBvcnQgUmVhY3QsIHsgdXNlU3RhdGUsIHVzZVJlZiwgdXNlRWZmZWN0LCBDaGFuZ2VFdmVudCB9IGZyb20gJ3JlYWN0JztcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJztcbmltcG9ydCB7IHNwYWNpbmdVbml0LCBmb250cyB9IGZyb20gJ0BuZGxhL2NvcmUnO1xuaW1wb3J0IHsgdXVpZCB9IGZyb20gJ0BuZGxhL3V0aWwnO1xuaW1wb3J0IFN1Z2dlc3Rpb25JbnB1dCBmcm9tICcuL1N1Z2dlc3Rpb25JbnB1dCc7XG5cbmNvbnN0IERFRkFVTFRfRFJPUERPV05fTUFYSEVJR0hUID0gJzI0MHB4JztcblxuY29uc3QgU3R5bGVkTGFiZWwgPSBzdHlsZWQubGFiZWxgXG4gIGZvbnQtd2VpZ2h0OiAke2ZvbnRzLndlaWdodC5zZW1pYm9sZH07XG5gO1xuXG5leHBvcnQgaW50ZXJmYWNlIFRhZ1R5cGUge1xuICBuYW1lOiBzdHJpbmc7XG4gIGlkOiBzdHJpbmc7XG59XG5cbmludGVyZmFjZSBQcm9wcyB7XG4gIGxhYmVsOiBzdHJpbmc7XG4gIHRhZ3M6IFRhZ1R5cGVbXTtcbiAgdGFnc1NlbGVjdGVkOiBzdHJpbmdbXTtcbiAgb25Ub2dnbGVUYWc6IChpZDogc3RyaW5nKSA9PiB2b2lkO1xuICBvbkNyZWF0ZVRhZzogKHRhZ05hbWU6IHN0cmluZykgPT4gdm9pZDtcbiAgaW5saW5lPzogYm9vbGVhbjtcbiAgcHJlZml4Pzogc3RyaW5nO1xufVxuXG5jb25zdCBzb3J0ZWRUYWdzID0gKHRhZ3M6IFRhZ1R5cGVbXSwgc2VsZWN0ZWRUYWdzOiBzdHJpbmdbXSk6IFRhZ1R5cGVbXSA9PlxuICB0YWdzXG4gICAgLmZpbHRlcigoeyBpZCB9KSA9PiBzZWxlY3RlZFRhZ3Muc29tZSgoaWRTZWxlY3RlZCkgPT4gaWRTZWxlY3RlZCA9PT0gaWQpKVxuICAgIC5zb3J0KChhLCBiKSA9PiBhLm5hbWUubG9jYWxlQ29tcGFyZShiLm5hbWUsICduYicpKTtcblxuY29uc3QgZ2V0U3VnZ2VzdGlvbnMgPSAodGFnczogVGFnVHlwZVtdLCBpbnB1dFZhbHVlOiBzdHJpbmcpOiBUYWdUeXBlW10gPT4ge1xuICBpZiAoaW5wdXRWYWx1ZSA9PT0gJycpIHtcbiAgICByZXR1cm4gW107XG4gIH1cbiAgY29uc3QgaW5wdXRMb3dlcmNhc2UgPSBpbnB1dFZhbHVlLnRvTG93ZXJDYXNlKCk7XG4gIHJldHVybiB0YWdzXG4gICAgLmZpbHRlcigoeyBuYW1lIH0pID0+IG5hbWUudG9Mb3dlckNhc2UoKS5zdGFydHNXaXRoKGlucHV0TG93ZXJjYXNlKSlcbiAgICAuc29ydCgoYSwgYikgPT4gYS5uYW1lLmxvY2FsZUNvbXBhcmUoYi5uYW1lLCAnbmInKSk7XG59O1xuXG5jb25zdCBUYWdTZWxlY3RvciA9ICh7IGxhYmVsLCB0YWdzLCB0YWdzU2VsZWN0ZWQsIG9uQ3JlYXRlVGFnLCBvblRvZ2dsZVRhZywgaW5saW5lLCBwcmVmaXggfTogUHJvcHMpID0+IHtcbiAgY29uc3QgW2lucHV0VmFsdWUsIHNldElucHV0VmFsdWVdID0gdXNlU3RhdGUoJycpO1xuICBjb25zdCBbZXhwYW5kZWQsIHNldEV4cGFuZGVkXSA9IHVzZVN0YXRlKGZhbHNlKTtcbiAgY29uc3QgW2Ryb3Bkb3duTWF4SGVpZ2h0LCBzZXREcm9wZG93bk1heEhlaWdodF0gPSB1c2VTdGF0ZShERUZBVUxUX0RST1BET1dOX01BWEhFSUdIVCk7XG4gIGNvbnN0IGNvbnRhaW5lclJlZiA9IHVzZVJlZjxIVE1MRGl2RWxlbWVudD4obnVsbCk7XG4gIGNvbnN0IGlucHV0SWRSZWYgPSB1c2VSZWY8c3RyaW5nPih1dWlkKCkpO1xuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgc2V0RXhwYW5kZWQoZmFsc2UpO1xuICB9LCBbdGFnc1NlbGVjdGVkXSk7XG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBjb25zdCBzZXRNYXhEcm9wZG93bk1heEhlaWdodCA9ICgpID0+IHtcbiAgICAgIGlmICghaW5saW5lICYmIGNvbnRhaW5lclJlZi5jdXJyZW50ICYmIHR5cGVvZiB3aW5kb3cgIT09ICd1bmRlZmluZWQnKSB7XG4gICAgICAgIC8vIENhbGN1bGF0ZSBkaXN0YW5jZSBmcm9tIGJvdHRvbSBvZiBjb250YWluZXIgdG8gYm90dG9tIG9mIHZpZXdwb3J0XG4gICAgICAgIGNvbnN0IGNvbnRhaW5lckJvdHRvbSA9IGNvbnRhaW5lclJlZi5jdXJyZW50LmdldEJvdW5kaW5nQ2xpZW50UmVjdCgpLmJvdHRvbTtcbiAgICAgICAgY29uc3Qgdmlld3BvcnRCb3R0b20gPSBkb2N1bWVudC5kb2N1bWVudEVsZW1lbnQuc2Nyb2xsSGVpZ2h0O1xuICAgICAgICBjb25zdCBtYXhEcm9wZG93bkhlaWdodCA9IHZpZXdwb3J0Qm90dG9tIC0gY29udGFpbmVyQm90dG9tO1xuICAgICAgICBzZXREcm9wZG93bk1heEhlaWdodChgJHttYXhEcm9wZG93bkhlaWdodCAtIHNwYWNpbmdVbml0fXB4YCk7XG4gICAgICB9XG4gICAgfTtcbiAgICBpZiAoIWlubGluZSAmJiB0eXBlb2Ygd2luZG93ICE9PSAndW5kZWZpbmVkJykge1xuICAgICAgaWYgKGV4cGFuZGVkKSB7XG4gICAgICAgIHNldE1heERyb3Bkb3duTWF4SGVpZ2h0KCk7XG4gICAgICAgIHdpbmRvdy5hZGRFdmVudExpc3RlbmVyKCdyZXNpemUnLCBzZXRNYXhEcm9wZG93bk1heEhlaWdodCk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICB3aW5kb3cucmVtb3ZlRXZlbnRMaXN0ZW5lcigncmVzaXplJywgc2V0TWF4RHJvcGRvd25NYXhIZWlnaHQpO1xuICAgICAgfVxuICAgIH1cbiAgICByZXR1cm4gKCkgPT4ge1xuICAgICAgdHlwZW9mIHdpbmRvdyAhPT0gJ3VuZGVmaW5lZCcgJiYgd2luZG93LnJlbW92ZUV2ZW50TGlzdGVuZXIoJ3Jlc2l6ZScsIHNldE1heERyb3Bkb3duTWF4SGVpZ2h0KTtcbiAgICB9O1xuICB9LCBbZXhwYW5kZWQsIGlubGluZV0pO1xuXG4gIHJldHVybiAoXG4gICAgPGRpdiByZWY9e2NvbnRhaW5lclJlZn0+XG4gICAgICA8U3R5bGVkTGFiZWwgaHRtbEZvcj17aW5wdXRJZFJlZi5jdXJyZW50fT57bGFiZWx9PC9TdHlsZWRMYWJlbD5cbiAgICAgIDxTdWdnZXN0aW9uSW5wdXRcbiAgICAgICAgb25DaGFuZ2U9eyhlOiBDaGFuZ2VFdmVudDxIVE1MSW5wdXRFbGVtZW50PikgPT4ge1xuICAgICAgICAgIGNvbnN0IHRhcmdldCA9IGUudGFyZ2V0IGFzIEhUTUxJbnB1dEVsZW1lbnQ7XG4gICAgICAgICAgc2V0SW5wdXRWYWx1ZSh0YXJnZXQudmFsdWUpO1xuICAgICAgICAgIHNldEV4cGFuZGVkKGZhbHNlKTtcbiAgICAgICAgfX1cbiAgICAgICAgc3VnZ2VzdGlvbnM9e2V4cGFuZGVkID8gdGFncyA6IGdldFN1Z2dlc3Rpb25zKHRhZ3MsIGlucHV0VmFsdWUpfVxuICAgICAgICB2YWx1ZT17aW5wdXRWYWx1ZX1cbiAgICAgICAgb25DcmVhdGVUYWc9e29uQ3JlYXRlVGFnfVxuICAgICAgICBvblRvZ2dsZVRhZz17b25Ub2dnbGVUYWd9XG4gICAgICAgIHNldElucHV0VmFsdWU9e3NldElucHV0VmFsdWV9XG4gICAgICAgIGFkZGVkVGFncz17c29ydGVkVGFncyh0YWdzLCB0YWdzU2VsZWN0ZWQpfVxuICAgICAgICBleHBhbmRlZD17ZXhwYW5kZWR9XG4gICAgICAgIHNldEV4cGFuZGVkPXtzZXRFeHBhbmRlZH1cbiAgICAgICAgZHJvcGRvd25NYXhIZWlnaHQ9e2Ryb3Bkb3duTWF4SGVpZ2h0fVxuICAgICAgICBpbmxpbmU9e2lubGluZX1cbiAgICAgICAgc2Nyb2xsQW5jaG9yRWxlbWVudD17Y29udGFpbmVyUmVmfVxuICAgICAgICBwcmVmaXg9e3ByZWZpeH1cbiAgICAgIC8+XG4gICAgPC9kaXY+XG4gICk7XG59O1xuXG5leHBvcnQgZGVmYXVsdCBUYWdTZWxlY3RvcjtcbiJdfQ== */"));
33
+
34
+ var sortedTags = function sortedTags(tags, selectedTags) {
35
+ return tags.filter(function (_ref) {
36
+ var id = _ref.id;
37
+ return selectedTags.some(function (idSelected) {
38
+ return idSelected === id;
39
+ });
40
+ }).sort(function (a, b) {
41
+ return a.name.localeCompare(b.name, 'nb');
42
+ });
43
+ };
44
+
45
+ var getSuggestions = function getSuggestions(tags, inputValue) {
46
+ if (inputValue === '') {
47
+ return [];
48
+ }
49
+
50
+ var inputLowercase = inputValue.toLowerCase();
51
+ return tags.filter(function (_ref2) {
52
+ var name = _ref2.name;
53
+ return name.toLowerCase().startsWith(inputLowercase);
54
+ }).sort(function (a, b) {
55
+ return a.name.localeCompare(b.name, 'nb');
56
+ });
57
+ };
58
+
59
+ var TagSelector = function TagSelector(_ref3) {
60
+ var label = _ref3.label,
61
+ tags = _ref3.tags,
62
+ tagsSelected = _ref3.tagsSelected,
63
+ onCreateTag = _ref3.onCreateTag,
64
+ onToggleTag = _ref3.onToggleTag,
65
+ inline = _ref3.inline,
66
+ prefix = _ref3.prefix;
67
+
68
+ var _useState = useState(''),
69
+ _useState2 = _slicedToArray(_useState, 2),
70
+ inputValue = _useState2[0],
71
+ setInputValue = _useState2[1];
72
+
73
+ var _useState3 = useState(false),
74
+ _useState4 = _slicedToArray(_useState3, 2),
75
+ expanded = _useState4[0],
76
+ setExpanded = _useState4[1];
77
+
78
+ var _useState5 = useState(DEFAULT_DROPDOWN_MAXHEIGHT),
79
+ _useState6 = _slicedToArray(_useState5, 2),
80
+ dropdownMaxHeight = _useState6[0],
81
+ setDropdownMaxHeight = _useState6[1];
82
+
83
+ var containerRef = useRef(null);
84
+ var inputIdRef = useRef(uuid());
85
+ useEffect(function () {
86
+ setExpanded(false);
87
+ }, [tagsSelected]);
88
+ useEffect(function () {
89
+ var setMaxDropdownMaxHeight = function setMaxDropdownMaxHeight() {
90
+ if (!inline && containerRef.current && typeof window !== 'undefined') {
91
+ // Calculate distance from bottom of container to bottom of viewport
92
+ var containerBottom = containerRef.current.getBoundingClientRect().bottom;
93
+ var viewportBottom = document.documentElement.scrollHeight;
94
+ var maxDropdownHeight = viewportBottom - containerBottom;
95
+ setDropdownMaxHeight("".concat(maxDropdownHeight - spacingUnit, "px"));
96
+ }
97
+ };
98
+
99
+ if (!inline && typeof window !== 'undefined') {
100
+ if (expanded) {
101
+ setMaxDropdownMaxHeight();
102
+ window.addEventListener('resize', setMaxDropdownMaxHeight);
103
+ } else {
104
+ window.removeEventListener('resize', setMaxDropdownMaxHeight);
105
+ }
106
+ }
107
+
108
+ return function () {
109
+ typeof window !== 'undefined' && window.removeEventListener('resize', setMaxDropdownMaxHeight);
110
+ };
111
+ }, [expanded, inline]);
112
+ return ___EmotionJSX("div", {
113
+ ref: containerRef
114
+ }, ___EmotionJSX(StyledLabel, {
115
+ htmlFor: inputIdRef.current
116
+ }, label), ___EmotionJSX(SuggestionInput, {
117
+ onChange: function onChange(e) {
118
+ var target = e.target;
119
+ setInputValue(target.value);
120
+ setExpanded(false);
121
+ },
122
+ suggestions: expanded ? tags : getSuggestions(tags, inputValue),
123
+ value: inputValue,
124
+ onCreateTag: onCreateTag,
125
+ onToggleTag: onToggleTag,
126
+ setInputValue: setInputValue,
127
+ addedTags: sortedTags(tags, tagsSelected),
128
+ expanded: expanded,
129
+ setExpanded: setExpanded,
130
+ dropdownMaxHeight: dropdownMaxHeight,
131
+ inline: inline,
132
+ scrollAnchorElement: containerRef,
133
+ prefix: prefix
134
+ }));
135
+ };
136
+
137
+ export default TagSelector;