@ndla/ui 55.0.5-alpha.0 → 55.0.8-alpha.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -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":"AAsB+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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AAsB+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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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=\"medium\" />\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=\"medium\" />\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=\"medium\" />\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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AA2BgC","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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AA2BgC","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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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=\"medium\" />\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=\"medium\" />\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=\"medium\" />\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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AAkDsC","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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AAkDsC","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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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=\"medium\" />\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=\"medium\" />\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=\"medium\" />\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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AAsDyC","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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AAsDyC","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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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=\"medium\" />\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=\"medium\" />\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=\"medium\" />\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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AAuF+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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AAuF+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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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=\"medium\" />\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=\"medium\" />\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=\"medium\" />\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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AA4GwE","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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AA4GwE","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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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=\"medium\" />\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=\"medium\" />\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=\"medium\" />\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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AA6H+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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AA6H+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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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=\"medium\" />\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=\"medium\" />\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=\"medium\" />\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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AA0IqC","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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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":"AA0IqC","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\n/** @jsxImportSource @emotion/react */\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  }\n  if (direction === -1) {\n    const idx = currentIdx - 1 < 0 ? arr.length - 1 : currentIdx - 1;\n    return arr[idx];\n  }\n  return arr[currentIdx];\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  }\n  return getNextElementInDirection(current, resultsContainingPaths, direction);\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?.querySelector &&\n            (keyboardPathNavigation.querySelector(\"a\") || keyboardPathNavigation.querySelector(\"button\"));\n\n          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?.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=\"medium\" />\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=\"medium\" />\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=\"medium\" />\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>{t(\"siteNav.navigate\")}</span>\n          <KeyboardReturn />\n          <span>{t(\"siteNav.select\")}</span>\n          <Esc />\n          <span>{t(\"siteNav.close\")}</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) {
@@ -167,7 +167,7 @@ const SearchResultSleeve = _ref => {
167
167
  extendHeight: frontpage ? 0 : 52,
168
168
  children: [infoText && _jsxs(StyledAside, {
169
169
  children: [_jsx(Wrench, {
170
- size: "normal"
170
+ size: "medium"
171
171
  }), _jsx("span", {
172
172
  children: infoText
173
173
  })]
@@ -178,7 +178,7 @@ const SearchResultSleeve = _ref => {
178
178
  to: allResultUrl,
179
179
  tabIndex: -1,
180
180
  children: [_jsx(SearchIcon, {
181
- size: "normal"
181
+ size: "medium"
182
182
  }), _jsx("strong", {
183
183
  ref: searchAllRef,
184
184
  children: searchString
@@ -190,7 +190,7 @@ const SearchResultSleeve = _ref => {
190
190
  to: suggestionUrl,
191
191
  tabIndex: -1,
192
192
  children: [_jsx(SearchIcon, {
193
- size: "normal"
193
+ size: "medium"
194
194
  }), _jsx("small", {
195
195
  children: t("searchPage.resultType.searchPhraseSuggestion")
196
196
  }), _jsx("strong", {