@ndla/ui 50.6.0 → 50.6.2

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.
@@ -21,27 +21,27 @@ const GO_TO_SUGGESTION = 'GO_TO_SUGGESTION';
21
21
  const StyledNoHits = /*#__PURE__*/_styled("div", {
22
22
  target: "e1mez1xb7",
23
23
  label: "StyledNoHits"
24
- })(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 { useTranslation } from 'react-i18next';\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 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"]} */"));
24
+ })(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 { useTranslation } from 'react-i18next';\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 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 size=\"normal\" />\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 size=\"normal\" />\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 size=\"normal\" />\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"]} */"));
25
25
  const StyledAside = /*#__PURE__*/_styled("aside", {
26
26
  target: "e1mez1xb6",
27
27
  label: "StyledAside"
28
28
  })(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({
29
29
  until: breakpoints.tablet
30
- }), "{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 { useTranslation } from 'react-i18next';\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 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"]} */"));
30
+ }), "{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 { useTranslation } from 'react-i18next';\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 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 size=\"normal\" />\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 size=\"normal\" />\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 size=\"normal\" />\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"]} */"));
31
31
  const SearchLinkContainer = /*#__PURE__*/_styled("div", {
32
32
  target: "e1mez1xb5",
33
33
  label: "SearchLinkContainer"
34
- })("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 { useTranslation } from 'react-i18next';\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 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"]} */"));
34
+ })("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 { useTranslation } from 'react-i18next';\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 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 size=\"normal\" />\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 size=\"normal\" />\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 size=\"normal\" />\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
  const StyledSearchLink = /*#__PURE__*/_styled(SafeLink, {
36
36
  target: "e1mez1xb4",
37
37
  label: "StyledSearchLink"
38
- })("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 { useTranslation } from 'react-i18next';\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 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"]} */"));
38
+ })("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 { useTranslation } from 'react-i18next';\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 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 size=\"normal\" />\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 size=\"normal\" />\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 size=\"normal\" />\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"]} */"));
39
39
  const StyledSearchResultsWrapper = /*#__PURE__*/_styled("section", {
40
40
  target: "e1mez1xb3",
41
41
  label: "StyledSearchResultsWrapper"
42
42
  })("background:#fff;width:100%;position:", props => props.frontpage ? 'absolute' : 'static', ";left:0;right:0;top:62px;border-radius:", misc.borderRadius, ";", mq.range({
43
43
  until: breakpoints.tablet
44
- }), "{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 { useTranslation } from 'react-i18next';\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 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"]} */"));
44
+ }), "{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 { useTranslation } from 'react-i18next';\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 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 size=\"normal\" />\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 size=\"normal\" />\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 size=\"normal\" />\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
  const StyledScrollableContent = /*#__PURE__*/_styled("div", {
46
46
  target: "e1mez1xb2",
47
47
  label: "StyledScrollableContent"
@@ -50,17 +50,17 @@ const StyledScrollableContent = /*#__PURE__*/_styled("div", {
50
50
  until: breakpoints.tabletWide
51
51
  }), "{max-height:calc(100vh - ", props => 200 - props.extendHeight, ");}", mq.range({
52
52
  until: breakpoints.tablet
53
- }), "{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 { useTranslation } from 'react-i18next';\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 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"]} */"));
53
+ }), "{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 { useTranslation } from 'react-i18next';\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 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 size=\"normal\" />\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 size=\"normal\" />\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 size=\"normal\" />\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"]} */"));
54
54
  const StyledFooter = /*#__PURE__*/_styled("div", {
55
55
  target: "e1mez1xb1",
56
56
  label: "StyledFooter"
57
57
  })(mq.range({
58
58
  until: breakpoints.tabletWide
59
- }), "{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 { useTranslation } from 'react-i18next';\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 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"]} */"));
59
+ }), "{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 { useTranslation } from 'react-i18next';\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 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 size=\"normal\" />\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 size=\"normal\" />\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 size=\"normal\" />\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"]} */"));
60
60
  const StyledInstructions = /*#__PURE__*/_styled("div", {
61
61
  target: "e1mez1xb0",
62
62
  label: "StyledInstructions"
63
- })("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 { useTranslation } from 'react-i18next';\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 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"]} */"));
63
+ })("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 { useTranslation } from 'react-i18next';\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 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 size=\"normal\" />\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 size=\"normal\" />\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 size=\"normal\" />\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"]} */"));
64
64
  const getNextElementInDirection = (current, arr, direction) => {
65
65
  const currentIdx = arr.indexOf(current);
66
66
  if (direction === 1) {
@@ -166,7 +166,7 @@ const SearchResultSleeve = _ref => {
166
166
  extendHeight: frontpage ? 0 : 52,
167
167
  children: [infoText && _jsxs(StyledAside, {
168
168
  children: [_jsx(Wrench, {
169
- className: "c-icon--22"
169
+ size: "normal"
170
170
  }), _jsx("span", {
171
171
  children: infoText
172
172
  })]
@@ -177,7 +177,7 @@ const SearchResultSleeve = _ref => {
177
177
  to: allResultUrl,
178
178
  tabIndex: -1,
179
179
  children: [_jsx(SearchIcon, {
180
- className: "c-icon--22"
180
+ size: "normal"
181
181
  }), _jsx("strong", {
182
182
  ref: searchAllRef,
183
183
  children: searchString
@@ -189,7 +189,7 @@ const SearchResultSleeve = _ref => {
189
189
  to: suggestionUrl,
190
190
  tabIndex: -1,
191
191
  children: [_jsx(SearchIcon, {
192
- className: "c-icon--22"
192
+ size: "normal"
193
193
  }), _jsx("small", {
194
194
  children: t('searchPage.resultType.searchPhraseSuggestion')
195
195
  }), _jsx("strong", {