@ndla/ui 45.0.6 → 45.0.8

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 (44) hide show
  1. package/README.md +1 -1
  2. package/es/CampaignBlock/CampaignBlock.js +7 -7
  3. package/es/Gloss/Gloss.js +9 -9
  4. package/es/LicenseByline/EmbedByline.js +8 -8
  5. package/es/MediaList/MediaList.js +32 -14
  6. package/es/Search/SearchResultSleeve.js +15 -24
  7. package/es/locale/messages-en.js +1 -0
  8. package/es/locale/messages-nb.js +1 -0
  9. package/es/locale/messages-nn.js +1 -0
  10. package/es/locale/messages-se.js +1 -0
  11. package/es/locale/messages-sma.js +1 -0
  12. package/lib/CampaignBlock/CampaignBlock.js +7 -7
  13. package/lib/Gloss/Gloss.d.ts +1 -1
  14. package/lib/Gloss/Gloss.js +9 -9
  15. package/lib/LicenseByline/EmbedByline.d.ts +5 -5
  16. package/lib/LicenseByline/EmbedByline.js +8 -8
  17. package/lib/MediaList/MediaList.d.ts +12 -5
  18. package/lib/MediaList/MediaList.js +31 -14
  19. package/lib/MediaList/index.d.ts +1 -0
  20. package/lib/Search/SearchResultSleeve.js +15 -24
  21. package/lib/index.d.ts +1 -0
  22. package/lib/locale/messages-en.d.ts +1 -0
  23. package/lib/locale/messages-en.js +1 -0
  24. package/lib/locale/messages-nb.d.ts +1 -0
  25. package/lib/locale/messages-nb.js +1 -0
  26. package/lib/locale/messages-nn.d.ts +1 -0
  27. package/lib/locale/messages-nn.js +1 -0
  28. package/lib/locale/messages-se.d.ts +1 -0
  29. package/lib/locale/messages-se.js +1 -0
  30. package/lib/locale/messages-sma.d.ts +1 -0
  31. package/lib/locale/messages-sma.js +1 -0
  32. package/package.json +4 -4
  33. package/src/CampaignBlock/CampaignBlock.tsx +1 -1
  34. package/src/Gloss/Gloss.tsx +2 -2
  35. package/src/LicenseByline/EmbedByline.tsx +7 -7
  36. package/src/MediaList/MediaList.tsx +38 -8
  37. package/src/MediaList/index.ts +2 -0
  38. package/src/Search/SearchResultSleeve.tsx +9 -20
  39. package/src/index.ts +2 -0
  40. package/src/locale/messages-en.ts +1 -0
  41. package/src/locale/messages-nb.ts +1 -0
  42. package/src/locale/messages-nn.ts +1 -0
  43. package/src/locale/messages-se.ts +1 -0
  44. package/src/locale/messages-sma.ts +1 -0
@@ -31,21 +31,21 @@ var GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';
31
31
  var StyledNoHits = /*#__PURE__*/_styled("div", {
32
32
  target: "e1mez1xb7",
33
33
  label: "StyledNoHits"
34
- })(fonts.sizes(16, 1.1), ";color:", colors.text.primary, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAqB+B","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        e.stopPropagation();\n        e.preventDefault();\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          const anchorTag =\n            searchSuggestionRef && searchSuggestionRef.current && searchSuggestionRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else if (keyboardPathNavigation === GO_TO_SEARCHPAGE || keyboardPathNavigation === undefined) {\n          const anchorTag = searchAllRef && searchAllRef.current && searchAllRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else {\n          if (keyboardPathNavigation instanceof HTMLElement) {\n            const toClick =\n              keyboardPathNavigation &&\n              keyboardPathNavigation.querySelector &&\n              (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n            toClick && toClick.click();\n          }\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
34
+ })(fonts.sizes(16, 1.1), ";color:", colors.text.primary, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAqB+B","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          searchSuggestionRef?.current?.closest('a')?.click();\n        } else if (keyboardPathNavigation instanceof HTMLElement) {\n          e.stopPropagation();\n          e.preventDefault();\n          const toClick =\n            keyboardPathNavigation &&\n            keyboardPathNavigation.querySelector &&\n            (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n          toClick && toClick.click();\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
35
35
  var StyledAside = /*#__PURE__*/_styled("aside", {
36
36
  target: "e1mez1xb6",
37
37
  label: "StyledAside"
38
38
  })(fonts.sizes('16px', '22px'), ";display:flex;align-items:flex-start;margin:0;color:", colors.text.primary, ";padding:", spacing.normal, " ", spacing.large, " ", spacing.normal, " ", spacing.normal, ";background:", colors.support.yellowLight, ";border-radius:", misc.borderRadius, ";span{display:block;max-width:700px;padding-left:", spacing.small, ";flex:1;}svg{transform:translateY(6px);}", mq.range({
39
39
  until: breakpoints.tablet
40
- }), "{display:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AA0BgC","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        e.stopPropagation();\n        e.preventDefault();\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          const anchorTag =\n            searchSuggestionRef && searchSuggestionRef.current && searchSuggestionRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else if (keyboardPathNavigation === GO_TO_SEARCHPAGE || keyboardPathNavigation === undefined) {\n          const anchorTag = searchAllRef && searchAllRef.current && searchAllRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else {\n          if (keyboardPathNavigation instanceof HTMLElement) {\n            const toClick =\n              keyboardPathNavigation &&\n              keyboardPathNavigation.querySelector &&\n              (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n            toClick && toClick.click();\n          }\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
40
+ }), "{display:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AA0BgC","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          searchSuggestionRef?.current?.closest('a')?.click();\n        } else if (keyboardPathNavigation instanceof HTMLElement) {\n          e.stopPropagation();\n          e.preventDefault();\n          const toClick =\n            keyboardPathNavigation &&\n            keyboardPathNavigation.querySelector &&\n            (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n          toClick && toClick.click();\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
41
41
  var SearchLinkContainer = /*#__PURE__*/_styled("div", {
42
42
  target: "e1mez1xb5",
43
43
  label: "SearchLinkContainer"
44
- })("margin:", spacing.normal, " -", spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAiDsC","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        e.stopPropagation();\n        e.preventDefault();\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          const anchorTag =\n            searchSuggestionRef && searchSuggestionRef.current && searchSuggestionRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else if (keyboardPathNavigation === GO_TO_SEARCHPAGE || keyboardPathNavigation === undefined) {\n          const anchorTag = searchAllRef && searchAllRef.current && searchAllRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else {\n          if (keyboardPathNavigation instanceof HTMLElement) {\n            const toClick =\n              keyboardPathNavigation &&\n              keyboardPathNavigation.querySelector &&\n              (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n            toClick && toClick.click();\n          }\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
44
+ })("margin:", spacing.normal, " -", spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAiDsC","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          searchSuggestionRef?.current?.closest('a')?.click();\n        } else if (keyboardPathNavigation instanceof HTMLElement) {\n          e.stopPropagation();\n          e.preventDefault();\n          const toClick =\n            keyboardPathNavigation &&\n            keyboardPathNavigation.querySelector &&\n            (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n          toClick && toClick.click();\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
45
45
  var StyledSearchLink = /*#__PURE__*/_styled(SafeLink, {
46
46
  target: "e1mez1xb4",
47
47
  label: "StyledSearchLink"
48
- })("width:100%;box-shadow:none;border:0;background:transparent;display:inline-flex;flex-grow:1;align-items:center;padding:", spacing.xsmall, " ", spacing.small, ";line-height:1.7rem;color:", colors.brand.primary, ";strong{font-weight:", fonts.weight.semibold, ";margin-left:", spacing.xsmall, ";}&:focus{", highlightStyle, ";}&:hover{strong{text-decoration:underline;}}small{color:", colors.text.light, ";padding-left:", spacing.xsmall, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAqDyC","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        e.stopPropagation();\n        e.preventDefault();\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          const anchorTag =\n            searchSuggestionRef && searchSuggestionRef.current && searchSuggestionRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else if (keyboardPathNavigation === GO_TO_SEARCHPAGE || keyboardPathNavigation === undefined) {\n          const anchorTag = searchAllRef && searchAllRef.current && searchAllRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else {\n          if (keyboardPathNavigation instanceof HTMLElement) {\n            const toClick =\n              keyboardPathNavigation &&\n              keyboardPathNavigation.querySelector &&\n              (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n            toClick && toClick.click();\n          }\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
48
+ })("width:100%;box-shadow:none;border:0;background:transparent;display:inline-flex;flex-grow:1;align-items:center;padding:", spacing.xsmall, " ", spacing.small, ";line-height:1.7rem;color:", colors.brand.primary, ";strong{font-weight:", fonts.weight.semibold, ";margin-left:", spacing.xsmall, ";}&:focus{", highlightStyle, ";}&:hover{strong{text-decoration:underline;}}small{color:", colors.text.light, ";padding-left:", spacing.xsmall, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAqDyC","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          searchSuggestionRef?.current?.closest('a')?.click();\n        } else if (keyboardPathNavigation instanceof HTMLElement) {\n          e.stopPropagation();\n          e.preventDefault();\n          const toClick =\n            keyboardPathNavigation &&\n            keyboardPathNavigation.querySelector &&\n            (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n          toClick && toClick.click();\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
49
49
  var StyledSearchResultsWrapper = /*#__PURE__*/_styled("section", {
50
50
  target: "e1mez1xb3",
51
51
  label: "StyledSearchResultsWrapper"
@@ -53,7 +53,7 @@ var StyledSearchResultsWrapper = /*#__PURE__*/_styled("section", {
53
53
  return props.frontpage ? 'absolute' : 'static';
54
54
  }, ";left:0;right:0;top:62px;border-radius:", misc.borderRadius, ";", mq.range({
55
55
  until: breakpoints.tablet
56
- }), "{position:fixed;left:0;right:0;bottom:0;top:74px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAsF+D","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        e.stopPropagation();\n        e.preventDefault();\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          const anchorTag =\n            searchSuggestionRef && searchSuggestionRef.current && searchSuggestionRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else if (keyboardPathNavigation === GO_TO_SEARCHPAGE || keyboardPathNavigation === undefined) {\n          const anchorTag = searchAllRef && searchAllRef.current && searchAllRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else {\n          if (keyboardPathNavigation instanceof HTMLElement) {\n            const toClick =\n              keyboardPathNavigation &&\n              keyboardPathNavigation.querySelector &&\n              (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n            toClick && toClick.click();\n          }\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
56
+ }), "{position:fixed;left:0;right:0;bottom:0;top:74px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAsF+D","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          searchSuggestionRef?.current?.closest('a')?.click();\n        } else if (keyboardPathNavigation instanceof HTMLElement) {\n          e.stopPropagation();\n          e.preventDefault();\n          const toClick =\n            keyboardPathNavigation &&\n            keyboardPathNavigation.querySelector &&\n            (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n          toClick && toClick.click();\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
57
57
  var StyledScrollableContent = /*#__PURE__*/_styled("div", {
58
58
  target: "e1mez1xb2",
59
59
  label: "StyledScrollableContent"
@@ -68,17 +68,17 @@ var StyledScrollableContent = /*#__PURE__*/_styled("div", {
68
68
  return 200 - props.extendHeight;
69
69
  }, ");}", mq.range({
70
70
  until: breakpoints.tablet
71
- }), "{padding:0 ", spacing.normal, " ", spacing.large, ";max-height:calc(100vh - 74px);}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AA2GwE","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        e.stopPropagation();\n        e.preventDefault();\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          const anchorTag =\n            searchSuggestionRef && searchSuggestionRef.current && searchSuggestionRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else if (keyboardPathNavigation === GO_TO_SEARCHPAGE || keyboardPathNavigation === undefined) {\n          const anchorTag = searchAllRef && searchAllRef.current && searchAllRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else {\n          if (keyboardPathNavigation instanceof HTMLElement) {\n            const toClick =\n              keyboardPathNavigation &&\n              keyboardPathNavigation.querySelector &&\n              (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n            toClick && toClick.click();\n          }\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
71
+ }), "{padding:0 ", spacing.normal, " ", spacing.large, ";max-height:calc(100vh - 74px);}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AA2GwE","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          searchSuggestionRef?.current?.closest('a')?.click();\n        } else if (keyboardPathNavigation instanceof HTMLElement) {\n          e.stopPropagation();\n          e.preventDefault();\n          const toClick =\n            keyboardPathNavigation &&\n            keyboardPathNavigation.querySelector &&\n            (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n          toClick && toClick.click();\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
72
72
  var StyledFooter = /*#__PURE__*/_styled("div", {
73
73
  target: "e1mez1xb1",
74
74
  label: "StyledFooter"
75
75
  })(mq.range({
76
76
  until: breakpoints.tabletWide
77
- }), "{display:none;}flex-direction:row-reverse;align-items:center;background:", colors.brand.greyLightest, ";border-top:1px solid ", colors.brand.greyLight, ";border-bottom-left-radius:", misc.borderRadius, ";border-bottom-right-radius:", misc.borderRadius, ";padding:", spacing.xsmall, " 0 ", spacing.xsmall, " ", spacing.large, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AA4H+B","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        e.stopPropagation();\n        e.preventDefault();\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          const anchorTag =\n            searchSuggestionRef && searchSuggestionRef.current && searchSuggestionRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else if (keyboardPathNavigation === GO_TO_SEARCHPAGE || keyboardPathNavigation === undefined) {\n          const anchorTag = searchAllRef && searchAllRef.current && searchAllRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else {\n          if (keyboardPathNavigation instanceof HTMLElement) {\n            const toClick =\n              keyboardPathNavigation &&\n              keyboardPathNavigation.querySelector &&\n              (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n            toClick && toClick.click();\n          }\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
77
+ }), "{display:none;}flex-direction:row-reverse;align-items:center;background:", colors.brand.greyLightest, ";border-top:1px solid ", colors.brand.greyLight, ";border-bottom-left-radius:", misc.borderRadius, ";border-bottom-right-radius:", misc.borderRadius, ";padding:", spacing.xsmall, " 0 ", spacing.xsmall, " ", spacing.large, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AA4H+B","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          searchSuggestionRef?.current?.closest('a')?.click();\n        } else if (keyboardPathNavigation instanceof HTMLElement) {\n          e.stopPropagation();\n          e.preventDefault();\n          const toClick =\n            keyboardPathNavigation &&\n            keyboardPathNavigation.querySelector &&\n            (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n          toClick && toClick.click();\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
78
78
  var StyledInstructions = /*#__PURE__*/_styled("div", {
79
79
  target: "e1mez1xb0",
80
80
  label: "StyledInstructions"
81
- })("flex:1;align-items:center;padding-right:", spacing.large, ";svg{width:24px;height:24px;border:1px solid ", colors.brand.grey, ";background:", colors.brand.greyLight, ";border-radius:2px;margin-left:2px;}span{display:inline-flex;", fonts.sizes(14, 1.1), ";margin:", spacing.xsmall, " ", spacing.small, " ", spacing.xsmall, " ", spacing.xsmall, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAyIqC","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        e.stopPropagation();\n        e.preventDefault();\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          const anchorTag =\n            searchSuggestionRef && searchSuggestionRef.current && searchSuggestionRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else if (keyboardPathNavigation === GO_TO_SEARCHPAGE || keyboardPathNavigation === undefined) {\n          const anchorTag = searchAllRef && searchAllRef.current && searchAllRef.current.closest('a');\n          if (anchorTag) {\n            anchorTag.click();\n          }\n        } else {\n          if (keyboardPathNavigation instanceof HTMLElement) {\n            const toClick =\n              keyboardPathNavigation &&\n              keyboardPathNavigation.querySelector &&\n              (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n            toClick && toClick.click();\n          }\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
81
+ })("flex:1;align-items:center;padding-right:", spacing.large, ";svg{width:24px;height:24px;border:1px solid ", colors.brand.grey, ";background:", colors.brand.greyLight, ";border-radius:2px;margin-left:2px;}span{display:inline-flex;", fonts.sizes(14, 1.1), ";margin:", spacing.xsmall, " ", spacing.small, " ", spacing.xsmall, " ", spacing.xsmall, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAyIqC","file":"SearchResultSleeve.tsx","sourcesContent":["/*\n * Copyright (c) 2019-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 { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from '@ndla/icons/common';\nimport SafeLink from '@ndla/safelink';\nimport { useTranslation } from 'react-i18next';\nimport ContentTypeResult from './ContentTypeResult';\nimport { highlightStyle } from './ContentTypeResultStyles';\nimport { ContentTypeResultType, Resource } from '../types';\n\nconst GO_TO_SEARCHPAGE = 'GO_TO_SEARCHPAGE';\nconst GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes('16px', '22px')};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? 'absolute' : 'static')};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  } else if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  } else {\n    return arr[currentIdx];\n  }\n};\n\nconst getDefaultCount = () => {\n  return window.innerWidth > 980 ? 7 : 3;\n};\n\nconst findPathForKeyboardNavigation = (\n  contentRef: HTMLDivElement | null,\n  current: HTMLElement | string | null,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const selectables = contentRef ? Array.from(contentRef.querySelectorAll('li')) : [];\n  const resultsContainingPaths: Array<string | HTMLElement> = (\n    [GO_TO_SEARCHPAGE, GO_TO_SUGGESTION] as Array<HTMLElement | string>\n  ).concat(...selectables);\n\n  // Nothing selected, goto either first or last depending on direction\n  if (current === null) {\n    switch (direction) {\n      case 1:\n        return resultsContainingPaths[0];\n      case -1:\n        return resultsContainingPaths[resultsContainingPaths.length - 1];\n      default:\n        return current;\n    }\n  } else {\n    return getNextElementInDirection(current, resultsContainingPaths, direction);\n  }\n};\n\ntype Props = {\n  result: Array<ContentTypeResultType>;\n  allResultUrl: string;\n  resourceToLinkProps: (resource: Resource) => {\n    to: string;\n  };\n  onNavigate?: VoidFunction;\n  infoText?: string;\n  ignoreContentTypeBadge?: boolean;\n  searchString: string;\n  loading: boolean;\n  frontpage?: boolean;\n  suggestion?: string;\n  suggestionUrl?: string;\n};\n\nconst SearchResultSleeve = ({\n  result,\n  allResultUrl,\n  resourceToLinkProps,\n  onNavigate,\n  infoText,\n  ignoreContentTypeBadge,\n  searchString,\n  loading,\n  frontpage,\n  suggestion,\n  suggestionUrl,\n}: Props) => {\n  const { t } = useTranslation();\n  const contentRef = useRef<HTMLDivElement>(null);\n  const searchAllRef = useRef<HTMLDivElement>(null);\n  const searchSuggestionRef = useRef<HTMLDivElement>(null);\n  const [keyboardPathNavigation, setKeyNavigation] = useState<HTMLElement | string | null>('');\n\n  useEffect(() => {\n    const onKeyDownEvent = (e: KeyboardEvent) => {\n      if (e.code === 'ArrowDown') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, 1);\n        });\n      } else if (e.code === 'ArrowUp') {\n        e.stopPropagation();\n        e.preventDefault();\n\n        setKeyNavigation((prevKeyPath) => {\n          return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);\n        });\n      } else if (e.code === 'Enter') {\n        if (keyboardPathNavigation === GO_TO_SUGGESTION) {\n          searchSuggestionRef?.current?.closest('a')?.click();\n        } else if (keyboardPathNavigation instanceof HTMLElement) {\n          e.stopPropagation();\n          e.preventDefault();\n          const toClick =\n            keyboardPathNavigation &&\n            keyboardPathNavigation.querySelector &&\n            (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));\n\n          toClick && toClick.click();\n        }\n      } else if (e.code === 'Tab') {\n        setKeyNavigation('');\n      }\n    };\n\n    window.addEventListener('keydown', onKeyDownEvent);\n    setKeyNavigation((prevKeyNav) => {\n      return findPathForKeyboardNavigation(contentRef.current, prevKeyNav, null);\n    });\n    return () => {\n      window.removeEventListener('keydown', onKeyDownEvent);\n    };\n  }, [result, contentRef, searchAllRef, keyboardPathNavigation]);\n\n  useEffect(() => {\n    const highlightedElement =\n      keyboardPathNavigation === GO_TO_SEARCHPAGE\n        ? searchAllRef.current\n        : contentRef.current && contentRef.current.querySelector('[data-highlighted=\"true\"]');\n\n    if (highlightedElement) {\n      highlightedElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest',\n      });\n    }\n  }, [keyboardPathNavigation]);\n\n  return (\n    <StyledSearchResultsWrapper frontpage={frontpage} ref={contentRef}>\n      <StyledScrollableContent extendHeight={frontpage ? 0 : 52}>\n        {infoText && (\n          <StyledAside>\n            <Wrench className=\"c-icon--22\" />\n            <span>{infoText}</span>\n          </StyledAside>\n        )}\n        <div>\n          <SearchLinkContainer>\n            <StyledSearchLink\n              css={keyboardPathNavigation === GO_TO_SEARCHPAGE && highlightStyle}\n              to={allResultUrl}\n              tabIndex={-1}\n            >\n              <SearchIcon className=\"c-icon--22\" />\n              <strong ref={searchAllRef}>{searchString}</strong>\n              <small>{t('welcomePage.searchAllInfo')}</small>\n            </StyledSearchLink>\n            {suggestion && suggestionUrl && (\n              <StyledSearchLink\n                css={keyboardPathNavigation === GO_TO_SUGGESTION && highlightStyle}\n                to={suggestionUrl}\n                tabIndex={-1}\n              >\n                <SearchIcon className=\"c-icon--22\" />\n                <small>{t('searchPage.resultType.searchPhraseSuggestion')}</small>\n                <strong ref={searchSuggestionRef}>{suggestion}</strong>\n              </StyledSearchLink>\n            )}\n          </SearchLinkContainer>\n          {result.map((contentTypeResult: ContentTypeResultType) => (\n            <ContentTypeResult\n              ignoreContentTypeBadge={!!ignoreContentTypeBadge}\n              onNavigate={onNavigate}\n              contentTypeResult={contentTypeResult}\n              resourceToLinkProps={resourceToLinkProps}\n              defaultCount={getDefaultCount()}\n              key={contentTypeResult.title}\n              keyboardPathNavigation={keyboardPathNavigation}\n              showAdditionalResources\n              messages={{\n                allResultLabel: t('searchPage.searchField.contentTypeResultShowMoreLabel'),\n                showLessResultLabel: t('searchPage.searchField.contentTypeResultShowLessLabel'),\n                noHit: t('searchPage.searchField.contentTypeResultNoHit'),\n              }}\n            />\n          ))}\n          {result.length === 0 && !loading && (\n            <StyledNoHits>{t('searchPage.searchField.contentTypeResultNoHit')}</StyledNoHits>\n          )}\n        </div>\n      </StyledScrollableContent>\n      <StyledFooter>\n        <StyledInstructions>\n          <ChevronUp />\n          <ChevronDown />\n          <span>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
82
82
  var getNextElementInDirection = function getNextElementInDirection(current, arr, direction) {
83
83
  var currentIdx = arr.indexOf(current);
84
84
  if (direction === 1) {
@@ -149,23 +149,14 @@ var SearchResultSleeve = function SearchResultSleeve(_ref2) {
149
149
  return findPathForKeyboardNavigation(contentRef.current, prevKeyPath, -1);
150
150
  });
151
151
  } else if (e.code === 'Enter') {
152
- e.stopPropagation();
153
- e.preventDefault();
154
152
  if (keyboardPathNavigation === GO_TO_SUGGESTION) {
155
- var anchorTag = searchSuggestionRef && searchSuggestionRef.current && searchSuggestionRef.current.closest('a');
156
- if (anchorTag) {
157
- anchorTag.click();
158
- }
159
- } else if (keyboardPathNavigation === GO_TO_SEARCHPAGE || keyboardPathNavigation === undefined) {
160
- var _anchorTag = searchAllRef && searchAllRef.current && searchAllRef.current.closest('a');
161
- if (_anchorTag) {
162
- _anchorTag.click();
163
- }
164
- } else {
165
- if (keyboardPathNavigation instanceof HTMLElement) {
166
- var toClick = keyboardPathNavigation && keyboardPathNavigation.querySelector && (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));
167
- toClick && toClick.click();
168
- }
153
+ var _searchSuggestionRef$, _searchSuggestionRef$2;
154
+ searchSuggestionRef === null || searchSuggestionRef === void 0 ? void 0 : (_searchSuggestionRef$ = searchSuggestionRef.current) === null || _searchSuggestionRef$ === void 0 ? void 0 : (_searchSuggestionRef$2 = _searchSuggestionRef$.closest('a')) === null || _searchSuggestionRef$2 === void 0 ? void 0 : _searchSuggestionRef$2.click();
155
+ } else if (keyboardPathNavigation instanceof HTMLElement) {
156
+ e.stopPropagation();
157
+ e.preventDefault();
158
+ var toClick = keyboardPathNavigation && keyboardPathNavigation.querySelector && (keyboardPathNavigation.querySelector('a') || keyboardPathNavigation.querySelector('button'));
159
+ toClick && toClick.click();
169
160
  }
170
161
  } else if (e.code === 'Tab') {
171
162
  setKeyNavigation('');