@ndla/ui 54.0.0 → 54.0.1
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.
- package/es/Article/ArticleByline.js +15 -7
- package/es/Article/ArticleParagraph.js +5 -5
- package/es/BlogPost/BlogPost.js +6 -5
- package/es/CampaignBlock/CampaignBlock.js +7 -7
- package/es/CodeBlock/CodeBlock.js +4 -4
- package/es/KeyFigure/KeyFigure.js +5 -5
- package/es/MyNdla/Resource/Folder.js +37 -24
- package/es/ResourceBox/ResourceBox.js +6 -6
- package/es/Search/SearchResultSleeve.js +11 -11
- package/es/Table/Table.js +6 -6
- package/es/TreeStructure/FolderItem.js +7 -7
- package/es/locale/messages-en.js +17 -3
- package/es/locale/messages-nb.js +17 -3
- package/es/locale/messages-nn.js +18 -4
- package/es/locale/messages-se.js +17 -3
- package/es/locale/messages-sma.js +17 -3
- package/lib/Article/ArticleByline.js +15 -7
- package/lib/Article/ArticleParagraph.js +5 -5
- package/lib/BlogPost/BlogPost.js +6 -5
- package/lib/CampaignBlock/CampaignBlock.js +7 -7
- package/lib/CodeBlock/CodeBlock.js +4 -4
- package/lib/KeyFigure/KeyFigure.js +5 -5
- package/lib/MyNdla/Resource/Folder.d.ts +3 -1
- package/lib/MyNdla/Resource/Folder.js +35 -22
- package/lib/ResourceBox/ResourceBox.js +6 -6
- package/lib/Search/SearchResultSleeve.js +11 -11
- package/lib/Table/Table.js +6 -6
- package/lib/TreeStructure/FolderItem.js +7 -7
- package/lib/locale/messages-en.d.ts +14 -0
- package/lib/locale/messages-en.js +17 -3
- package/lib/locale/messages-nb.d.ts +14 -0
- package/lib/locale/messages-nb.js +17 -3
- package/lib/locale/messages-nn.d.ts +15 -1
- package/lib/locale/messages-nn.js +18 -4
- package/lib/locale/messages-se.d.ts +14 -0
- package/lib/locale/messages-se.js +17 -3
- package/lib/locale/messages-sma.d.ts +14 -0
- package/lib/locale/messages-sma.js +17 -3
- package/package.json +10 -10
- package/src/Article/ArticleByline.tsx +9 -2
- package/src/Article/ArticleParagraph.tsx +3 -0
- package/src/BlogPost/BlogPost.tsx +2 -1
- package/src/CampaignBlock/CampaignBlock.tsx +1 -1
- package/src/CodeBlock/CodeBlock.tsx +1 -1
- package/src/KeyFigure/KeyFigure.tsx +1 -1
- package/src/MyNdla/Resource/Folder.stories.tsx +5 -0
- package/src/MyNdla/Resource/Folder.tsx +41 -8
- package/src/ResourceBox/ResourceBox.tsx +2 -2
- package/src/Search/SearchResultSleeve.tsx +3 -3
- package/src/Table/Table.stories.tsx +39 -0
- package/src/Table/Table.tsx +11 -0
- package/src/TreeStructure/FolderItem.tsx +1 -1
- package/src/locale/messages-en.ts +14 -0
- package/src/locale/messages-nb.ts +14 -0
- package/src/locale/messages-nn.ts +15 -1
- package/src/locale/messages-se.ts +14 -0
- package/src/locale/messages-sma.ts +14 -0
|
@@ -27,27 +27,27 @@ const GO_TO_SUGGESTION = "GO_TO_SUGGESTION";
|
|
|
27
27
|
const StyledNoHits = /*#__PURE__*/(0, _base.default)("div", {
|
|
28
28
|
target: "e1mez1xb7",
|
|
29
29
|
label: "StyledNoHits"
|
|
30
|
-
})(_core.fonts.sizes(16, 1.1), ";color:", _core.colors.text.primary, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAqB+B","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
|
|
30
|
+
})(_core.fonts.sizes(16, 1.1), ";color:", _core.colors.text.primary, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAqB+B","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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"]} */"));
|
|
31
31
|
const StyledAside = /*#__PURE__*/(0, _base.default)("aside", {
|
|
32
32
|
target: "e1mez1xb6",
|
|
33
33
|
label: "StyledAside"
|
|
34
34
|
})(_core.fonts.sizes("16px", "22px"), ";display:flex;align-items:flex-start;margin:0;color:", _core.colors.text.primary, ";padding:", _core.spacing.normal, " ", _core.spacing.large, " ", _core.spacing.normal, " ", _core.spacing.normal, ";background:", _core.colors.support.yellowLight, ";border-radius:", _core.misc.borderRadius, ";span{display:block;max-width:700px;padding-left:", _core.spacing.small, ";flex:1;}svg{transform:translateY(6px);}", _core.mq.range({
|
|
35
35
|
until: _core.breakpoints.tablet
|
|
36
|
-
}), "{display:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AA0BgC","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
|
|
36
|
+
}), "{display:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AA0BgC","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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"]} */"));
|
|
37
37
|
const SearchLinkContainer = /*#__PURE__*/(0, _base.default)("div", {
|
|
38
38
|
target: "e1mez1xb5",
|
|
39
39
|
label: "SearchLinkContainer"
|
|
40
|
-
})("margin:", _core.spacing.normal, " -", _core.spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAiDsC","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
|
|
40
|
+
})("margin:", _core.spacing.normal, " -", _core.spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAiDsC","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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"]} */"));
|
|
41
41
|
const StyledSearchLink = /*#__PURE__*/(0, _base.default)(_safelink.SafeLink, {
|
|
42
42
|
target: "e1mez1xb4",
|
|
43
43
|
label: "StyledSearchLink"
|
|
44
|
-
})("width:100%;box-shadow:none;border:0;background:transparent;display:inline-flex;flex-grow:1;align-items:center;padding:", _core.spacing.xsmall, " ", _core.spacing.small, ";line-height:1.7rem;color:", _core.colors.brand.primary, ";strong{font-weight:", _core.fonts.weight.semibold, ";margin-left:", _core.spacing.xsmall, ";}&:focus{", _ContentTypeResultStyles.highlightStyle, ";}&:hover{strong{text-decoration:underline;}}small{color:", _core.colors.text.light, ";padding-left:", _core.spacing.xsmall, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAqDyC","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
|
|
44
|
+
})("width:100%;box-shadow:none;border:0;background:transparent;display:inline-flex;flex-grow:1;align-items:center;padding:", _core.spacing.xsmall, " ", _core.spacing.small, ";line-height:1.7rem;color:", _core.colors.brand.primary, ";strong{font-weight:", _core.fonts.weight.semibold, ";margin-left:", _core.spacing.xsmall, ";}&:focus{", _ContentTypeResultStyles.highlightStyle, ";}&:hover{strong{text-decoration:underline;}}small{color:", _core.colors.text.light, ";padding-left:", _core.spacing.xsmall, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAqDyC","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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"]} */"));
|
|
45
45
|
const StyledSearchResultsWrapper = /*#__PURE__*/(0, _base.default)("section", {
|
|
46
46
|
target: "e1mez1xb3",
|
|
47
47
|
label: "StyledSearchResultsWrapper"
|
|
48
48
|
})("background:#fff;width:100%;position:", props => props.frontpage ? "absolute" : "static", ";left:0;right:0;top:62px;border-radius:", _core.misc.borderRadius, ";", _core.mq.range({
|
|
49
49
|
until: _core.breakpoints.tablet
|
|
50
|
-
}), "{position:fixed;left:0;right:0;bottom:0;top:74px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAsF+D","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
|
|
50
|
+
}), "{position:fixed;left:0;right:0;bottom:0;top:74px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAsF+D","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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"]} */"));
|
|
51
51
|
const StyledScrollableContent = /*#__PURE__*/(0, _base.default)("div", {
|
|
52
52
|
target: "e1mez1xb2",
|
|
53
53
|
label: "StyledScrollableContent"
|
|
@@ -56,17 +56,17 @@ const StyledScrollableContent = /*#__PURE__*/(0, _base.default)("div", {
|
|
|
56
56
|
until: _core.breakpoints.tabletWide
|
|
57
57
|
}), "{max-height:calc(100vh - ", props => 200 - props.extendHeight, ");}", _core.mq.range({
|
|
58
58
|
until: _core.breakpoints.tablet
|
|
59
|
-
}), "{padding:0 ", _core.spacing.normal, " ", _core.spacing.large, ";max-height:calc(100vh - 74px);}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AA2GwE","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
|
|
59
|
+
}), "{padding:0 ", _core.spacing.normal, " ", _core.spacing.large, ";max-height:calc(100vh - 74px);}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AA2GwE","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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"]} */"));
|
|
60
60
|
const StyledFooter = /*#__PURE__*/(0, _base.default)("div", {
|
|
61
61
|
target: "e1mez1xb1",
|
|
62
62
|
label: "StyledFooter"
|
|
63
63
|
})(_core.mq.range({
|
|
64
64
|
until: _core.breakpoints.tabletWide
|
|
65
|
-
}), "{display:none;}flex-direction:row-reverse;align-items:center;background:", _core.colors.brand.greyLightest, ";border-top:1px solid ", _core.colors.brand.greyLight, ";border-bottom-left-radius:", _core.misc.borderRadius, ";border-bottom-right-radius:", _core.misc.borderRadius, ";padding:", _core.spacing.xsmall, " 0 ", _core.spacing.xsmall, " ", _core.spacing.large, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AA4H+B","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
|
|
65
|
+
}), "{display:none;}flex-direction:row-reverse;align-items:center;background:", _core.colors.brand.greyLightest, ";border-top:1px solid ", _core.colors.brand.greyLight, ";border-bottom-left-radius:", _core.misc.borderRadius, ";border-bottom-right-radius:", _core.misc.borderRadius, ";padding:", _core.spacing.xsmall, " 0 ", _core.spacing.xsmall, " ", _core.spacing.large, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AA4H+B","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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"]} */"));
|
|
66
66
|
const StyledInstructions = /*#__PURE__*/(0, _base.default)("div", {
|
|
67
67
|
target: "e1mez1xb0",
|
|
68
68
|
label: "StyledInstructions"
|
|
69
|
-
})("flex:1;align-items:center;padding-right:", _core.spacing.large, ";svg{width:24px;height:24px;border:1px solid ", _core.colors.brand.grey, ";background:", _core.colors.brand.greyLight, ";border-radius:2px;margin-left:2px;}span{display:inline-flex;", _core.fonts.sizes(14, 1.1), ";margin:", _core.spacing.xsmall, " ", _core.spacing.small, " ", _core.spacing.xsmall, " ", _core.spacing.xsmall, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAyIqC","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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>Naviger med piltastene</span>\n          <KeyboardReturn />\n          <span>Velg</span>\n          <Esc />\n          <span>Lukk søk</span>\n        </StyledInstructions>\n      </StyledFooter>\n    </StyledSearchResultsWrapper>\n  );\n};\n\nexport default SearchResultSleeve;\n"]} */"));
|
|
69
|
+
})("flex:1;align-items:center;padding-right:", _core.spacing.large, ";svg{width:24px;height:24px;border:1px solid ", _core.colors.brand.grey, ";background:", _core.colors.brand.greyLight, ";border-radius:2px;margin-left:2px;}span{display:inline-flex;", _core.fonts.sizes(14, 1.1), ";margin:", _core.spacing.xsmall, " ", _core.spacing.small, " ", _core.spacing.xsmall, " ", _core.spacing.xsmall, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["SearchResultSleeve.tsx"],"names":[],"mappings":"AAyIqC","file":"SearchResultSleeve.tsx","sourcesContent":["/**\n * Copyright (c) 2019-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from \"react\";\nimport { useTranslation } from \"react-i18next\";\nimport styled from \"@emotion/styled\";\nimport { breakpoints, colors, fonts, misc, mq, spacing } from \"@ndla/core\";\nimport { ChevronDown, ChevronUp, Esc, KeyboardReturn, Search as SearchIcon, Wrench } from \"@ndla/icons/common\";\nimport { SafeLink } from \"@ndla/safelink\";\nimport ContentTypeResult from \"./ContentTypeResult\";\nimport { highlightStyle } from \"./ContentTypeResultStyles\";\nimport { ContentTypeResultType, Resource } from \"../types\";\n\nconst GO_TO_SEARCHPAGE = \"GO_TO_SEARCHPAGE\";\nconst GO_TO_SUGGESTION = \"GO_TO_SUGGESTION\";\n\nconst StyledNoHits = styled.div`\n  ${fonts.sizes(16, 1.1)};\n  color: ${colors.text.primary};\n`;\n\nconst StyledAside = styled.aside`\n  ${fonts.sizes(\"16px\", \"22px\")};\n  display: flex;\n  align-items: flex-start;\n  margin: 0;\n  color: ${colors.text.primary};\n  padding: ${spacing.normal} ${spacing.large} ${spacing.normal} ${spacing.normal};\n  background: ${colors.support.yellowLight};\n  border-radius: ${misc.borderRadius};\n  span {\n    display: block;\n    max-width: 700px;\n    padding-left: ${spacing.small};\n    flex: 1;\n  }\n  svg {\n    transform: translateY(6px);\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    display: none;\n  }\n`;\n\nconst SearchLinkContainer = styled.div`\n  margin: ${spacing.normal} -${spacing.small};\n`;\n\nconst StyledSearchLink = styled(SafeLink)`\n  width: 100%;\n  box-shadow: none;\n  border: 0;\n  background: transparent;\n  display: inline-flex;\n  flex-grow: 1;\n  align-items: center;\n  padding: ${spacing.xsmall} ${spacing.small};\n  line-height: 1.7rem;\n  color: ${colors.brand.primary};\n  strong {\n    font-weight: ${fonts.weight.semibold};\n    margin-left: ${spacing.xsmall};\n  }\n  &:focus {\n    ${highlightStyle}\n  }\n  &:hover {\n    strong {\n      text-decoration: underline;\n    }\n  }\n  small {\n    color: ${colors.text.light};\n    padding-left: ${spacing.xsmall};\n  }\n`;\n\ntype WrapperProps = {\n  frontpage?: boolean;\n};\n\nconst StyledSearchResultsWrapper = styled.section<WrapperProps>`\n  background: #fff;\n  width: 100%;\n  position: ${(props) => (props.frontpage ? \"absolute\" : \"static\")};\n  left: 0;\n  right: 0;\n  top: 62px;\n  border-radius: ${misc.borderRadius};\n  ${mq.range({ until: breakpoints.tablet })} {\n    position: fixed;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    top: 74px;\n  }\n`;\n\ntype StyledScrollableContentProps = {\n  extendHeight: number;\n};\n\nconst StyledScrollableContent = styled.div<StyledScrollableContentProps>`\n  max-height: calc(100vh - ${(props) => 260 - props.extendHeight}px);\n  overflow-y: auto;\n  -webkit-overflow-scrolling: touch;\n  overflow-x: hidden;\n  // prettier-ignore\n  padding: ${(props) =>\n    props.extendHeight ? spacing.normal : spacing.large} ${spacing.large} ${spacing.large} ${spacing.large};\n  ${mq.range({ from: breakpoints.tablet, until: breakpoints.tabletWide })} {\n    max-height: calc(100vh - ${(props) => 200 - props.extendHeight});\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    padding: 0 ${spacing.normal} ${spacing.large};\n    max-height: calc(100vh - 74px);\n  }\n`;\n\nconst StyledFooter = styled.div`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: none;\n  }\n  flex-direction: row-reverse;\n  align-items: center;\n  background: ${colors.brand.greyLightest};\n  border-top: 1px solid ${colors.brand.greyLight};\n  border-bottom-left-radius: ${misc.borderRadius};\n  border-bottom-right-radius: ${misc.borderRadius};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.large};\n`;\n\nconst StyledInstructions = styled.div`\n  flex: 1;\n  align-items: center;\n  padding-right: ${spacing.large};\n  svg {\n    width: 24px;\n    height: 24px;\n    border: 1px solid ${colors.brand.grey};\n    background: ${colors.brand.greyLight};\n    border-radius: 2px;\n    margin-left: 2px;\n  }\n  span {\n    display: inline-flex;\n    ${fonts.sizes(14, 1.1)};\n    margin: ${spacing.xsmall} ${spacing.small} ${spacing.xsmall} ${spacing.xsmall};\n  }\n`;\n\nconst getNextElementInDirection = (\n  current: HTMLElement | string,\n  arr: Array<HTMLElement | string>,\n  direction: 1 | -1 | null,\n): HTMLElement | string | null => {\n  const currentIdx = arr.indexOf(current);\n\n  if (direction === 1) {\n    const idx = currentIdx + 1 > arr.length - 1 ? 0 : currentIdx + 1;\n    return arr[idx];\n  }\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"]} */"));
|
|
70
70
|
const getNextElementInDirection = (current, arr, direction) => {
|
|
71
71
|
const currentIdx = arr.indexOf(current);
|
|
72
72
|
if (direction === 1) {
|
|
@@ -224,11 +224,11 @@ const SearchResultSleeve = _ref => {
|
|
|
224
224
|
}), (0, _jsxRuntime.jsx)(StyledFooter, {
|
|
225
225
|
children: (0, _jsxRuntime.jsxs)(StyledInstructions, {
|
|
226
226
|
children: [(0, _jsxRuntime.jsx)(_common.ChevronUp, {}), (0, _jsxRuntime.jsx)(_common.ChevronDown, {}), (0, _jsxRuntime.jsx)("span", {
|
|
227
|
-
children: "
|
|
227
|
+
children: t("siteNav.navigate")
|
|
228
228
|
}), (0, _jsxRuntime.jsx)(_common.KeyboardReturn, {}), (0, _jsxRuntime.jsx)("span", {
|
|
229
|
-
children: "
|
|
229
|
+
children: t("siteNav.select")
|
|
230
230
|
}), (0, _jsxRuntime.jsx)(_common.Esc, {}), (0, _jsxRuntime.jsx)("span", {
|
|
231
|
-
children: "
|
|
231
|
+
children: t("siteNav.close")
|
|
232
232
|
})]
|
|
233
233
|
})
|
|
234
234
|
})]
|