@ndla/ui 17.0.0 → 18.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 (58) hide show
  1. package/es/Article/ArticleFavoritesButton.js +4 -3
  2. package/es/MyNdla/Resource/Folder.js +6 -6
  3. package/es/MyNdla/index.js +1 -2
  4. package/es/Resource/ListResource.js +6 -6
  5. package/es/TagSelector/SuggestionInput.js +111 -56
  6. package/es/TagSelector/Suggestions.js +19 -15
  7. package/es/TagSelector/TagSelector.js +8 -7
  8. package/es/TreeStructure/FolderItem.js +5 -5
  9. package/es/index.js +1 -1
  10. package/es/locale/messages-en.js +8 -1
  11. package/es/locale/messages-nb.js +9 -2
  12. package/es/locale/messages-nn.js +10 -3
  13. package/es/locale/messages-se.js +9 -2
  14. package/es/locale/messages-sma.js +9 -2
  15. package/lib/Article/ArticleFavoritesButton.js +4 -3
  16. package/lib/MyNdla/Resource/Folder.js +6 -6
  17. package/lib/MyNdla/index.d.ts +1 -2
  18. package/lib/MyNdla/index.js +0 -8
  19. package/lib/Resource/ListResource.js +6 -6
  20. package/lib/TagSelector/SuggestionInput.js +111 -57
  21. package/lib/TagSelector/Suggestions.js +26 -23
  22. package/lib/TagSelector/TagSelector.js +8 -7
  23. package/lib/TreeStructure/FolderItem.js +5 -5
  24. package/lib/index.d.ts +1 -1
  25. package/lib/index.js +0 -7
  26. package/lib/locale/messages-en.d.ts +7 -0
  27. package/lib/locale/messages-en.js +8 -1
  28. package/lib/locale/messages-nb.d.ts +7 -0
  29. package/lib/locale/messages-nb.js +9 -2
  30. package/lib/locale/messages-nn.d.ts +7 -0
  31. package/lib/locale/messages-nn.js +10 -3
  32. package/lib/locale/messages-se.d.ts +7 -0
  33. package/lib/locale/messages-se.js +9 -2
  34. package/lib/locale/messages-sma.d.ts +7 -0
  35. package/lib/locale/messages-sma.js +9 -2
  36. package/package.json +5 -5
  37. package/src/Article/ArticleFavoritesButton.tsx +4 -3
  38. package/src/MyNdla/Resource/Folder.tsx +1 -1
  39. package/src/MyNdla/index.ts +1 -2
  40. package/src/Resource/ListResource.tsx +1 -1
  41. package/src/TagSelector/SuggestionInput.tsx +90 -24
  42. package/src/TagSelector/Suggestions.tsx +14 -0
  43. package/src/TagSelector/TagSelector.tsx +6 -4
  44. package/src/TreeStructure/FolderItem.tsx +5 -2
  45. package/src/index.ts +1 -1
  46. package/src/locale/messages-en.ts +9 -1
  47. package/src/locale/messages-nb.ts +9 -2
  48. package/src/locale/messages-nn.ts +10 -3
  49. package/src/locale/messages-se.ts +9 -2
  50. package/src/locale/messages-sma.ts +9 -2
  51. package/es/MyNdla/Navigation/VerticalNavigation.js +0 -51
  52. package/es/MyNdla/Navigation/index.js +0 -2
  53. package/lib/MyNdla/Navigation/VerticalNavigation.d.ts +0 -10
  54. package/lib/MyNdla/Navigation/VerticalNavigation.js +0 -61
  55. package/lib/MyNdla/Navigation/index.d.ts +0 -2
  56. package/lib/MyNdla/Navigation/index.js +0 -15
  57. package/src/MyNdla/Navigation/VerticalNavigation.tsx +0 -93
  58. package/src/MyNdla/Navigation/index.ts +0 -2
@@ -33,53 +33,72 @@ import { uuid } from '@ndla/util';
33
33
  import Suggestions from './Suggestions';
34
34
  import { jsx as ___EmotionJSX } from "@emotion/core";
35
35
 
36
- var Cross = /*#__PURE__*/_styled(CrossRaw, {
36
+ var SuggestionTextWrapper = _styled("div", {
37
37
  target: "e55qeml0",
38
+ label: "SuggestionTextWrapper"
39
+ })(fonts.sizes(18), ";position:absolute;display:flex;flex-grow:1;left:0;right:0;overflow:hidden;max-height:", spacing.large, ";padding:8.333px;padding-right:", spacing.large, ";span{color:", 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"]} */"));
40
+
41
+ var SuggestionText = function SuggestionText(_ref) {
42
+ var value = _ref.value,
43
+ suggestionValue = _ref.suggestionValue;
44
+ return ___EmotionJSX(SuggestionTextWrapper, null, !!value && ___EmotionJSX(React.Fragment, null, ___EmotionJSX("span", null, value), ___EmotionJSX("span", null, suggestionValue.substring(value.length))));
45
+ };
46
+
47
+ var Cross = /*#__PURE__*/_styled(CrossRaw, {
48
+ target: "e55qeml1",
38
49
  label: "Cross"
39
- })("margin-left:", spacing.xxsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AAqB8B","file":"SuggestionInput.tsx","sourcesContent":["/*\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useState, useRef, useEffect, ReactNode, RefObject, ChangeEvent, KeyboardEvent } from 'react';\nimport { isMobile } from 'react-device-detect';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport Button, { IconButtonDualStates } from '@ndla/button';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport { Cross as CrossRaw } from '@ndla/icons/action';\nimport { spacing, colors, misc, animations, fonts } from '@ndla/core';\nimport Tooltip from '@ndla/tooltip';\nimport { uuid } from '@ndla/util';\nimport Suggestions from './Suggestions';\nimport type { TagType } from './TagSelector';\n\nconst Cross = styled(CrossRaw)`\n  margin-left: ${spacing.xxsmall};\n`;\n\nconst SuggestionInputContainer = styled.div`\n  margin-bottom: ${spacing.large};\n`;\n\nconst StyledInput = styled.input`\n  flex-grow: 1;\n  border: 0;\n  outline: none;\n  background: transparent;\n  ${fonts.sizes(18)};\n`;\nconst StyledInputWrapper = styled.div`\n  display: flex;\n  flex-wrap: wrap;\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.greyLighter};\n  transition: border-color ${animations.durations.normal} ease;\n  border-radius: ${misc.borderRadius};\n  &:focus-within {\n    border-color: ${colors.brand.primary};\n  }\n`;\n\nconst CombinedInputAndDropdownWrapper = styled.div`\n  display: flex;\n  flex-grow: 1;\n`;\n\ninterface SuggestionInputProps {\n  suggestions: TagType[];\n  value: string;\n  onChange: (e: ChangeEvent<HTMLInputElement>) => void;\n  setExpanded: (expanded: boolean) => void;\n  expanded: boolean;\n  onToggleTag: (id: string) => void;\n  setInputValue: (value: string) => void;\n  onCreateTag: (tagName: string) => void;\n  addedTags: TagType[];\n  dropdownMaxHeight: string;\n  prefix?: string | ReactNode;\n  inline?: boolean;\n  scrollAnchorElement: RefObject<HTMLDivElement>;\n}\n\nconst SuggestionInput = ({\n  suggestions,\n  value,\n  setInputValue,\n  onCreateTag,\n  onChange,\n  onToggleTag,\n  addedTags,\n  setExpanded,\n  expanded,\n  dropdownMaxHeight,\n  prefix,\n  inline,\n  scrollAnchorElement,\n}: SuggestionInputProps) => {\n  const { t } = useTranslation();\n  const [currentHighlightedIndex, setCurrentHighlightedIndex] = useState(0);\n  const [hasFocus, setHasFocus] = useState(false);\n  const initalRender = useRef(true);\n  const inputRef = useRef<HTMLInputElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n  const suggestionIdRef = useRef<string>(uuid());\n\n  useEffect(() => {\n    setCurrentHighlightedIndex(0);\n  }, [suggestions]);\n\n  useEffect(() => {\n    if (!initalRender.current) {\n      inputRef.current?.focus();\n    } else {\n      initalRender.current = false;\n    }\n  }, [addedTags]);\n\n  useEffect(() => {\n    const selectedSuggestionElement = document\n      .getElementById(suggestionIdRef.current)\n      ?.querySelector('[aria-selected=\"true\"]');\n    if (selectedSuggestionElement) {\n      // Do we need to scroll this into view?\n      selectedSuggestionElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [currentHighlightedIndex]);\n\n  const hasBeenAdded = (id: string) => addedTags.some(({ id: idAdded }) => idAdded === id);\n\n  return (\n    <SuggestionInputContainer ref={containerRef}>\n      <StyledInputWrapper>\n        {addedTags.map(({ id, name }) => (\n          <Button\n            aria-label={t('tagSelector.removeTag', { name })}\n            onClick={() => onToggleTag(id)}\n            light\n            borderShape=\"rounded\"\n            key={id}\n            size=\"small\">\n            {prefix}\n            {name}\n            <Cross />\n          </Button>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          <StyledInput\n            placeholder={t('tagSelector.placeholder')}\n            value={value}\n            autoComplete=\"off\"\n            onBlur={(e) => {\n              const relatedTarget = e.relatedTarget as HTMLElement;\n              if (!relatedTarget?.dataset?.suggestionbutton) {\n                setExpanded(false);\n                setHasFocus(false);\n              }\n            }}\n            onChange={onChange}\n            onFocus={() => {\n              if (isMobile && scrollAnchorElement?.current) {\n                scrollAnchorElement.current.scrollIntoView({\n                  behavior: 'smooth',\n                });\n              }\n              setHasFocus(true);\n            }}\n            ref={inputRef}\n            onKeyDown={(e: KeyboardEvent<HTMLInputElement>) => {\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n              } else if (e.key === 'Enter' || e.key === 'Tab') {\n                if (value !== '' || expanded) {\n                  if (suggestions.length > 0) {\n                    if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                      onToggleTag(suggestions[currentHighlightedIndex].id);\n                    }\n                    setInputValue('');\n                    if (e.key === 'Enter') {\n                      e.preventDefault();\n                    }\n                  } else {\n                    onCreateTag(value);\n                    setInputValue('');\n                    e.preventDefault();\n                  }\n                } else if (e.key === 'Enter') {\n                  e.preventDefault();\n                }\n              } else if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n              } else if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n              }\n            }}\n          />\n          <Tooltip tooltip={expanded ? t('tagSelector.hideAllTags') : t('tagSelector.showAllTags')}>\n            <IconButtonDualStates\n              data-suggestionbutton\n              ariaLabelActive={t('tagSelector.showAllTags')}\n              ariaLabelInActive={t('tagSelector.hideAllTags')}\n              active={expanded}\n              greyLighter\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              aria-controls={suggestionIdRef.current}\n              onClick={() => {\n                setInputValue('');\n                setExpanded(!expanded);\n                inputRef.current?.focus();\n              }}\n            />\n          </Tooltip>\n        </CombinedInputAndDropdownWrapper>\n      </StyledInputWrapper>\n      <div id={suggestionIdRef.current} aria-live=\"polite\">\n        {(hasFocus || expanded) && suggestions.length > 0 ? (\n          <Suggestions\n            inline={inline}\n            dropdownMaxHeight={dropdownMaxHeight}\n            suggestions={suggestions}\n            currentHighlightedIndex={currentHighlightedIndex}\n            onToggleTag={onToggleTag}\n            hasBeenAdded={hasBeenAdded}\n          />\n        ) : null}\n      </div>\n    </SuggestionInputContainer>\n  );\n};\n\nexport default SuggestionInput;\n"]} */"));
50
+ })("margin-left:", 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"]} */"));
40
51
 
41
52
  var SuggestionInputContainer = _styled("div", {
42
- target: "e55qeml1",
53
+ target: "e55qeml2",
43
54
  label: "SuggestionInputContainer"
44
- })("margin-bottom:", spacing.large, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AAyB2C","file":"SuggestionInput.tsx","sourcesContent":["/*\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useState, useRef, useEffect, ReactNode, RefObject, ChangeEvent, KeyboardEvent } from 'react';\nimport { isMobile } from 'react-device-detect';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport Button, { IconButtonDualStates } from '@ndla/button';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport { Cross as CrossRaw } from '@ndla/icons/action';\nimport { spacing, colors, misc, animations, fonts } from '@ndla/core';\nimport Tooltip from '@ndla/tooltip';\nimport { uuid } from '@ndla/util';\nimport Suggestions from './Suggestions';\nimport type { TagType } from './TagSelector';\n\nconst Cross = styled(CrossRaw)`\n  margin-left: ${spacing.xxsmall};\n`;\n\nconst SuggestionInputContainer = styled.div`\n  margin-bottom: ${spacing.large};\n`;\n\nconst StyledInput = styled.input`\n  flex-grow: 1;\n  border: 0;\n  outline: none;\n  background: transparent;\n  ${fonts.sizes(18)};\n`;\nconst StyledInputWrapper = styled.div`\n  display: flex;\n  flex-wrap: wrap;\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.greyLighter};\n  transition: border-color ${animations.durations.normal} ease;\n  border-radius: ${misc.borderRadius};\n  &:focus-within {\n    border-color: ${colors.brand.primary};\n  }\n`;\n\nconst CombinedInputAndDropdownWrapper = styled.div`\n  display: flex;\n  flex-grow: 1;\n`;\n\ninterface SuggestionInputProps {\n  suggestions: TagType[];\n  value: string;\n  onChange: (e: ChangeEvent<HTMLInputElement>) => void;\n  setExpanded: (expanded: boolean) => void;\n  expanded: boolean;\n  onToggleTag: (id: string) => void;\n  setInputValue: (value: string) => void;\n  onCreateTag: (tagName: string) => void;\n  addedTags: TagType[];\n  dropdownMaxHeight: string;\n  prefix?: string | ReactNode;\n  inline?: boolean;\n  scrollAnchorElement: RefObject<HTMLDivElement>;\n}\n\nconst SuggestionInput = ({\n  suggestions,\n  value,\n  setInputValue,\n  onCreateTag,\n  onChange,\n  onToggleTag,\n  addedTags,\n  setExpanded,\n  expanded,\n  dropdownMaxHeight,\n  prefix,\n  inline,\n  scrollAnchorElement,\n}: SuggestionInputProps) => {\n  const { t } = useTranslation();\n  const [currentHighlightedIndex, setCurrentHighlightedIndex] = useState(0);\n  const [hasFocus, setHasFocus] = useState(false);\n  const initalRender = useRef(true);\n  const inputRef = useRef<HTMLInputElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n  const suggestionIdRef = useRef<string>(uuid());\n\n  useEffect(() => {\n    setCurrentHighlightedIndex(0);\n  }, [suggestions]);\n\n  useEffect(() => {\n    if (!initalRender.current) {\n      inputRef.current?.focus();\n    } else {\n      initalRender.current = false;\n    }\n  }, [addedTags]);\n\n  useEffect(() => {\n    const selectedSuggestionElement = document\n      .getElementById(suggestionIdRef.current)\n      ?.querySelector('[aria-selected=\"true\"]');\n    if (selectedSuggestionElement) {\n      // Do we need to scroll this into view?\n      selectedSuggestionElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [currentHighlightedIndex]);\n\n  const hasBeenAdded = (id: string) => addedTags.some(({ id: idAdded }) => idAdded === id);\n\n  return (\n    <SuggestionInputContainer ref={containerRef}>\n      <StyledInputWrapper>\n        {addedTags.map(({ id, name }) => (\n          <Button\n            aria-label={t('tagSelector.removeTag', { name })}\n            onClick={() => onToggleTag(id)}\n            light\n            borderShape=\"rounded\"\n            key={id}\n            size=\"small\">\n            {prefix}\n            {name}\n            <Cross />\n          </Button>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          <StyledInput\n            placeholder={t('tagSelector.placeholder')}\n            value={value}\n            autoComplete=\"off\"\n            onBlur={(e) => {\n              const relatedTarget = e.relatedTarget as HTMLElement;\n              if (!relatedTarget?.dataset?.suggestionbutton) {\n                setExpanded(false);\n                setHasFocus(false);\n              }\n            }}\n            onChange={onChange}\n            onFocus={() => {\n              if (isMobile && scrollAnchorElement?.current) {\n                scrollAnchorElement.current.scrollIntoView({\n                  behavior: 'smooth',\n                });\n              }\n              setHasFocus(true);\n            }}\n            ref={inputRef}\n            onKeyDown={(e: KeyboardEvent<HTMLInputElement>) => {\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n              } else if (e.key === 'Enter' || e.key === 'Tab') {\n                if (value !== '' || expanded) {\n                  if (suggestions.length > 0) {\n                    if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                      onToggleTag(suggestions[currentHighlightedIndex].id);\n                    }\n                    setInputValue('');\n                    if (e.key === 'Enter') {\n                      e.preventDefault();\n                    }\n                  } else {\n                    onCreateTag(value);\n                    setInputValue('');\n                    e.preventDefault();\n                  }\n                } else if (e.key === 'Enter') {\n                  e.preventDefault();\n                }\n              } else if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n              } else if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n              }\n            }}\n          />\n          <Tooltip tooltip={expanded ? t('tagSelector.hideAllTags') : t('tagSelector.showAllTags')}>\n            <IconButtonDualStates\n              data-suggestionbutton\n              ariaLabelActive={t('tagSelector.showAllTags')}\n              ariaLabelInActive={t('tagSelector.hideAllTags')}\n              active={expanded}\n              greyLighter\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              aria-controls={suggestionIdRef.current}\n              onClick={() => {\n                setInputValue('');\n                setExpanded(!expanded);\n                inputRef.current?.focus();\n              }}\n            />\n          </Tooltip>\n        </CombinedInputAndDropdownWrapper>\n      </StyledInputWrapper>\n      <div id={suggestionIdRef.current} aria-live=\"polite\">\n        {(hasFocus || expanded) && suggestions.length > 0 ? (\n          <Suggestions\n            inline={inline}\n            dropdownMaxHeight={dropdownMaxHeight}\n            suggestions={suggestions}\n            currentHighlightedIndex={currentHighlightedIndex}\n            onToggleTag={onToggleTag}\n            hasBeenAdded={hasBeenAdded}\n          />\n        ) : null}\n      </div>\n    </SuggestionInputContainer>\n  );\n};\n\nexport default SuggestionInput;\n"]} */"));
55
+ })("margin-bottom:", 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"]} */"));
45
56
 
46
57
  var StyledInput = _styled("input", {
47
- target: "e55qeml2",
58
+ target: "e55qeml3",
48
59
  label: "StyledInput"
49
- })("flex-grow:1;border:0;outline:none;background:transparent;", fonts.sizes(18), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AA6BgC","file":"SuggestionInput.tsx","sourcesContent":["/*\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useState, useRef, useEffect, ReactNode, RefObject, ChangeEvent, KeyboardEvent } from 'react';\nimport { isMobile } from 'react-device-detect';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport Button, { IconButtonDualStates } from '@ndla/button';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport { Cross as CrossRaw } from '@ndla/icons/action';\nimport { spacing, colors, misc, animations, fonts } from '@ndla/core';\nimport Tooltip from '@ndla/tooltip';\nimport { uuid } from '@ndla/util';\nimport Suggestions from './Suggestions';\nimport type { TagType } from './TagSelector';\n\nconst Cross = styled(CrossRaw)`\n  margin-left: ${spacing.xxsmall};\n`;\n\nconst SuggestionInputContainer = styled.div`\n  margin-bottom: ${spacing.large};\n`;\n\nconst StyledInput = styled.input`\n  flex-grow: 1;\n  border: 0;\n  outline: none;\n  background: transparent;\n  ${fonts.sizes(18)};\n`;\nconst StyledInputWrapper = styled.div`\n  display: flex;\n  flex-wrap: wrap;\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.greyLighter};\n  transition: border-color ${animations.durations.normal} ease;\n  border-radius: ${misc.borderRadius};\n  &:focus-within {\n    border-color: ${colors.brand.primary};\n  }\n`;\n\nconst CombinedInputAndDropdownWrapper = styled.div`\n  display: flex;\n  flex-grow: 1;\n`;\n\ninterface SuggestionInputProps {\n  suggestions: TagType[];\n  value: string;\n  onChange: (e: ChangeEvent<HTMLInputElement>) => void;\n  setExpanded: (expanded: boolean) => void;\n  expanded: boolean;\n  onToggleTag: (id: string) => void;\n  setInputValue: (value: string) => void;\n  onCreateTag: (tagName: string) => void;\n  addedTags: TagType[];\n  dropdownMaxHeight: string;\n  prefix?: string | ReactNode;\n  inline?: boolean;\n  scrollAnchorElement: RefObject<HTMLDivElement>;\n}\n\nconst SuggestionInput = ({\n  suggestions,\n  value,\n  setInputValue,\n  onCreateTag,\n  onChange,\n  onToggleTag,\n  addedTags,\n  setExpanded,\n  expanded,\n  dropdownMaxHeight,\n  prefix,\n  inline,\n  scrollAnchorElement,\n}: SuggestionInputProps) => {\n  const { t } = useTranslation();\n  const [currentHighlightedIndex, setCurrentHighlightedIndex] = useState(0);\n  const [hasFocus, setHasFocus] = useState(false);\n  const initalRender = useRef(true);\n  const inputRef = useRef<HTMLInputElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n  const suggestionIdRef = useRef<string>(uuid());\n\n  useEffect(() => {\n    setCurrentHighlightedIndex(0);\n  }, [suggestions]);\n\n  useEffect(() => {\n    if (!initalRender.current) {\n      inputRef.current?.focus();\n    } else {\n      initalRender.current = false;\n    }\n  }, [addedTags]);\n\n  useEffect(() => {\n    const selectedSuggestionElement = document\n      .getElementById(suggestionIdRef.current)\n      ?.querySelector('[aria-selected=\"true\"]');\n    if (selectedSuggestionElement) {\n      // Do we need to scroll this into view?\n      selectedSuggestionElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [currentHighlightedIndex]);\n\n  const hasBeenAdded = (id: string) => addedTags.some(({ id: idAdded }) => idAdded === id);\n\n  return (\n    <SuggestionInputContainer ref={containerRef}>\n      <StyledInputWrapper>\n        {addedTags.map(({ id, name }) => (\n          <Button\n            aria-label={t('tagSelector.removeTag', { name })}\n            onClick={() => onToggleTag(id)}\n            light\n            borderShape=\"rounded\"\n            key={id}\n            size=\"small\">\n            {prefix}\n            {name}\n            <Cross />\n          </Button>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          <StyledInput\n            placeholder={t('tagSelector.placeholder')}\n            value={value}\n            autoComplete=\"off\"\n            onBlur={(e) => {\n              const relatedTarget = e.relatedTarget as HTMLElement;\n              if (!relatedTarget?.dataset?.suggestionbutton) {\n                setExpanded(false);\n                setHasFocus(false);\n              }\n            }}\n            onChange={onChange}\n            onFocus={() => {\n              if (isMobile && scrollAnchorElement?.current) {\n                scrollAnchorElement.current.scrollIntoView({\n                  behavior: 'smooth',\n                });\n              }\n              setHasFocus(true);\n            }}\n            ref={inputRef}\n            onKeyDown={(e: KeyboardEvent<HTMLInputElement>) => {\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n              } else if (e.key === 'Enter' || e.key === 'Tab') {\n                if (value !== '' || expanded) {\n                  if (suggestions.length > 0) {\n                    if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                      onToggleTag(suggestions[currentHighlightedIndex].id);\n                    }\n                    setInputValue('');\n                    if (e.key === 'Enter') {\n                      e.preventDefault();\n                    }\n                  } else {\n                    onCreateTag(value);\n                    setInputValue('');\n                    e.preventDefault();\n                  }\n                } else if (e.key === 'Enter') {\n                  e.preventDefault();\n                }\n              } else if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n              } else if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n              }\n            }}\n          />\n          <Tooltip tooltip={expanded ? t('tagSelector.hideAllTags') : t('tagSelector.showAllTags')}>\n            <IconButtonDualStates\n              data-suggestionbutton\n              ariaLabelActive={t('tagSelector.showAllTags')}\n              ariaLabelInActive={t('tagSelector.hideAllTags')}\n              active={expanded}\n              greyLighter\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              aria-controls={suggestionIdRef.current}\n              onClick={() => {\n                setInputValue('');\n                setExpanded(!expanded);\n                inputRef.current?.focus();\n              }}\n            />\n          </Tooltip>\n        </CombinedInputAndDropdownWrapper>\n      </StyledInputWrapper>\n      <div id={suggestionIdRef.current} aria-live=\"polite\">\n        {(hasFocus || expanded) && suggestions.length > 0 ? (\n          <Suggestions\n            inline={inline}\n            dropdownMaxHeight={dropdownMaxHeight}\n            suggestions={suggestions}\n            currentHighlightedIndex={currentHighlightedIndex}\n            onToggleTag={onToggleTag}\n            hasBeenAdded={hasBeenAdded}\n          />\n        ) : null}\n      </div>\n    </SuggestionInputContainer>\n  );\n};\n\nexport default SuggestionInput;\n"]} */"));
60
+ })("flex-grow:1;border:0;outline:none;", 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"]} */"));
50
61
 
51
62
  var StyledInputWrapper = _styled("div", {
52
- target: "e55qeml3",
63
+ target: "e55qeml4",
53
64
  label: "StyledInputWrapper"
54
- })("display:flex;flex-wrap:wrap;gap:", spacing.xsmall, ";padding:", spacing.small, ";border:1px solid ", colors.brand.greyLighter, ";transition:border-color ", animations.durations.normal, " ease;border-radius:", misc.borderRadius, ";&:focus-within{border-color:", colors.brand.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AAoCqC","file":"SuggestionInput.tsx","sourcesContent":["/*\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useState, useRef, useEffect, ReactNode, RefObject, ChangeEvent, KeyboardEvent } from 'react';\nimport { isMobile } from 'react-device-detect';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport Button, { IconButtonDualStates } from '@ndla/button';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport { Cross as CrossRaw } from '@ndla/icons/action';\nimport { spacing, colors, misc, animations, fonts } from '@ndla/core';\nimport Tooltip from '@ndla/tooltip';\nimport { uuid } from '@ndla/util';\nimport Suggestions from './Suggestions';\nimport type { TagType } from './TagSelector';\n\nconst Cross = styled(CrossRaw)`\n  margin-left: ${spacing.xxsmall};\n`;\n\nconst SuggestionInputContainer = styled.div`\n  margin-bottom: ${spacing.large};\n`;\n\nconst StyledInput = styled.input`\n  flex-grow: 1;\n  border: 0;\n  outline: none;\n  background: transparent;\n  ${fonts.sizes(18)};\n`;\nconst StyledInputWrapper = styled.div`\n  display: flex;\n  flex-wrap: wrap;\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.greyLighter};\n  transition: border-color ${animations.durations.normal} ease;\n  border-radius: ${misc.borderRadius};\n  &:focus-within {\n    border-color: ${colors.brand.primary};\n  }\n`;\n\nconst CombinedInputAndDropdownWrapper = styled.div`\n  display: flex;\n  flex-grow: 1;\n`;\n\ninterface SuggestionInputProps {\n  suggestions: TagType[];\n  value: string;\n  onChange: (e: ChangeEvent<HTMLInputElement>) => void;\n  setExpanded: (expanded: boolean) => void;\n  expanded: boolean;\n  onToggleTag: (id: string) => void;\n  setInputValue: (value: string) => void;\n  onCreateTag: (tagName: string) => void;\n  addedTags: TagType[];\n  dropdownMaxHeight: string;\n  prefix?: string | ReactNode;\n  inline?: boolean;\n  scrollAnchorElement: RefObject<HTMLDivElement>;\n}\n\nconst SuggestionInput = ({\n  suggestions,\n  value,\n  setInputValue,\n  onCreateTag,\n  onChange,\n  onToggleTag,\n  addedTags,\n  setExpanded,\n  expanded,\n  dropdownMaxHeight,\n  prefix,\n  inline,\n  scrollAnchorElement,\n}: SuggestionInputProps) => {\n  const { t } = useTranslation();\n  const [currentHighlightedIndex, setCurrentHighlightedIndex] = useState(0);\n  const [hasFocus, setHasFocus] = useState(false);\n  const initalRender = useRef(true);\n  const inputRef = useRef<HTMLInputElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n  const suggestionIdRef = useRef<string>(uuid());\n\n  useEffect(() => {\n    setCurrentHighlightedIndex(0);\n  }, [suggestions]);\n\n  useEffect(() => {\n    if (!initalRender.current) {\n      inputRef.current?.focus();\n    } else {\n      initalRender.current = false;\n    }\n  }, [addedTags]);\n\n  useEffect(() => {\n    const selectedSuggestionElement = document\n      .getElementById(suggestionIdRef.current)\n      ?.querySelector('[aria-selected=\"true\"]');\n    if (selectedSuggestionElement) {\n      // Do we need to scroll this into view?\n      selectedSuggestionElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [currentHighlightedIndex]);\n\n  const hasBeenAdded = (id: string) => addedTags.some(({ id: idAdded }) => idAdded === id);\n\n  return (\n    <SuggestionInputContainer ref={containerRef}>\n      <StyledInputWrapper>\n        {addedTags.map(({ id, name }) => (\n          <Button\n            aria-label={t('tagSelector.removeTag', { name })}\n            onClick={() => onToggleTag(id)}\n            light\n            borderShape=\"rounded\"\n            key={id}\n            size=\"small\">\n            {prefix}\n            {name}\n            <Cross />\n          </Button>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          <StyledInput\n            placeholder={t('tagSelector.placeholder')}\n            value={value}\n            autoComplete=\"off\"\n            onBlur={(e) => {\n              const relatedTarget = e.relatedTarget as HTMLElement;\n              if (!relatedTarget?.dataset?.suggestionbutton) {\n                setExpanded(false);\n                setHasFocus(false);\n              }\n            }}\n            onChange={onChange}\n            onFocus={() => {\n              if (isMobile && scrollAnchorElement?.current) {\n                scrollAnchorElement.current.scrollIntoView({\n                  behavior: 'smooth',\n                });\n              }\n              setHasFocus(true);\n            }}\n            ref={inputRef}\n            onKeyDown={(e: KeyboardEvent<HTMLInputElement>) => {\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n              } else if (e.key === 'Enter' || e.key === 'Tab') {\n                if (value !== '' || expanded) {\n                  if (suggestions.length > 0) {\n                    if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                      onToggleTag(suggestions[currentHighlightedIndex].id);\n                    }\n                    setInputValue('');\n                    if (e.key === 'Enter') {\n                      e.preventDefault();\n                    }\n                  } else {\n                    onCreateTag(value);\n                    setInputValue('');\n                    e.preventDefault();\n                  }\n                } else if (e.key === 'Enter') {\n                  e.preventDefault();\n                }\n              } else if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n              } else if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n              }\n            }}\n          />\n          <Tooltip tooltip={expanded ? t('tagSelector.hideAllTags') : t('tagSelector.showAllTags')}>\n            <IconButtonDualStates\n              data-suggestionbutton\n              ariaLabelActive={t('tagSelector.showAllTags')}\n              ariaLabelInActive={t('tagSelector.hideAllTags')}\n              active={expanded}\n              greyLighter\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              aria-controls={suggestionIdRef.current}\n              onClick={() => {\n                setInputValue('');\n                setExpanded(!expanded);\n                inputRef.current?.focus();\n              }}\n            />\n          </Tooltip>\n        </CombinedInputAndDropdownWrapper>\n      </StyledInputWrapper>\n      <div id={suggestionIdRef.current} aria-live=\"polite\">\n        {(hasFocus || expanded) && suggestions.length > 0 ? (\n          <Suggestions\n            inline={inline}\n            dropdownMaxHeight={dropdownMaxHeight}\n            suggestions={suggestions}\n            currentHighlightedIndex={currentHighlightedIndex}\n            onToggleTag={onToggleTag}\n            hasBeenAdded={hasBeenAdded}\n          />\n        ) : null}\n      </div>\n    </SuggestionInputContainer>\n  );\n};\n\nexport default SuggestionInput;\n"]} */"));
65
+ })("display:flex;flex-wrap:wrap;gap:", spacing.xsmall, ";padding:", spacing.small, ";border:1px solid ", colors.brand.neutral7, ";transition:border-color ", animations.durations.normal, " ease;border-radius:", misc.borderRadius, ";&:focus-within{border-color:", colors.brand.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"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"]} */"));
55
66
 
56
67
  var CombinedInputAndDropdownWrapper = _styled("div", {
57
- target: "e55qeml4",
68
+ target: "e55qeml5",
58
69
  label: "CombinedInputAndDropdownWrapper"
59
70
  })(process.env.NODE_ENV === "production" ? {
60
- name: "1ilyui9",
61
- styles: "display:flex;flex-grow:1;"
71
+ name: "19uf65v",
72
+ styles: "display:flex;flex-grow:1;position:relative;"
62
73
  } : {
63
- name: "1ilyui9",
64
- styles: "display:flex;flex-grow:1;",
65
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SuggestionInput.tsx"],"names":[],"mappings":"AAiDkD","file":"SuggestionInput.tsx","sourcesContent":["/*\n * Copyright (c) 2022-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useState, useRef, useEffect, ReactNode, RefObject, ChangeEvent, KeyboardEvent } from 'react';\nimport { isMobile } from 'react-device-detect';\nimport { useTranslation } from 'react-i18next';\nimport styled from '@emotion/styled';\nimport Button, { IconButtonDualStates } from '@ndla/button';\nimport { ChevronDown, ChevronUp } from '@ndla/icons/common';\nimport { Cross as CrossRaw } from '@ndla/icons/action';\nimport { spacing, colors, misc, animations, fonts } from '@ndla/core';\nimport Tooltip from '@ndla/tooltip';\nimport { uuid } from '@ndla/util';\nimport Suggestions from './Suggestions';\nimport type { TagType } from './TagSelector';\n\nconst Cross = styled(CrossRaw)`\n  margin-left: ${spacing.xxsmall};\n`;\n\nconst SuggestionInputContainer = styled.div`\n  margin-bottom: ${spacing.large};\n`;\n\nconst StyledInput = styled.input`\n  flex-grow: 1;\n  border: 0;\n  outline: none;\n  background: transparent;\n  ${fonts.sizes(18)};\n`;\nconst StyledInputWrapper = styled.div`\n  display: flex;\n  flex-wrap: wrap;\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.greyLighter};\n  transition: border-color ${animations.durations.normal} ease;\n  border-radius: ${misc.borderRadius};\n  &:focus-within {\n    border-color: ${colors.brand.primary};\n  }\n`;\n\nconst CombinedInputAndDropdownWrapper = styled.div`\n  display: flex;\n  flex-grow: 1;\n`;\n\ninterface SuggestionInputProps {\n  suggestions: TagType[];\n  value: string;\n  onChange: (e: ChangeEvent<HTMLInputElement>) => void;\n  setExpanded: (expanded: boolean) => void;\n  expanded: boolean;\n  onToggleTag: (id: string) => void;\n  setInputValue: (value: string) => void;\n  onCreateTag: (tagName: string) => void;\n  addedTags: TagType[];\n  dropdownMaxHeight: string;\n  prefix?: string | ReactNode;\n  inline?: boolean;\n  scrollAnchorElement: RefObject<HTMLDivElement>;\n}\n\nconst SuggestionInput = ({\n  suggestions,\n  value,\n  setInputValue,\n  onCreateTag,\n  onChange,\n  onToggleTag,\n  addedTags,\n  setExpanded,\n  expanded,\n  dropdownMaxHeight,\n  prefix,\n  inline,\n  scrollAnchorElement,\n}: SuggestionInputProps) => {\n  const { t } = useTranslation();\n  const [currentHighlightedIndex, setCurrentHighlightedIndex] = useState(0);\n  const [hasFocus, setHasFocus] = useState(false);\n  const initalRender = useRef(true);\n  const inputRef = useRef<HTMLInputElement>(null);\n  const containerRef = useRef<HTMLDivElement>(null);\n  const suggestionIdRef = useRef<string>(uuid());\n\n  useEffect(() => {\n    setCurrentHighlightedIndex(0);\n  }, [suggestions]);\n\n  useEffect(() => {\n    if (!initalRender.current) {\n      inputRef.current?.focus();\n    } else {\n      initalRender.current = false;\n    }\n  }, [addedTags]);\n\n  useEffect(() => {\n    const selectedSuggestionElement = document\n      .getElementById(suggestionIdRef.current)\n      ?.querySelector('[aria-selected=\"true\"]');\n    if (selectedSuggestionElement) {\n      // Do we need to scroll this into view?\n      selectedSuggestionElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [currentHighlightedIndex]);\n\n  const hasBeenAdded = (id: string) => addedTags.some(({ id: idAdded }) => idAdded === id);\n\n  return (\n    <SuggestionInputContainer ref={containerRef}>\n      <StyledInputWrapper>\n        {addedTags.map(({ id, name }) => (\n          <Button\n            aria-label={t('tagSelector.removeTag', { name })}\n            onClick={() => onToggleTag(id)}\n            light\n            borderShape=\"rounded\"\n            key={id}\n            size=\"small\">\n            {prefix}\n            {name}\n            <Cross />\n          </Button>\n        ))}\n        <CombinedInputAndDropdownWrapper>\n          <StyledInput\n            placeholder={t('tagSelector.placeholder')}\n            value={value}\n            autoComplete=\"off\"\n            onBlur={(e) => {\n              const relatedTarget = e.relatedTarget as HTMLElement;\n              if (!relatedTarget?.dataset?.suggestionbutton) {\n                setExpanded(false);\n                setHasFocus(false);\n              }\n            }}\n            onChange={onChange}\n            onFocus={() => {\n              if (isMobile && scrollAnchorElement?.current) {\n                scrollAnchorElement.current.scrollIntoView({\n                  behavior: 'smooth',\n                });\n              }\n              setHasFocus(true);\n            }}\n            ref={inputRef}\n            onKeyDown={(e: KeyboardEvent<HTMLInputElement>) => {\n              if (e.key === 'Escape') {\n                setExpanded(false);\n                e.preventDefault();\n              } else if (e.key === 'Enter' || e.key === 'Tab') {\n                if (value !== '' || expanded) {\n                  if (suggestions.length > 0) {\n                    if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {\n                      onToggleTag(suggestions[currentHighlightedIndex].id);\n                    }\n                    setInputValue('');\n                    if (e.key === 'Enter') {\n                      e.preventDefault();\n                    }\n                  } else {\n                    onCreateTag(value);\n                    setInputValue('');\n                    e.preventDefault();\n                  }\n                } else if (e.key === 'Enter') {\n                  e.preventDefault();\n                }\n              } else if (e.key === 'ArrowUp') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1,\n                );\n                e.preventDefault();\n              } else if (e.key === 'ArrowDown') {\n                setCurrentHighlightedIndex(\n                  currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1,\n                );\n                e.preventDefault();\n              }\n            }}\n          />\n          <Tooltip tooltip={expanded ? t('tagSelector.hideAllTags') : t('tagSelector.showAllTags')}>\n            <IconButtonDualStates\n              data-suggestionbutton\n              ariaLabelActive={t('tagSelector.showAllTags')}\n              ariaLabelInActive={t('tagSelector.hideAllTags')}\n              active={expanded}\n              greyLighter\n              inactiveIcon={<ChevronDown />}\n              activeIcon={<ChevronUp />}\n              size=\"small\"\n              aria-controls={suggestionIdRef.current}\n              onClick={() => {\n                setInputValue('');\n                setExpanded(!expanded);\n                inputRef.current?.focus();\n              }}\n            />\n          </Tooltip>\n        </CombinedInputAndDropdownWrapper>\n      </StyledInputWrapper>\n      <div id={suggestionIdRef.current} aria-live=\"polite\">\n        {(hasFocus || expanded) && suggestions.length > 0 ? (\n          <Suggestions\n            inline={inline}\n            dropdownMaxHeight={dropdownMaxHeight}\n            suggestions={suggestions}\n            currentHighlightedIndex={currentHighlightedIndex}\n            onToggleTag={onToggleTag}\n            hasBeenAdded={hasBeenAdded}\n          />\n        ) : null}\n      </div>\n    </SuggestionInputContainer>\n  );\n};\n\nexport default SuggestionInput;\n"]} */",
74
+ name: "19uf65v",
75
+ styles: "display:flex;flex-grow:1;position:relative;",
76
+ 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"]} */",
66
77
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
67
78
  });
68
79
 
69
- var SuggestionInput = function SuggestionInput(_ref) {
70
- var suggestions = _ref.suggestions,
71
- value = _ref.value,
72
- setInputValue = _ref.setInputValue,
73
- onCreateTag = _ref.onCreateTag,
74
- onChange = _ref.onChange,
75
- onToggleTag = _ref.onToggleTag,
76
- addedTags = _ref.addedTags,
77
- setExpanded = _ref.setExpanded,
78
- expanded = _ref.expanded,
79
- dropdownMaxHeight = _ref.dropdownMaxHeight,
80
- prefix = _ref.prefix,
81
- inline = _ref.inline,
82
- scrollAnchorElement = _ref.scrollAnchorElement;
80
+ var StyledTagButton = /*#__PURE__*/_styled(Button, {
81
+ target: "e55qeml6",
82
+ label: "StyledTagButton"
83
+ })(function (_ref2) {
84
+ var enableTagButtonAnimation = _ref2.enableTagButtonAnimation;
85
+ return enableTagButtonAnimation ? animations.fadeInScaled(animations.durations.slow) : '';
86
+ }, 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"]} */");
87
+
88
+ var SuggestionInput = function SuggestionInput(_ref3) {
89
+ var suggestions = _ref3.suggestions,
90
+ value = _ref3.value,
91
+ setInputValue = _ref3.setInputValue,
92
+ onCreateTag = _ref3.onCreateTag,
93
+ onChange = _ref3.onChange,
94
+ onToggleTag = _ref3.onToggleTag,
95
+ addedTags = _ref3.addedTags,
96
+ setExpanded = _ref3.setExpanded,
97
+ expanded = _ref3.expanded,
98
+ dropdownMaxHeight = _ref3.dropdownMaxHeight,
99
+ prefix = _ref3.prefix,
100
+ inline = _ref3.inline,
101
+ scrollAnchorElement = _ref3.scrollAnchorElement;
83
102
 
84
103
  var _useTranslation = useTranslation(),
85
104
  t = _useTranslation.t;
@@ -98,6 +117,10 @@ var SuggestionInput = function SuggestionInput(_ref) {
98
117
  var inputRef = useRef(null);
99
118
  var containerRef = useRef(null);
100
119
  var suggestionIdRef = useRef(uuid());
120
+ var initalTags = useRef(addedTags.map(function (_ref4) {
121
+ var id = _ref4.id;
122
+ return id;
123
+ }));
101
124
  useEffect(function () {
102
125
  setCurrentHighlightedIndex(0);
103
126
  }, [suggestions]);
@@ -125,18 +148,19 @@ var SuggestionInput = function SuggestionInput(_ref) {
125
148
  }, [currentHighlightedIndex]);
126
149
 
127
150
  var hasBeenAdded = function hasBeenAdded(id) {
128
- return addedTags.some(function (_ref2) {
129
- var idAdded = _ref2.id;
151
+ return addedTags.some(function (_ref5) {
152
+ var idAdded = _ref5.id;
130
153
  return idAdded === id;
131
154
  });
132
155
  };
133
156
 
134
157
  return ___EmotionJSX(SuggestionInputContainer, {
135
158
  ref: containerRef
136
- }, ___EmotionJSX(StyledInputWrapper, null, addedTags.map(function (_ref3) {
137
- var id = _ref3.id,
138
- name = _ref3.name;
139
- return ___EmotionJSX(Button, {
159
+ }, ___EmotionJSX(StyledInputWrapper, null, addedTags.map(function (_ref6) {
160
+ var id = _ref6.id,
161
+ name = _ref6.name;
162
+ return ___EmotionJSX(StyledTagButton, {
163
+ enableTagButtonAnimation: !initalTags.current.includes(id),
140
164
  "aria-label": t('tagSelector.removeTag', {
141
165
  name: name
142
166
  }),
@@ -148,7 +172,10 @@ var SuggestionInput = function SuggestionInput(_ref) {
148
172
  key: id,
149
173
  size: "small"
150
174
  }, prefix, name, ___EmotionJSX(Cross, null));
151
- }), ___EmotionJSX(CombinedInputAndDropdownWrapper, null, ___EmotionJSX(StyledInput, {
175
+ }), ___EmotionJSX(CombinedInputAndDropdownWrapper, null, suggestions[currentHighlightedIndex] && ___EmotionJSX(SuggestionText, {
176
+ value: value,
177
+ suggestionValue: suggestions[currentHighlightedIndex].name
178
+ }), ___EmotionJSX(StyledInput, {
152
179
  placeholder: t('tagSelector.placeholder'),
153
180
  value: value,
154
181
  autoComplete: "off",
@@ -174,36 +201,64 @@ var SuggestionInput = function SuggestionInput(_ref) {
174
201
  },
175
202
  ref: inputRef,
176
203
  onKeyDown: function onKeyDown(e) {
204
+ if (!['Enter', ' ', 'Tab', 'ArrowDown', 'ArrowUp', 'Backspace'].includes(e.key)) {
205
+ return;
206
+ }
207
+
208
+ var trimmedValue = value.replace(/\s/g, '');
209
+
177
210
  if (e.key === 'Escape') {
178
211
  setExpanded(false);
179
212
  e.preventDefault();
180
- } else if (e.key === 'Enter' || e.key === 'Tab') {
181
- if (value !== '' || expanded) {
182
- if (suggestions.length > 0) {
183
- if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {
184
- onToggleTag(suggestions[currentHighlightedIndex].id);
185
- }
186
-
187
- setInputValue('');
188
-
189
- if (e.key === 'Enter') {
190
- e.preventDefault();
191
- }
192
- } else {
193
- onCreateTag(value);
194
- setInputValue('');
213
+ return;
214
+ }
215
+
216
+ if (e.key === 'Backspace' && trimmedValue === '' && addedTags.length) {
217
+ // Remove the added last tag
218
+ onToggleTag(addedTags[addedTags.length - 1].id);
219
+ return;
220
+ }
221
+
222
+ if (e.key === 'ArrowUp') {
223
+ setCurrentHighlightedIndex(currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1);
224
+ e.preventDefault();
225
+ return;
226
+ }
227
+
228
+ if (e.key === 'ArrowDown') {
229
+ setCurrentHighlightedIndex(currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1);
230
+ e.preventDefault();
231
+ return;
232
+ }
233
+
234
+ if (trimmedValue === '' && !expanded) {
235
+ if (e.key === 'Enter' || e.key === ' ') {
236
+ e.preventDefault();
237
+ }
238
+
239
+ return;
240
+ }
241
+
242
+ if (e.key === 'Enter' || e.key === 'Tab' || e.key === ' ') {
243
+ if (suggestions.length > 0) {
244
+ if (!hasBeenAdded(suggestions[currentHighlightedIndex].id)) {
245
+ onToggleTag(suggestions[currentHighlightedIndex].id);
246
+ } else if (trimmedValue.length < suggestions[currentHighlightedIndex].name.length) {
247
+ onCreateTag(trimmedValue);
195
248
  e.preventDefault();
196
249
  }
197
- } else if (e.key === 'Enter') {
250
+
251
+ setInputValue('');
198
252
  e.preventDefault();
253
+ return;
199
254
  }
200
- } else if (e.key === 'ArrowUp') {
201
- setCurrentHighlightedIndex(currentHighlightedIndex - 1 < 0 ? suggestions.length - 1 : currentHighlightedIndex - 1);
202
- e.preventDefault();
203
- } else if (e.key === 'ArrowDown') {
204
- setCurrentHighlightedIndex(currentHighlightedIndex + 1 >= suggestions.length ? 0 : currentHighlightedIndex + 1);
255
+
256
+ onCreateTag(trimmedValue);
257
+ setInputValue('');
205
258
  e.preventDefault();
206
259
  }
260
+
261
+ return;
207
262
  }
208
263
  }), ___EmotionJSX(Tooltip, {
209
264
  tooltip: expanded ? t('tagSelector.hideAllTags') : t('tagSelector.showAllTags')