@ndla/ui 17.0.0 → 19.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 (86) hide show
  1. package/es/Article/ArticleFavoritesButton.js +4 -3
  2. package/es/Masthead/MastheadAuthModal.js +8 -3
  3. package/es/MyNdla/Resource/Folder.js +11 -10
  4. package/es/MyNdla/index.js +1 -2
  5. package/es/Resource/BlockResource.js +14 -8
  6. package/es/Resource/ListResource.js +15 -9
  7. package/es/Resource/resourceComponents.js +12 -11
  8. package/es/TagSelector/SuggestionInput.js +111 -56
  9. package/es/TagSelector/Suggestions.js +19 -15
  10. package/es/TagSelector/TagSelector.js +8 -7
  11. package/es/TreeStructure/FolderItem.js +5 -5
  12. package/es/TreeStructure/FolderItems.js +8 -7
  13. package/es/TreeStructure/TreeStructure.js +65 -80
  14. package/es/TreeStructure/keyboardNavigation/keyboardNavigation.js +23 -11
  15. package/es/index.js +1 -1
  16. package/es/locale/messages-en.js +10 -1
  17. package/es/locale/messages-nb.js +11 -2
  18. package/es/locale/messages-nn.js +12 -3
  19. package/es/locale/messages-se.js +11 -2
  20. package/es/locale/messages-sma.js +11 -2
  21. package/lib/Article/ArticleFavoritesButton.js +4 -3
  22. package/lib/Masthead/MastheadAuthModal.js +14 -7
  23. package/lib/MyNdla/Resource/Folder.d.ts +4 -3
  24. package/lib/MyNdla/Resource/Folder.js +11 -10
  25. package/lib/MyNdla/index.d.ts +1 -2
  26. package/lib/MyNdla/index.js +0 -8
  27. package/lib/Resource/BlockResource.d.ts +4 -3
  28. package/lib/Resource/BlockResource.js +14 -7
  29. package/lib/Resource/ListResource.d.ts +4 -3
  30. package/lib/Resource/ListResource.js +15 -8
  31. package/lib/Resource/resourceComponents.d.ts +4 -1
  32. package/lib/Resource/resourceComponents.js +14 -13
  33. package/lib/TagSelector/SuggestionInput.js +111 -57
  34. package/lib/TagSelector/Suggestions.js +26 -23
  35. package/lib/TagSelector/TagSelector.js +8 -7
  36. package/lib/TreeStructure/FolderItem.js +5 -5
  37. package/lib/TreeStructure/FolderItems.d.ts +1 -1
  38. package/lib/TreeStructure/FolderItems.js +8 -8
  39. package/lib/TreeStructure/TreeStructure.d.ts +6 -1
  40. package/lib/TreeStructure/TreeStructure.js +66 -80
  41. package/lib/TreeStructure/TreeStructure.types.d.ts +5 -3
  42. package/lib/TreeStructure/keyboardNavigation/keyboardNavigation.js +23 -11
  43. package/lib/TreeStructure/keyboardNavigation/keyboardNavigation.types.d.ts +1 -1
  44. package/lib/index.d.ts +1 -1
  45. package/lib/index.js +0 -7
  46. package/lib/locale/messages-en.d.ts +9 -0
  47. package/lib/locale/messages-en.js +10 -1
  48. package/lib/locale/messages-nb.d.ts +9 -0
  49. package/lib/locale/messages-nb.js +11 -2
  50. package/lib/locale/messages-nn.d.ts +9 -0
  51. package/lib/locale/messages-nn.js +12 -3
  52. package/lib/locale/messages-se.d.ts +9 -0
  53. package/lib/locale/messages-se.js +11 -2
  54. package/lib/locale/messages-sma.d.ts +9 -0
  55. package/lib/locale/messages-sma.js +11 -2
  56. package/package.json +5 -5
  57. package/src/Article/ArticleFavoritesButton.tsx +4 -3
  58. package/src/Masthead/MastheadAuthModal.tsx +9 -0
  59. package/src/MyNdla/Resource/Folder.tsx +7 -7
  60. package/src/MyNdla/index.ts +1 -2
  61. package/src/Resource/BlockResource.tsx +7 -6
  62. package/src/Resource/ListResource.tsx +8 -7
  63. package/src/Resource/resourceComponents.tsx +8 -1
  64. package/src/TagSelector/SuggestionInput.tsx +90 -24
  65. package/src/TagSelector/Suggestions.tsx +14 -0
  66. package/src/TagSelector/TagSelector.tsx +6 -4
  67. package/src/TreeStructure/FolderItem.tsx +5 -2
  68. package/src/TreeStructure/FolderItems.tsx +4 -3
  69. package/src/TreeStructure/TreeStructure.tsx +54 -42
  70. package/src/TreeStructure/TreeStructure.types.ts +5 -3
  71. package/src/TreeStructure/keyboardNavigation/keyboardNavigation.ts +7 -7
  72. package/src/TreeStructure/keyboardNavigation/keyboardNavigation.types.ts +1 -1
  73. package/src/index.ts +1 -1
  74. package/src/locale/messages-en.ts +11 -1
  75. package/src/locale/messages-nb.ts +11 -2
  76. package/src/locale/messages-nn.ts +12 -3
  77. package/src/locale/messages-se.ts +11 -2
  78. package/src/locale/messages-sma.ts +11 -2
  79. package/es/MyNdla/Navigation/VerticalNavigation.js +0 -51
  80. package/es/MyNdla/Navigation/index.js +0 -2
  81. package/lib/MyNdla/Navigation/VerticalNavigation.d.ts +0 -10
  82. package/lib/MyNdla/Navigation/VerticalNavigation.js +0 -61
  83. package/lib/MyNdla/Navigation/index.d.ts +0 -2
  84. package/lib/MyNdla/Navigation/index.js +0 -15
  85. package/src/MyNdla/Navigation/VerticalNavigation.tsx +0 -93
  86. package/src/MyNdla/Navigation/index.ts +0 -2
@@ -51,49 +51,67 @@ function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
51
51
 
52
52
  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)."; }
53
53
 
54
- var Cross = ( /*#__PURE__*/0, _styledBase["default"])(_action.Cross, {
54
+ var SuggestionTextWrapper = (0, _styledBase["default"])("div", {
55
55
  target: "e55qeml0",
56
+ label: "SuggestionTextWrapper"
57
+ })(_core.fonts.sizes(18), ";position:absolute;display:flex;flex-grow:1;left:0;right:0;overflow:hidden;max-height:", _core.spacing.large, ";padding:8.333px;padding-right:", _core.spacing.large, ";span{color:", _core.colors.brand.grey, ";white-space:nowrap;overflow:hidden !important;text-overflow:ellipsis;&:first-of-type{color:transparent;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AAqBwC","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 SuggestionTextWrapper = styled.div`\n  ${fonts.sizes(18)};\n  position: absolute;\n  display: flex;\n  flex-grow: 1;\n  left: 0;\n  right: 0;\n  overflow: hidden;\n  max-height: ${spacing.large};\n  padding: 8.333px;\n  padding-right: ${spacing.large};\n  span {\n    color: ${colors.brand.grey};\n    white-space: nowrap;\n    overflow: hidden !important;\n    text-overflow: ellipsis;\n    &:first-of-type {\n      color: transparent;\n    }\n  }\n`;\n\nconst SuggestionText = ({ value, suggestionValue }: { value: string; suggestionValue: string }) => (\n  <SuggestionTextWrapper>\n    {!!value && (\n      <>\n        <span>{value}</span>\n        <span>{suggestionValue.substring(value.length)}</span>\n      </>\n    )}\n  </SuggestionTextWrapper>\n);\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  ${fonts.sizes(18)};\n  z-index: 1;\n  position: relative;\n  background: transparent;\n`;\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.neutral7};\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  position: relative;\n`;\n\nconst StyledTagButton = styled(Button)<{ enableTagButtonAnimation: boolean }>`\n  ${({ enableTagButtonAnimation }) =>\n    enableTagButtonAnimation ? animations.fadeInScaled(animations.durations.slow) : ''}\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  const initalTags = useRef<string[]>(addedTags.map(({ id }) => id));\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          <StyledTagButton\n            enableTagButtonAnimation={!initalTags.current.includes(id)}\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          </StyledTagButton>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          {suggestions[currentHighlightedIndex] && (\n            <SuggestionText value={value} suggestionValue={suggestions[currentHighlightedIndex].name} />\n          )}\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 (!['Enter', ' ', 'Tab', 'ArrowDown', 'ArrowUp', 'Backspace'].includes(e.key)) {\n                return;\n              }\n              const trimmedValue = value.replace(/\\s/g, '');\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'Backspace' && trimmedValue === '' && addedTags.length) {\n                // Remove the added last tag\n                onToggleTag(addedTags[addedTags.length - 1].id);\n                return;\n              }\n              if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (trimmedValue === '' && !expanded) {\n                if (e.key === 'Enter' || e.key === ' ') {\n                  e.preventDefault();\n                }\n                return;\n              }\n              if (e.key === 'Enter' || e.key === 'Tab' || e.key === ' ') {\n                if (suggestions.length > 0) {\n                  if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                    onToggleTag(suggestions[currentHighlightedIndex].id);\n                  } else if (trimmedValue.length < suggestions[currentHighlightedIndex].name.length) {\n                    onCreateTag(trimmedValue);\n                    e.preventDefault();\n                  }\n                  setInputValue('');\n                  e.preventDefault();\n                  return;\n                }\n                onCreateTag(trimmedValue);\n                setInputValue('');\n                e.preventDefault();\n              }\n              return;\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"]} */"));
58
+
59
+ var SuggestionText = function SuggestionText(_ref) {
60
+ var value = _ref.value,
61
+ suggestionValue = _ref.suggestionValue;
62
+ return (0, _core2.jsx)(SuggestionTextWrapper, null, !!value && (0, _core2.jsx)(_react["default"].Fragment, null, (0, _core2.jsx)("span", null, value), (0, _core2.jsx)("span", null, suggestionValue.substring(value.length))));
63
+ };
64
+
65
+ var Cross = ( /*#__PURE__*/0, _styledBase["default"])(_action.Cross, {
66
+ target: "e55qeml1",
56
67
  label: "Cross"
57
- })("margin-left:", _core.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"]} */"));
68
+ })("margin-left:", _core.spacing.xxsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AAsD8B","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 SuggestionTextWrapper = styled.div`\n  ${fonts.sizes(18)};\n  position: absolute;\n  display: flex;\n  flex-grow: 1;\n  left: 0;\n  right: 0;\n  overflow: hidden;\n  max-height: ${spacing.large};\n  padding: 8.333px;\n  padding-right: ${spacing.large};\n  span {\n    color: ${colors.brand.grey};\n    white-space: nowrap;\n    overflow: hidden !important;\n    text-overflow: ellipsis;\n    &:first-of-type {\n      color: transparent;\n    }\n  }\n`;\n\nconst SuggestionText = ({ value, suggestionValue }: { value: string; suggestionValue: string }) => (\n  <SuggestionTextWrapper>\n    {!!value && (\n      <>\n        <span>{value}</span>\n        <span>{suggestionValue.substring(value.length)}</span>\n      </>\n    )}\n  </SuggestionTextWrapper>\n);\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  ${fonts.sizes(18)};\n  z-index: 1;\n  position: relative;\n  background: transparent;\n`;\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.neutral7};\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  position: relative;\n`;\n\nconst StyledTagButton = styled(Button)<{ enableTagButtonAnimation: boolean }>`\n  ${({ enableTagButtonAnimation }) =>\n    enableTagButtonAnimation ? animations.fadeInScaled(animations.durations.slow) : ''}\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  const initalTags = useRef<string[]>(addedTags.map(({ id }) => id));\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          <StyledTagButton\n            enableTagButtonAnimation={!initalTags.current.includes(id)}\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          </StyledTagButton>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          {suggestions[currentHighlightedIndex] && (\n            <SuggestionText value={value} suggestionValue={suggestions[currentHighlightedIndex].name} />\n          )}\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 (!['Enter', ' ', 'Tab', 'ArrowDown', 'ArrowUp', 'Backspace'].includes(e.key)) {\n                return;\n              }\n              const trimmedValue = value.replace(/\\s/g, '');\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'Backspace' && trimmedValue === '' && addedTags.length) {\n                // Remove the added last tag\n                onToggleTag(addedTags[addedTags.length - 1].id);\n                return;\n              }\n              if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (trimmedValue === '' && !expanded) {\n                if (e.key === 'Enter' || e.key === ' ') {\n                  e.preventDefault();\n                }\n                return;\n              }\n              if (e.key === 'Enter' || e.key === 'Tab' || e.key === ' ') {\n                if (suggestions.length > 0) {\n                  if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                    onToggleTag(suggestions[currentHighlightedIndex].id);\n                  } else if (trimmedValue.length < suggestions[currentHighlightedIndex].name.length) {\n                    onCreateTag(trimmedValue);\n                    e.preventDefault();\n                  }\n                  setInputValue('');\n                  e.preventDefault();\n                  return;\n                }\n                onCreateTag(trimmedValue);\n                setInputValue('');\n                e.preventDefault();\n              }\n              return;\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"]} */"));
58
69
  var SuggestionInputContainer = (0, _styledBase["default"])("div", {
59
- target: "e55qeml1",
70
+ target: "e55qeml2",
60
71
  label: "SuggestionInputContainer"
61
- })("margin-bottom:", _core.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"]} */"));
72
+ })("margin-bottom:", _core.spacing.large, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AA0D2C","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 SuggestionTextWrapper = styled.div`\n  ${fonts.sizes(18)};\n  position: absolute;\n  display: flex;\n  flex-grow: 1;\n  left: 0;\n  right: 0;\n  overflow: hidden;\n  max-height: ${spacing.large};\n  padding: 8.333px;\n  padding-right: ${spacing.large};\n  span {\n    color: ${colors.brand.grey};\n    white-space: nowrap;\n    overflow: hidden !important;\n    text-overflow: ellipsis;\n    &:first-of-type {\n      color: transparent;\n    }\n  }\n`;\n\nconst SuggestionText = ({ value, suggestionValue }: { value: string; suggestionValue: string }) => (\n  <SuggestionTextWrapper>\n    {!!value && (\n      <>\n        <span>{value}</span>\n        <span>{suggestionValue.substring(value.length)}</span>\n      </>\n    )}\n  </SuggestionTextWrapper>\n);\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  ${fonts.sizes(18)};\n  z-index: 1;\n  position: relative;\n  background: transparent;\n`;\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.neutral7};\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  position: relative;\n`;\n\nconst StyledTagButton = styled(Button)<{ enableTagButtonAnimation: boolean }>`\n  ${({ enableTagButtonAnimation }) =>\n    enableTagButtonAnimation ? animations.fadeInScaled(animations.durations.slow) : ''}\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  const initalTags = useRef<string[]>(addedTags.map(({ id }) => id));\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          <StyledTagButton\n            enableTagButtonAnimation={!initalTags.current.includes(id)}\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          </StyledTagButton>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          {suggestions[currentHighlightedIndex] && (\n            <SuggestionText value={value} suggestionValue={suggestions[currentHighlightedIndex].name} />\n          )}\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 (!['Enter', ' ', 'Tab', 'ArrowDown', 'ArrowUp', 'Backspace'].includes(e.key)) {\n                return;\n              }\n              const trimmedValue = value.replace(/\\s/g, '');\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'Backspace' && trimmedValue === '' && addedTags.length) {\n                // Remove the added last tag\n                onToggleTag(addedTags[addedTags.length - 1].id);\n                return;\n              }\n              if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (trimmedValue === '' && !expanded) {\n                if (e.key === 'Enter' || e.key === ' ') {\n                  e.preventDefault();\n                }\n                return;\n              }\n              if (e.key === 'Enter' || e.key === 'Tab' || e.key === ' ') {\n                if (suggestions.length > 0) {\n                  if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                    onToggleTag(suggestions[currentHighlightedIndex].id);\n                  } else if (trimmedValue.length < suggestions[currentHighlightedIndex].name.length) {\n                    onCreateTag(trimmedValue);\n                    e.preventDefault();\n                  }\n                  setInputValue('');\n                  e.preventDefault();\n                  return;\n                }\n                onCreateTag(trimmedValue);\n                setInputValue('');\n                e.preventDefault();\n              }\n              return;\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"]} */"));
62
73
  var StyledInput = (0, _styledBase["default"])("input", {
63
- target: "e55qeml2",
74
+ target: "e55qeml3",
64
75
  label: "StyledInput"
65
- })("flex-grow:1;border:0;outline:none;background:transparent;", _core.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"]} */"));
76
+ })("flex-grow:1;border:0;outline:none;", _core.fonts.sizes(18), ";z-index:1;position:relative;background:transparent;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AA8DgC","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 SuggestionTextWrapper = styled.div`\n  ${fonts.sizes(18)};\n  position: absolute;\n  display: flex;\n  flex-grow: 1;\n  left: 0;\n  right: 0;\n  overflow: hidden;\n  max-height: ${spacing.large};\n  padding: 8.333px;\n  padding-right: ${spacing.large};\n  span {\n    color: ${colors.brand.grey};\n    white-space: nowrap;\n    overflow: hidden !important;\n    text-overflow: ellipsis;\n    &:first-of-type {\n      color: transparent;\n    }\n  }\n`;\n\nconst SuggestionText = ({ value, suggestionValue }: { value: string; suggestionValue: string }) => (\n  <SuggestionTextWrapper>\n    {!!value && (\n      <>\n        <span>{value}</span>\n        <span>{suggestionValue.substring(value.length)}</span>\n      </>\n    )}\n  </SuggestionTextWrapper>\n);\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  ${fonts.sizes(18)};\n  z-index: 1;\n  position: relative;\n  background: transparent;\n`;\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.neutral7};\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  position: relative;\n`;\n\nconst StyledTagButton = styled(Button)<{ enableTagButtonAnimation: boolean }>`\n  ${({ enableTagButtonAnimation }) =>\n    enableTagButtonAnimation ? animations.fadeInScaled(animations.durations.slow) : ''}\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  const initalTags = useRef<string[]>(addedTags.map(({ id }) => id));\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          <StyledTagButton\n            enableTagButtonAnimation={!initalTags.current.includes(id)}\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          </StyledTagButton>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          {suggestions[currentHighlightedIndex] && (\n            <SuggestionText value={value} suggestionValue={suggestions[currentHighlightedIndex].name} />\n          )}\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 (!['Enter', ' ', 'Tab', 'ArrowDown', 'ArrowUp', 'Backspace'].includes(e.key)) {\n                return;\n              }\n              const trimmedValue = value.replace(/\\s/g, '');\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'Backspace' && trimmedValue === '' && addedTags.length) {\n                // Remove the added last tag\n                onToggleTag(addedTags[addedTags.length - 1].id);\n                return;\n              }\n              if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (trimmedValue === '' && !expanded) {\n                if (e.key === 'Enter' || e.key === ' ') {\n                  e.preventDefault();\n                }\n                return;\n              }\n              if (e.key === 'Enter' || e.key === 'Tab' || e.key === ' ') {\n                if (suggestions.length > 0) {\n                  if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                    onToggleTag(suggestions[currentHighlightedIndex].id);\n                  } else if (trimmedValue.length < suggestions[currentHighlightedIndex].name.length) {\n                    onCreateTag(trimmedValue);\n                    e.preventDefault();\n                  }\n                  setInputValue('');\n                  e.preventDefault();\n                  return;\n                }\n                onCreateTag(trimmedValue);\n                setInputValue('');\n                e.preventDefault();\n              }\n              return;\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
77
  var StyledInputWrapper = (0, _styledBase["default"])("div", {
67
- target: "e55qeml3",
78
+ target: "e55qeml4",
68
79
  label: "StyledInputWrapper"
69
- })("display:flex;flex-wrap:wrap;gap:", _core.spacing.xsmall, ";padding:", _core.spacing.small, ";border:1px solid ", _core.colors.brand.greyLighter, ";transition:border-color ", _core.animations.durations.normal, " ease;border-radius:", _core.misc.borderRadius, ";&:focus-within{border-color:", _core.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"]} */"));
80
+ })("display:flex;flex-wrap:wrap;gap:", _core.spacing.xsmall, ";padding:", _core.spacing.small, ";border:1px solid ", _core.colors.brand.neutral7, ";transition:border-color ", _core.animations.durations.normal, " ease;border-radius:", _core.misc.borderRadius, ";&:focus-within{border-color:", _core.colors.brand.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AAwEqC","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 SuggestionTextWrapper = styled.div`\n  ${fonts.sizes(18)};\n  position: absolute;\n  display: flex;\n  flex-grow: 1;\n  left: 0;\n  right: 0;\n  overflow: hidden;\n  max-height: ${spacing.large};\n  padding: 8.333px;\n  padding-right: ${spacing.large};\n  span {\n    color: ${colors.brand.grey};\n    white-space: nowrap;\n    overflow: hidden !important;\n    text-overflow: ellipsis;\n    &:first-of-type {\n      color: transparent;\n    }\n  }\n`;\n\nconst SuggestionText = ({ value, suggestionValue }: { value: string; suggestionValue: string }) => (\n  <SuggestionTextWrapper>\n    {!!value && (\n      <>\n        <span>{value}</span>\n        <span>{suggestionValue.substring(value.length)}</span>\n      </>\n    )}\n  </SuggestionTextWrapper>\n);\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  ${fonts.sizes(18)};\n  z-index: 1;\n  position: relative;\n  background: transparent;\n`;\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.neutral7};\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  position: relative;\n`;\n\nconst StyledTagButton = styled(Button)<{ enableTagButtonAnimation: boolean }>`\n  ${({ enableTagButtonAnimation }) =>\n    enableTagButtonAnimation ? animations.fadeInScaled(animations.durations.slow) : ''}\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  const initalTags = useRef<string[]>(addedTags.map(({ id }) => id));\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          <StyledTagButton\n            enableTagButtonAnimation={!initalTags.current.includes(id)}\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          </StyledTagButton>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          {suggestions[currentHighlightedIndex] && (\n            <SuggestionText value={value} suggestionValue={suggestions[currentHighlightedIndex].name} />\n          )}\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 (!['Enter', ' ', 'Tab', 'ArrowDown', 'ArrowUp', 'Backspace'].includes(e.key)) {\n                return;\n              }\n              const trimmedValue = value.replace(/\\s/g, '');\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'Backspace' && trimmedValue === '' && addedTags.length) {\n                // Remove the added last tag\n                onToggleTag(addedTags[addedTags.length - 1].id);\n                return;\n              }\n              if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (trimmedValue === '' && !expanded) {\n                if (e.key === 'Enter' || e.key === ' ') {\n                  e.preventDefault();\n                }\n                return;\n              }\n              if (e.key === 'Enter' || e.key === 'Tab' || e.key === ' ') {\n                if (suggestions.length > 0) {\n                  if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                    onToggleTag(suggestions[currentHighlightedIndex].id);\n                  } else if (trimmedValue.length < suggestions[currentHighlightedIndex].name.length) {\n                    onCreateTag(trimmedValue);\n                    e.preventDefault();\n                  }\n                  setInputValue('');\n                  e.preventDefault();\n                  return;\n                }\n                onCreateTag(trimmedValue);\n                setInputValue('');\n                e.preventDefault();\n              }\n              return;\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"]} */"));
70
81
  var CombinedInputAndDropdownWrapper = (0, _styledBase["default"])("div", {
71
- target: "e55qeml4",
82
+ target: "e55qeml5",
72
83
  label: "CombinedInputAndDropdownWrapper"
73
84
  })(process.env.NODE_ENV === "production" ? {
74
- name: "1ilyui9",
75
- styles: "display:flex;flex-grow:1;"
85
+ name: "19uf65v",
86
+ styles: "display:flex;flex-grow:1;position:relative;"
76
87
  } : {
77
- name: "1ilyui9",
78
- styles: "display:flex;flex-grow:1;",
79
- 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"]} */",
88
+ name: "19uf65v",
89
+ styles: "display:flex;flex-grow:1;position:relative;",
90
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AAqFkD","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 SuggestionTextWrapper = styled.div`\n  ${fonts.sizes(18)};\n  position: absolute;\n  display: flex;\n  flex-grow: 1;\n  left: 0;\n  right: 0;\n  overflow: hidden;\n  max-height: ${spacing.large};\n  padding: 8.333px;\n  padding-right: ${spacing.large};\n  span {\n    color: ${colors.brand.grey};\n    white-space: nowrap;\n    overflow: hidden !important;\n    text-overflow: ellipsis;\n    &:first-of-type {\n      color: transparent;\n    }\n  }\n`;\n\nconst SuggestionText = ({ value, suggestionValue }: { value: string; suggestionValue: string }) => (\n  <SuggestionTextWrapper>\n    {!!value && (\n      <>\n        <span>{value}</span>\n        <span>{suggestionValue.substring(value.length)}</span>\n      </>\n    )}\n  </SuggestionTextWrapper>\n);\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  ${fonts.sizes(18)};\n  z-index: 1;\n  position: relative;\n  background: transparent;\n`;\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.neutral7};\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  position: relative;\n`;\n\nconst StyledTagButton = styled(Button)<{ enableTagButtonAnimation: boolean }>`\n  ${({ enableTagButtonAnimation }) =>\n    enableTagButtonAnimation ? animations.fadeInScaled(animations.durations.slow) : ''}\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  const initalTags = useRef<string[]>(addedTags.map(({ id }) => id));\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          <StyledTagButton\n            enableTagButtonAnimation={!initalTags.current.includes(id)}\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          </StyledTagButton>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          {suggestions[currentHighlightedIndex] && (\n            <SuggestionText value={value} suggestionValue={suggestions[currentHighlightedIndex].name} />\n          )}\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 (!['Enter', ' ', 'Tab', 'ArrowDown', 'ArrowUp', 'Backspace'].includes(e.key)) {\n                return;\n              }\n              const trimmedValue = value.replace(/\\s/g, '');\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'Backspace' && trimmedValue === '' && addedTags.length) {\n                // Remove the added last tag\n                onToggleTag(addedTags[addedTags.length - 1].id);\n                return;\n              }\n              if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (trimmedValue === '' && !expanded) {\n                if (e.key === 'Enter' || e.key === ' ') {\n                  e.preventDefault();\n                }\n                return;\n              }\n              if (e.key === 'Enter' || e.key === 'Tab' || e.key === ' ') {\n                if (suggestions.length > 0) {\n                  if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                    onToggleTag(suggestions[currentHighlightedIndex].id);\n                  } else if (trimmedValue.length < suggestions[currentHighlightedIndex].name.length) {\n                    onCreateTag(trimmedValue);\n                    e.preventDefault();\n                  }\n                  setInputValue('');\n                  e.preventDefault();\n                  return;\n                }\n                onCreateTag(trimmedValue);\n                setInputValue('');\n                e.preventDefault();\n              }\n              return;\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"]} */",
80
91
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
81
92
  });
82
-
83
- var SuggestionInput = function SuggestionInput(_ref) {
84
- var suggestions = _ref.suggestions,
85
- value = _ref.value,
86
- setInputValue = _ref.setInputValue,
87
- onCreateTag = _ref.onCreateTag,
88
- onChange = _ref.onChange,
89
- onToggleTag = _ref.onToggleTag,
90
- addedTags = _ref.addedTags,
91
- setExpanded = _ref.setExpanded,
92
- expanded = _ref.expanded,
93
- dropdownMaxHeight = _ref.dropdownMaxHeight,
94
- prefix = _ref.prefix,
95
- inline = _ref.inline,
96
- scrollAnchorElement = _ref.scrollAnchorElement;
93
+ var StyledTagButton = ( /*#__PURE__*/0, _styledBase["default"])(_button["default"], {
94
+ target: "e55qeml6",
95
+ label: "StyledTagButton"
96
+ })(function (_ref2) {
97
+ var enableTagButtonAnimation = _ref2.enableTagButtonAnimation;
98
+ return enableTagButtonAnimation ? _core.animations.fadeInScaled(_core.animations.durations.slow) : '';
99
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AA2F6E","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 SuggestionTextWrapper = styled.div`\n  ${fonts.sizes(18)};\n  position: absolute;\n  display: flex;\n  flex-grow: 1;\n  left: 0;\n  right: 0;\n  overflow: hidden;\n  max-height: ${spacing.large};\n  padding: 8.333px;\n  padding-right: ${spacing.large};\n  span {\n    color: ${colors.brand.grey};\n    white-space: nowrap;\n    overflow: hidden !important;\n    text-overflow: ellipsis;\n    &:first-of-type {\n      color: transparent;\n    }\n  }\n`;\n\nconst SuggestionText = ({ value, suggestionValue }: { value: string; suggestionValue: string }) => (\n  <SuggestionTextWrapper>\n    {!!value && (\n      <>\n        <span>{value}</span>\n        <span>{suggestionValue.substring(value.length)}</span>\n      </>\n    )}\n  </SuggestionTextWrapper>\n);\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  ${fonts.sizes(18)};\n  z-index: 1;\n  position: relative;\n  background: transparent;\n`;\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.neutral7};\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  position: relative;\n`;\n\nconst StyledTagButton = styled(Button)<{ enableTagButtonAnimation: boolean }>`\n  ${({ enableTagButtonAnimation }) =>\n    enableTagButtonAnimation ? animations.fadeInScaled(animations.durations.slow) : ''}\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  const initalTags = useRef<string[]>(addedTags.map(({ id }) => id));\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          <StyledTagButton\n            enableTagButtonAnimation={!initalTags.current.includes(id)}\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          </StyledTagButton>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          {suggestions[currentHighlightedIndex] && (\n            <SuggestionText value={value} suggestionValue={suggestions[currentHighlightedIndex].name} />\n          )}\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 (!['Enter', ' ', 'Tab', 'ArrowDown', 'ArrowUp', 'Backspace'].includes(e.key)) {\n                return;\n              }\n              const trimmedValue = value.replace(/\\s/g, '');\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'Backspace' && trimmedValue === '' && addedTags.length) {\n                // Remove the added last tag\n                onToggleTag(addedTags[addedTags.length - 1].id);\n                return;\n              }\n              if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n                return;\n              }\n              if (trimmedValue === '' && !expanded) {\n                if (e.key === 'Enter' || e.key === ' ') {\n                  e.preventDefault();\n                }\n                return;\n              }\n              if (e.key === 'Enter' || e.key === 'Tab' || e.key === ' ') {\n                if (suggestions.length > 0) {\n                  if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                    onToggleTag(suggestions[currentHighlightedIndex].id);\n                  } else if (trimmedValue.length < suggestions[currentHighlightedIndex].name.length) {\n                    onCreateTag(trimmedValue);\n                    e.preventDefault();\n                  }\n                  setInputValue('');\n                  e.preventDefault();\n                  return;\n                }\n                onCreateTag(trimmedValue);\n                setInputValue('');\n                e.preventDefault();\n              }\n              return;\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"]} */");
100
+
101
+ var SuggestionInput = function SuggestionInput(_ref3) {
102
+ var suggestions = _ref3.suggestions,
103
+ value = _ref3.value,
104
+ setInputValue = _ref3.setInputValue,
105
+ onCreateTag = _ref3.onCreateTag,
106
+ onChange = _ref3.onChange,
107
+ onToggleTag = _ref3.onToggleTag,
108
+ addedTags = _ref3.addedTags,
109
+ setExpanded = _ref3.setExpanded,
110
+ expanded = _ref3.expanded,
111
+ dropdownMaxHeight = _ref3.dropdownMaxHeight,
112
+ prefix = _ref3.prefix,
113
+ inline = _ref3.inline,
114
+ scrollAnchorElement = _ref3.scrollAnchorElement;
97
115
 
98
116
  var _useTranslation = (0, _reactI18next.useTranslation)(),
99
117
  t = _useTranslation.t;
@@ -112,6 +130,10 @@ var SuggestionInput = function SuggestionInput(_ref) {
112
130
  var inputRef = (0, _react.useRef)(null);
113
131
  var containerRef = (0, _react.useRef)(null);
114
132
  var suggestionIdRef = (0, _react.useRef)((0, _util.uuid)());
133
+ var initalTags = (0, _react.useRef)(addedTags.map(function (_ref4) {
134
+ var id = _ref4.id;
135
+ return id;
136
+ }));
115
137
  (0, _react.useEffect)(function () {
116
138
  setCurrentHighlightedIndex(0);
117
139
  }, [suggestions]);
@@ -139,18 +161,19 @@ var SuggestionInput = function SuggestionInput(_ref) {
139
161
  }, [currentHighlightedIndex]);
140
162
 
141
163
  var hasBeenAdded = function hasBeenAdded(id) {
142
- return addedTags.some(function (_ref2) {
143
- var idAdded = _ref2.id;
164
+ return addedTags.some(function (_ref5) {
165
+ var idAdded = _ref5.id;
144
166
  return idAdded === id;
145
167
  });
146
168
  };
147
169
 
148
170
  return (0, _core2.jsx)(SuggestionInputContainer, {
149
171
  ref: containerRef
150
- }, (0, _core2.jsx)(StyledInputWrapper, null, addedTags.map(function (_ref3) {
151
- var id = _ref3.id,
152
- name = _ref3.name;
153
- return (0, _core2.jsx)(_button["default"], {
172
+ }, (0, _core2.jsx)(StyledInputWrapper, null, addedTags.map(function (_ref6) {
173
+ var id = _ref6.id,
174
+ name = _ref6.name;
175
+ return (0, _core2.jsx)(StyledTagButton, {
176
+ enableTagButtonAnimation: !initalTags.current.includes(id),
154
177
  "aria-label": t('tagSelector.removeTag', {
155
178
  name: name
156
179
  }),
@@ -162,7 +185,10 @@ var SuggestionInput = function SuggestionInput(_ref) {
162
185
  key: id,
163
186
  size: "small"
164
187
  }, prefix, name, (0, _core2.jsx)(Cross, null));
165
- }), (0, _core2.jsx)(CombinedInputAndDropdownWrapper, null, (0, _core2.jsx)(StyledInput, {
188
+ }), (0, _core2.jsx)(CombinedInputAndDropdownWrapper, null, suggestions[currentHighlightedIndex] && (0, _core2.jsx)(SuggestionText, {
189
+ value: value,
190
+ suggestionValue: suggestions[currentHighlightedIndex].name
191
+ }), (0, _core2.jsx)(StyledInput, {
166
192
  placeholder: t('tagSelector.placeholder'),
167
193
  value: value,
168
194
  autoComplete: "off",
@@ -188,36 +214,64 @@ var SuggestionInput = function SuggestionInput(_ref) {
188
214
  },
189
215
  ref: inputRef,
190
216
  onKeyDown: function onKeyDown(e) {
217
+ if (!['Enter', ' ', 'Tab', 'ArrowDown', 'ArrowUp', 'Backspace'].includes(e.key)) {
218
+ return;
219
+ }
220
+
221
+ var trimmedValue = value.replace(/\s/g, '');
222
+
191
223
  if (e.key === 'Escape') {
192
224
  setExpanded(false);
193
225
  e.preventDefault();
194
- } else if (e.key === 'Enter' || e.key === 'Tab') {
195
- if (value !== '' || expanded) {
196
- if (suggestions.length > 0) {
197
- if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {
198
- onToggleTag(suggestions[currentHighlightedIndex].id);
199
- }
200
-
201
- setInputValue('');
202
-
203
- if (e.key === 'Enter') {
204
- e.preventDefault();
205
- }
206
- } else {
207
- onCreateTag(value);
208
- setInputValue('');
226
+ return;
227
+ }
228
+
229
+ if (e.key === 'Backspace' && trimmedValue === '' && addedTags.length) {
230
+ // Remove the added last tag
231
+ onToggleTag(addedTags[addedTags.length - 1].id);
232
+ return;
233
+ }
234
+
235
+ if (e.key === 'ArrowUp') {
236
+ setCurrentHighlightedIndex(currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1);
237
+ e.preventDefault();
238
+ return;
239
+ }
240
+
241
+ if (e.key === 'ArrowDown') {
242
+ setCurrentHighlightedIndex(currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1);
243
+ e.preventDefault();
244
+ return;
245
+ }
246
+
247
+ if (trimmedValue === '' && !expanded) {
248
+ if (e.key === 'Enter' || e.key === ' ') {
249
+ e.preventDefault();
250
+ }
251
+
252
+ return;
253
+ }
254
+
255
+ if (e.key === 'Enter' || e.key === 'Tab' || e.key === ' ') {
256
+ if (suggestions.length > 0) {
257
+ if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {
258
+ onToggleTag(suggestions[currentHighlightedIndex].id);
259
+ } else if (trimmedValue.length < suggestions[currentHighlightedIndex].name.length) {
260
+ onCreateTag(trimmedValue);
209
261
  e.preventDefault();
210
262
  }
211
- } else if (e.key === 'Enter') {
263
+
264
+ setInputValue('');
212
265
  e.preventDefault();
266
+ return;
213
267
  }
214
- } else if (e.key === 'ArrowUp') {
215
- setCurrentHighlightedIndex(currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1);
216
- e.preventDefault();
217
- } else if (e.key === 'ArrowDown') {
218
- setCurrentHighlightedIndex(currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1);
268
+
269
+ onCreateTag(trimmedValue);
270
+ setInputValue('');
219
271
  e.preventDefault();
220
272
  }
273
+
274
+ return;
221
275
  }
222
276
  }), (0, _core2.jsx)(_tooltip["default"], {
223
277
  tooltip: expanded ? t('tagSelector.hideAllTags') : t('tagSelector.showAllTags')