@ndla/ui 34.0.1 → 34.1.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/Figure/Figure.js +3 -2
- package/es/FileList/File.js +6 -6
- package/es/LanguageSelector/LanguageSelector.js +68 -99
- package/es/NDLAFilm/AboutNdlaFilm.js +13 -12
- package/es/NDLAFilm/AllMoviesAlphabetically.js +79 -144
- package/es/NDLAFilm/FilmContentCard.js +41 -26
- package/es/NDLAFilm/FilmContentCardTags.js +5 -3
- package/es/NDLAFilm/FilmMovieList.js +13 -8
- package/es/NDLAFilm/FilmMovieSearch.js +6 -5
- package/es/NDLAFilm/FilmSlideshow.js +44 -20
- package/es/NDLAFilm/filmStyles.js +2 -2
- package/es/ResourceGroup/ResourceItem.js +72 -48
- package/es/Search/ActiveFilterContent.js +6 -5
- package/es/Search/ContentTypeResult.js +6 -3
- package/es/SearchTypeResult/ActiveFilterContent.js +9 -10
- package/es/Topic/Topic.js +171 -213
- package/es/all.css +1 -1
- package/es/locale/messages-en.js +3 -1
- package/es/locale/messages-nb.js +3 -1
- package/es/locale/messages-nn.js +7 -5
- package/es/locale/messages-se.js +2 -0
- package/es/locale/messages-sma.js +3 -1
- package/lib/Figure/Figure.d.ts +2 -1
- package/lib/Figure/Figure.js +3 -2
- package/lib/FileList/File.js +6 -6
- package/lib/LanguageSelector/LanguageSelector.d.ts +6 -15
- package/lib/LanguageSelector/LanguageSelector.js +65 -99
- package/lib/NDLAFilm/AboutNdlaFilm.js +11 -14
- package/lib/NDLAFilm/AllMoviesAlphabetically.d.ts +1 -2
- package/lib/NDLAFilm/AllMoviesAlphabetically.js +77 -142
- package/lib/NDLAFilm/FilmContentCard.d.ts +7 -0
- package/lib/NDLAFilm/FilmContentCard.js +41 -26
- package/lib/NDLAFilm/FilmContentCardTags.d.ts +2 -1
- package/lib/NDLAFilm/FilmContentCardTags.js +5 -3
- package/lib/NDLAFilm/FilmMovieList.js +12 -7
- package/lib/NDLAFilm/FilmMovieSearch.js +5 -4
- package/lib/NDLAFilm/FilmSlideshow.js +44 -20
- package/lib/NDLAFilm/filmStyles.js +2 -2
- package/lib/Resource/resourceComponents.d.ts +1 -1
- package/lib/ResourceGroup/ResourceItem.d.ts +2 -2
- package/lib/ResourceGroup/ResourceItem.js +72 -48
- package/lib/Search/ActiveFilterContent.d.ts +1 -1
- package/lib/Search/ActiveFilterContent.js +9 -5
- package/lib/Search/ContentTypeResult.js +6 -3
- package/lib/SearchTypeResult/ActiveFilterContent.d.ts +1 -1
- package/lib/SearchTypeResult/ActiveFilterContent.js +12 -10
- package/lib/Topic/Topic.js +170 -215
- package/lib/all.css +1 -1
- package/lib/locale/messages-en.d.ts +2 -0
- package/lib/locale/messages-en.js +3 -1
- package/lib/locale/messages-nb.d.ts +2 -0
- package/lib/locale/messages-nb.js +3 -1
- package/lib/locale/messages-nn.d.ts +4 -2
- package/lib/locale/messages-nn.js +7 -5
- package/lib/locale/messages-se.d.ts +2 -0
- package/lib/locale/messages-se.js +2 -0
- package/lib/locale/messages-sma.d.ts +2 -0
- package/lib/locale/messages-sma.js +3 -1
- package/package.json +15 -14
- package/src/Figure/Figure.tsx +6 -2
- package/src/FileList/File.tsx +4 -4
- package/src/LanguageSelector/LanguageSelector.stories.tsx +48 -0
- package/src/LanguageSelector/LanguageSelector.tsx +71 -149
- package/src/NDLAFilm/AboutNdlaFilm.tsx +11 -14
- package/src/NDLAFilm/AllMoviesAlphabetically.tsx +44 -160
- package/src/NDLAFilm/FilmContentCard.tsx +40 -21
- package/src/NDLAFilm/FilmContentCardTags.tsx +3 -2
- package/src/NDLAFilm/FilmMovieList.tsx +14 -7
- package/src/NDLAFilm/FilmMovieSearch.tsx +2 -2
- package/src/NDLAFilm/FilmSlideshow.tsx +49 -40
- package/src/NDLAFilm/filmStyles.ts +1 -1
- package/src/ResourceGroup/ResourceItem.tsx +79 -94
- package/src/Search/ActiveFilterContent.tsx +4 -3
- package/src/Search/ContentTypeResult.tsx +3 -1
- package/src/SearchTypeResult/ActiveFilterContent.tsx +7 -8
- package/src/Topic/Topic.tsx +166 -193
- package/src/locale/messages-en.ts +3 -1
- package/src/locale/messages-nb.ts +3 -1
- package/src/locale/messages-nn.ts +5 -4
- package/src/locale/messages-se.ts +2 -0
- package/src/locale/messages-sma.ts +3 -1
- package/src/main.scss +0 -1
- package/es/LanguageSelector/LanguageSelectorContent.js +0 -61
- package/es/Subject/SubjectCarousel.js +0 -133
- package/lib/LanguageSelector/LanguageSelectorContent.d.ts +0 -15
- package/lib/LanguageSelector/LanguageSelectorContent.js +0 -68
- package/lib/Subject/SubjectCarousel.d.ts +0 -18
- package/lib/Subject/SubjectCarousel.js +0 -138
- package/src/LanguageSelector/LanguageSelectorContent.tsx +0 -80
- package/src/NDLAFilm/component.film-movielist.scss +0 -105
- package/src/Subject/SubjectCarousel.tsx +0 -162
package/lib/Topic/Topic.js
CHANGED
|
@@ -1,6 +1,5 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
|
|
3
|
-
function _typeof(obj) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (obj) { return typeof obj; } : function (obj) { return obj && "function" == typeof Symbol && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }, _typeof(obj); }
|
|
4
3
|
Object.defineProperty(exports, "__esModule", {
|
|
5
4
|
value: true
|
|
6
5
|
});
|
|
@@ -10,8 +9,8 @@ var _react = _interopRequireDefault(require("react"));
|
|
|
10
9
|
var _core = require("@ndla/core");
|
|
11
10
|
var _htmlReactParser = _interopRequireDefault(require("html-react-parser"));
|
|
12
11
|
var _common = require("@ndla/icons/common");
|
|
13
|
-
var _modal =
|
|
14
|
-
var _button =
|
|
12
|
+
var _modal = require("@ndla/modal");
|
|
13
|
+
var _button = require("@ndla/button");
|
|
15
14
|
var _action = require("@ndla/icons/action");
|
|
16
15
|
var _react2 = require("@emotion/react");
|
|
17
16
|
var _reactI18next = require("react-i18next");
|
|
@@ -20,268 +19,224 @@ var _Navigation = require("../Navigation");
|
|
|
20
19
|
var _Image = require("../Image");
|
|
21
20
|
var _Messages = require("../Messages");
|
|
22
21
|
var _jsxRuntime = require("@emotion/react/jsx-runtime");
|
|
23
|
-
function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function _getRequireWildcardCache(nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); }
|
|
24
|
-
function _interopRequireWildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || _typeof(obj) !== "object" && typeof obj !== "function") { return { "default": obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj["default"] = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
|
|
25
22
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }
|
|
26
23
|
function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
|
|
27
|
-
var Wrapper = /*#__PURE__*/(0, _base["default"])("
|
|
28
|
-
target: "
|
|
24
|
+
var Wrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
25
|
+
target: "e111uita13",
|
|
29
26
|
label: "Wrapper"
|
|
30
|
-
})(function (props) {
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
}), "{padding:60px 160px;};label:Wrapper;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAmCO","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
38
|
-
}, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAgC0C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
27
|
+
})("display:flex;flex-direction:column;gap:", _core.spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA6B0B","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
28
|
+
var frameStyle = /*#__PURE__*/(0, _react2.css)(_core.mq.range({
|
|
29
|
+
from: _core.breakpoints.tabletWide
|
|
30
|
+
}), "{padding:40px 40px;border:2px solid ", _core.colors.brand.neutral7, ";}", _core.mq.range({
|
|
31
|
+
from: _core.breakpoints.desktop
|
|
32
|
+
}), "{padding:40px 80px;}", _core.mq.range({
|
|
33
|
+
from: '1180px'
|
|
34
|
+
}), "{padding:60px 160px;};label:frameStyle;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAmCsB","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
35
|
+
var _invertedStyle = /*#__PURE__*/(0, _react2.css)("color:", _core.colors.white, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAgD0B","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
39
36
|
var TopicHeaderVisualElementWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
40
|
-
target: "
|
|
37
|
+
target: "e111uita12",
|
|
41
38
|
label: "TopicHeaderVisualElementWrapper"
|
|
42
|
-
})("
|
|
39
|
+
})("position:relative;width:100px;height:100px;", _core.mq.range({
|
|
43
40
|
from: _core.breakpoints.mobileWide
|
|
44
41
|
}), "{width:150px;height:150px;}", _core.mq.range({
|
|
45
|
-
from: _core.breakpoints.
|
|
46
|
-
}), "{width:200px;height:200px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAiDkD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
42
|
+
from: _core.breakpoints.tabletWide
|
|
43
|
+
}), "{width:200px;height:200px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAoDkD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
47
44
|
var ShowVisualElementWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
48
|
-
target: "
|
|
45
|
+
target: "e111uita11",
|
|
49
46
|
label: "ShowVisualElementWrapper"
|
|
50
47
|
})(process.env.NODE_ENV === "production" ? {
|
|
51
|
-
name: "
|
|
52
|
-
styles: "border-radius:50%;width:100%;height:100%;overflow:hidden;-webkit-mask-image:-webkit-radial-gradient(white, black)"
|
|
48
|
+
name: "1v6dfx3",
|
|
49
|
+
styles: "border-radius:50%;width:100%;height:100%;overflow:hidden;aspect-ratio:1;-webkit-mask-image:-webkit-radial-gradient(white, black)"
|
|
53
50
|
} : {
|
|
54
|
-
name: "
|
|
55
|
-
styles: "border-radius:50%;width:100%;height:100%;overflow:hidden;-webkit-mask-image:-webkit-radial-gradient(white, black)",
|
|
56
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAiE2C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
|
|
51
|
+
name: "1v6dfx3",
|
|
52
|
+
styles: "border-radius:50%;width:100%;height:100%;overflow:hidden;aspect-ratio:1;-webkit-mask-image:-webkit-radial-gradient(white, black)",
|
|
53
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAkE2C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
|
|
57
54
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
58
55
|
});
|
|
59
|
-
var VisualElementButton = /*#__PURE__*/(0, _base["default"])(_button
|
|
60
|
-
target: "
|
|
56
|
+
var VisualElementButton = /*#__PURE__*/(0, _base["default"])(_button.ButtonV2, {
|
|
57
|
+
target: "e111uita10",
|
|
61
58
|
label: "VisualElementButton"
|
|
62
|
-
})("color:", _core.colors.brand.secondary, ";width:100%;height:100%;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAyE0C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
59
|
+
})("color:", _core.colors.brand.secondary, ";width:100%;height:100%;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA2E4C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
63
60
|
var TopicHeaderImage = /*#__PURE__*/(0, _base["default"])("img", {
|
|
64
|
-
target: "
|
|
61
|
+
target: "e111uita9",
|
|
65
62
|
label: "TopicHeaderImage"
|
|
66
|
-
})("border-radius:50%;width:100%;height:100%;object-fit:cover;transition:transform ", _core.animations.durations.fast, ";", VisualElementButton, ":hover &{transform:scale(1.1);opacity:1.2;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA+EmC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
63
|
+
})("border-radius:50%;aspect-ratio:1;width:100%;height:100%;object-fit:cover;transition:transform ", _core.animations.durations.fast, ";", VisualElementButton, ":hover &{transform:scale(1.1);opacity:1.2;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAiFmC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
67
64
|
var ExpandVisualElementButton = /*#__PURE__*/(0, _base["default"])("span", {
|
|
68
|
-
target: "
|
|
65
|
+
target: "e111uita8",
|
|
69
66
|
label: "ExpandVisualElementButton"
|
|
70
|
-
})("position:absolute;right:-10px;bottom:-4px;transition:all ", _core.animations.durations.fast, ";", VisualElementButton, ":hover &{right:10px;}", _core.mq.range({
|
|
67
|
+
})("position:absolute;right:-10px;bottom:-4px;transition:all ", _core.animations.durations.fast, ";svg{width:24px;height:24px;}", VisualElementButton, ":hover &{right:10px;}", _core.mq.range({
|
|
71
68
|
from: _core.breakpoints.mobileWide
|
|
72
|
-
}), "{right:0;bottom:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA2F6C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
69
|
+
}), "{right:0;bottom:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA8F6C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
73
70
|
var TopicHeaderOverlay = /*#__PURE__*/(0, _base["default"])("div", {
|
|
74
|
-
target: "e111uita8",
|
|
75
|
-
label: "TopicHeaderOverlay"
|
|
76
|
-
})("background:black;opacity:0.05;position:absolute;top:0;left:0;bottom:0;right:0;border-radius:50%;transition:opacity ", _core.animations.durations.fast, ";", VisualElementButton, ":hover &{opacity:0.1;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAyGqC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
77
|
-
var _ref4 = process.env.NODE_ENV === "production" ? {
|
|
78
|
-
name: "1s37602-TopicHeading",
|
|
79
|
-
styles: "color:#fff;label:TopicHeading;"
|
|
80
|
-
} : {
|
|
81
|
-
name: "1s37602-TopicHeading",
|
|
82
|
-
styles: "color:#fff;label:TopicHeading;",
|
|
83
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA2IO","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
|
|
84
|
-
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
85
|
-
};
|
|
86
|
-
var TopicHeading = /*#__PURE__*/(0, _base["default"])("h1", {
|
|
87
71
|
target: "e111uita7",
|
|
88
|
-
label: "
|
|
89
|
-
})("margin:", _core.spacing.medium, " 0 0;font-weight:", _core.fonts.weight.bold, ";display:flex;flex-wrap:wrap;align-items:center;", _core.fonts.sizes('24px', '28px'), ";", _core.mq.range({
|
|
90
|
-
|
|
91
|
-
}), "{", _core.fonts.sizes('32px', '28px'), ";margin:40px 0 0;}", _core.mq.range({
|
|
92
|
-
from: _core.breakpoints.desktop
|
|
93
|
-
}), "{margin:50px 0 0;", _core.fonts.sizes('38px', '48px'), ";}", function (props) {
|
|
94
|
-
return props.invertedStyle && _ref4;
|
|
95
|
-
}, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAwH6C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
96
|
-
var StyledHeadingText = /*#__PURE__*/(0, _base["default"])("span", {
|
|
72
|
+
label: "TopicHeaderOverlay"
|
|
73
|
+
})("background:black;opacity:0.05;position:absolute;top:0;left:0;bottom:0;right:0;border-radius:50%;transition:opacity ", _core.animations.durations.fast, ";", VisualElementButton, ":hover &{opacity:0.1;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAgHqC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
74
|
+
var TopicIntroductionWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
97
75
|
target: "e111uita6",
|
|
98
|
-
label: "
|
|
99
|
-
})(process.env.NODE_ENV === "production" ? {
|
|
100
|
-
|
|
101
|
-
styles: "margin-right:28px"
|
|
102
|
-
} : {
|
|
103
|
-
name: "h6gn7j",
|
|
104
|
-
styles: "margin-right:28px",
|
|
105
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAgJqC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
|
|
106
|
-
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
107
|
-
});
|
|
108
|
-
var StyledAdditionalResourceMark = /*#__PURE__*/(0, _base["default"])("span", {
|
|
76
|
+
label: "TopicIntroductionWrapper"
|
|
77
|
+
})("display:flex;gap:", _core.spacing.xsmall, ";justify-content:space-between;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA+H2C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
78
|
+
var HeadingWrapper = /*#__PURE__*/(0, _base["default"])("hgroup", {
|
|
109
79
|
target: "e111uita5",
|
|
110
|
-
label: "
|
|
111
|
-
})("text-align:center;display:inline-block;line-height:18px;width:20px;height:20px;border:1px solid ", _core.colors.brand.dark, ";border-radius:100px;margin-right:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAoJgD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
112
|
-
var StyledAdditionalResource = /*#__PURE__*/(0, _base["default"])("span", {
|
|
113
|
-
target: "e111uita4",
|
|
114
|
-
label: "StyledAdditionalResource"
|
|
115
|
-
})("font-weight:", _core.fonts.weight.semibold, ";", _core.fonts.sizes('12px', '15px'), ";color:", _core.colors.brand.dark, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA8J4C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
116
|
-
var _ref3 = process.env.NODE_ENV === "production" ? {
|
|
117
|
-
name: "9ayvfu-TopicIntroduction",
|
|
118
|
-
styles: "color:#fff;label:TopicIntroduction;"
|
|
119
|
-
} : {
|
|
120
|
-
name: "9ayvfu-TopicIntroduction",
|
|
121
|
-
styles: "color:#fff;label:TopicIntroduction;",
|
|
122
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA6KO","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
|
|
123
|
-
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
124
|
-
};
|
|
80
|
+
label: "HeadingWrapper"
|
|
81
|
+
})("display:flex;flex-wrap:wrap;align-items:center;gap:", _core.spacing.small, ";h1{margin:0;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAqIoC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
125
82
|
var TopicIntroduction = /*#__PURE__*/(0, _base["default"])("div", {
|
|
126
|
-
target: "
|
|
83
|
+
target: "e111uita4",
|
|
127
84
|
label: "TopicIntroduction"
|
|
128
|
-
})("font-weight:", _core.fonts.weight.light, ";max-width:612px;
|
|
85
|
+
})("font-weight:", _core.fonts.weight.light, ";max-width:612px;", _core.mq.range({
|
|
129
86
|
from: _core.breakpoints.tablet
|
|
130
|
-
}), "{", _core.fonts.sizes('22px', '32px'), ";}", function (props) {
|
|
131
|
-
|
|
132
|
-
}, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAoKmD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
133
|
-
var _ref2 = process.env.NODE_ENV === "production" ? {
|
|
87
|
+
}), "{", _core.fonts.sizes('22px', '32px'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA+IoC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
88
|
+
var _ref = process.env.NODE_ENV === "production" ? {
|
|
134
89
|
name: "1pu3m2l-StyledButtonWrapper",
|
|
135
90
|
styles: "button{color:#fff;&:hover,&:focus{color:#fff;}};label:StyledButtonWrapper;"
|
|
136
91
|
} : {
|
|
137
92
|
name: "1pu3m2l-StyledButtonWrapper",
|
|
138
93
|
styles: "button{color:#fff;&:hover,&:focus{color:#fff;}};label:StyledButtonWrapper;",
|
|
139
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAwLO","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
|
|
94
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA6JO","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
|
|
140
95
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
141
96
|
};
|
|
142
97
|
var StyledButtonWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
143
|
-
target: "
|
|
98
|
+
target: "e111uita3",
|
|
144
99
|
label: "StyledButtonWrapper"
|
|
145
100
|
})("margin-top:", _core.spacing.small, ";padding:", _core.spacing.xsmall, " 0 ", _core.spacing.xsmall, " ", _core.spacing.medium, ";border-left:6px solid ", _core.colors.brand.light, ";", function (props) {
|
|
146
|
-
return props.invertedStyle &&
|
|
147
|
-
}, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAkLqD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
148
|
-
var
|
|
149
|
-
|
|
150
|
-
|
|
101
|
+
return props.invertedStyle && _ref;
|
|
102
|
+
}, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAuJqD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
103
|
+
var AdditionalIcon = /*#__PURE__*/(0, _base["default"])("span", {
|
|
104
|
+
target: "e111uita2",
|
|
105
|
+
label: "AdditionalIcon"
|
|
106
|
+
})(process.env.NODE_ENV === "production" ? {
|
|
107
|
+
name: "1qitjsl",
|
|
108
|
+
styles: "padding:1px;border:1px solid currentColor;border-radius:100%;font-size:15px;width:25px;text-align:center"
|
|
151
109
|
} : {
|
|
152
|
-
name: "
|
|
153
|
-
styles: "
|
|
154
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA0MO","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
|
|
110
|
+
name: "1qitjsl",
|
|
111
|
+
styles: "padding:1px;border:1px solid currentColor;border-radius:100%;font-size:15px;width:25px;text-align:center",
|
|
112
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAwKkC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */",
|
|
155
113
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
156
|
-
};
|
|
114
|
+
});
|
|
157
115
|
var StyledContentWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
158
116
|
target: "e111uita1",
|
|
159
117
|
label: "StyledContentWrapper"
|
|
160
|
-
})("padding-top:", _core.spacing.normal, ";margin-top:0;border-left:6px solid ", _core.colors.brand.light, ";", function (props) {
|
|
161
|
-
|
|
162
|
-
}, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAmMsD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
163
|
-
var StyledNavigationBoxWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
118
|
+
})("padding-top:", _core.spacing.normal, ";border-left:6px solid ", _core.colors.brand.light, ";color:", _core.colors.text.primary, ";background-color:", _core.colors.white, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAiLsD","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
119
|
+
var ModalHeader = /*#__PURE__*/(0, _base["default"])(_modal.ModalHeaderV2, {
|
|
164
120
|
target: "e111uita0",
|
|
165
|
-
label: "
|
|
166
|
-
})("padding-top:", _core.spacing.xxsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AA+M6C","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport Modal, { ModalCloseButton, ModalHeader, ModalBody } from '@ndla/modal';\nimport Button from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\ntype FrameProps = {\n  frame?: boolean;\n};\n\nconst Wrapper = styled.section<FrameProps>`\n  ${(props) =>\n    props.frame &&\n    css`\n      ${mq.range({ from: breakpoints.tabletWide })} {\n        padding: 40px 40px;\n        border: 2px solid #d1d6db;\n      }\n      ${mq.range({ from: breakpoints.desktop })} {\n        padding: 40px 80px;\n      }\n      ${mq.range({ from: '1180px' })} {\n        padding: 60px 160px;\n      }\n    `}\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  float: right;\n  margin-left: ${spacing.normal};\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(Button)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicHeading = styled.h1<InvertItProps>`\n  margin: ${spacing.medium} 0 0;\n  font-weight: ${fonts.weight.bold};\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  ${fonts.sizes('24px', '28px')};\n\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('32px', '28px')};\n    margin: 40px 0 0;\n  }\n\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin: 50px 0 0;\n    ${fonts.sizes('38px', '48px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledHeadingText = styled.span`\n  margin-right: 28px;\n`;\n\nconst StyledAdditionalResourceMark = styled.span`\n  text-align: center;\n  display: inline-block;\n  line-height: 18px;\n  width: 20px;\n  height: 20px;\n  border: 1px solid ${colors.brand.dark};\n  border-radius: 100px;\n  margin-right: 7px;\n`;\nconst StyledAdditionalResource = styled.span`\n  font-weight: ${fonts.weight.semibold};\n  ${fonts.sizes('12px', '15px')};\n  color: ${colors.brand.dark};\n`;\n\nconst TopicIntroduction = styled.div<InvertItProps>`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  margin-top: ${spacing.xsmall};\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      color: #fff;\n    `}\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  margin-top: 0;\n  border-left: 6px solid ${colors.brand.light};\n\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      background: #fff;\n    `}\n`;\n\nconst StyledNavigationBoxWrapper = styled.div`\n  padding-top: ${spacing.xxsmall};\n`;\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  return (\n    <Wrapper frame={frame} data-testid=\"nav-topic-about\">\n      {isLoading ? (\n        <Loader />\n      ) : (\n        <>\n          {topic && (\n            <>\n              {topic.image && (\n                <TopicHeaderVisualElementWrapper>\n                  {topic.visualElement ? (\n                    <>\n                      <Modal\n                        label={t('topicPage.imageModal')}\n                        activateButton={\n                          <VisualElementButton\n                            stripped\n                            title={\n                              topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')\n                            }>\n                            <ShowVisualElementWrapper>\n                              <TopicHeaderImage\n                                src={`${topic.image.url}?${makeSrcQueryString(\n                                  800,\n                                  topic.image.crop,\n                                  topic.image.focalPoint,\n                                )}`}\n                                alt={topic.image.alt}\n                              />\n                              <TopicHeaderOverlay />\n                            </ShowVisualElementWrapper>\n                            <ExpandVisualElementButton>\n                              {topic.visualElement.type === 'image' && (\n                                <ExpandTwoArrows style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'video' && (\n                                <PlayCircleFilled style={{ width: '24px', height: '24px' }} />\n                              )}\n                              {topic.visualElement.type === 'other' && (\n                                <CursorClick style={{ width: '24px', height: '24px' }} />\n                              )}\n                            </ExpandVisualElementButton>\n                          </VisualElementButton>\n                        }\n                        animation=\"subtle\"\n                        animationDuration={50}\n                        backgroundColor=\"white\"\n                        size=\"large\">\n                        {(onClose: () => void) => (\n                          <>\n                            <ModalHeader>\n                              <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                            </ModalHeader>\n                            <ModalBody modifier=\"no-side-padding-mobile\">\n                              {topic.visualElement && topic.visualElement.element}\n                            </ModalBody>\n                          </>\n                        )}\n                      </Modal>\n                    </>\n                  ) : (\n                    <TopicHeaderImage\n                      src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                      alt={topic.image.alt}\n                    />\n                  )}\n                </TopicHeaderVisualElementWrapper>\n              )}\n              <TopicHeading invertedStyle={invertedStyle} id={id} tabIndex={-1}>\n                <StyledHeadingText>{topic.title}</StyledHeadingText>\n                {isAdditionalTopic && (\n                  <StyledAdditionalResource>\n                    <StyledAdditionalResourceMark>T</StyledAdditionalResourceMark>\n                    {t('navigation.additionalTopic')}\n                  </StyledAdditionalResource>\n                )}\n              </TopicHeading>\n              {messageBox && <MessageBox>{messageBox}</MessageBox>}\n              <TopicIntroduction invertedStyle={invertedStyle}>\n                {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n              </TopicIntroduction>\n              {onToggleShowContent && (\n                <StyledButtonWrapper invertedStyle={invertedStyle}>\n                  <Button\n                    aria-expanded={!!showContent}\n                    link\n                    onClick={() => {\n                      onToggleShowContent();\n                    }}>\n                    {showContent ? (\n                      <>\n                        {t('navigation.showShorterDescription')} <ChevronUp />\n                      </>\n                    ) : (\n                      <>\n                        {t('navigation.showLongerDescription')} <ChevronDown />\n                      </>\n                    )}\n                  </Button>\n                </StyledButtonWrapper>\n              )}\n              {showContent && <StyledContentWrapper invertedStyle={invertedStyle}>{children}</StyledContentWrapper>}\n              {subTopics && subTopics.length !== 0 && (\n                <StyledNavigationBoxWrapper>\n                  <NavigationBox\n                    colorMode=\"light\"\n                    heading={t('navigation.topics')}\n                    items={subTopics}\n                    onClick={onSubTopicSelected}\n                    invertedStyle={invertedStyle}\n                  />\n                </StyledNavigationBoxWrapper>\n              )}\n              {topic.resources}\n            </>\n          )}\n        </>\n      )}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
167
|
-
var
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
121
|
+
label: "ModalHeader"
|
|
122
|
+
})("padding:", _core.spacing.small, " ", _core.spacing.nsmall, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Topic.tsx"],"names":[],"mappings":"AAwLyC","file":"Topic.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { ReactNode, MouseEvent, ComponentType } from 'react';\nimport styled from '@emotion/styled';\nimport { animations, breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\n\nimport parse from 'html-react-parser';\nimport { ChevronDown, ChevronUp, PlayCircleFilled } from '@ndla/icons/common';\nimport { ModalCloseButton, ModalV2, ModalHeaderV2 } from '@ndla/modal';\nimport { ButtonV2 } from '@ndla/button';\nimport { CursorClick, ExpandTwoArrows } from '@ndla/icons/action';\nimport { css } from '@emotion/react';\nimport { useTranslation } from 'react-i18next';\nimport Loader from './Loader';\nimport { ItemProps } from '../Navigation/NavigationBox';\nimport { NavigationBox } from '../Navigation';\nimport { makeSrcQueryString, ImageCrop, ImageFocalPoint } from '../Image';\nimport { MessageBox } from '../Messages';\n\ntype InvertItProps = {\n  invertedStyle?: boolean;\n};\n\nconst Wrapper = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n`;\n\nconst frameStyle = css`\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: 40px 40px;\n    border: 2px solid ${colors.brand.neutral7};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding: 40px 80px;\n  }\n  ${mq.range({ from: '1180px' })} {\n    padding: 60px 160px;\n  }\n`;\n\nconst _invertedStyle = css`\n  color: ${colors.white};\n`;\n\nconst TopicHeaderVisualElementWrapper = styled.div`\n  position: relative;\n  width: 100px;\n  height: 100px;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    width: 150px;\n    height: 150px;\n  }\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    width: 200px;\n    height: 200px;\n  }\n`;\n\nconst ShowVisualElementWrapper = styled.div`\n  border-radius: 50%;\n  width: 100%;\n  height: 100%;\n  overflow: hidden;\n  aspect-ratio: 1;\n  -webkit-mask-image: -webkit-radial-gradient(white, black); /* Safari fix */\n`;\n\nconst VisualElementButton = styled(ButtonV2)`\n  color: ${colors.brand.secondary};\n  width: 100%;\n  height: 100%;\n`;\n\nconst TopicHeaderImage = styled.img`\n  border-radius: 50%;\n  aspect-ratio: 1;\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  transition: transform ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    transform: scale(1.1);\n    opacity: 1.2;\n  }\n`;\n\nconst ExpandVisualElementButton = styled.span`\n  position: absolute;\n  right: -10px;\n  bottom: -4px;\n  transition: all ${animations.durations.fast};\n  svg {\n    width: 24px;\n    height: 24px;\n  }\n  ${VisualElementButton}:hover & {\n    right: 10px;\n  }\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    right: 0;\n    bottom: 0;\n  }\n`;\n\nconst TopicHeaderOverlay = styled.div`\n  background: black;\n  opacity: 0.05;\n  position: absolute;\n  top: 0;\n  left: 0;\n  bottom: 0;\n  right: 0;\n  border-radius: 50%;\n  transition: opacity ${animations.durations.fast};\n  ${VisualElementButton}:hover & {\n    opacity: 0.1;\n  }\n`;\n\nconst TopicIntroductionWrapper = styled.div`\n  display: flex;\n  gap: ${spacing.xsmall};\n  justify-content: space-between;\n`;\n\nconst HeadingWrapper = styled.hgroup`\n  display: flex;\n  flex-wrap: wrap;\n  align-items: center;\n  gap: ${spacing.small};\n  h1 {\n    margin: 0;\n  }\n`;\n\nconst TopicIntroduction = styled.div`\n  font-weight: ${fonts.weight.light};\n  max-width: 612px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('22px', '32px')};\n  }\n`;\n\nconst StyledButtonWrapper = styled.div<InvertItProps>`\n  margin-top: ${spacing.small};\n  padding: ${spacing.xsmall} 0 ${spacing.xsmall} ${spacing.medium};\n  border-left: 6px solid ${colors.brand.light};\n  ${(props) =>\n    props.invertedStyle &&\n    css`\n      button {\n        color: #fff;\n        &:hover,\n        &:focus {\n          color: #fff;\n        }\n      }\n    `}\n`;\n\nconst AdditionalIcon = styled.span`\n  padding: 1px;\n  border: 1px solid currentColor;\n  border-radius: 100%;\n  font-size: 15px;\n  width: 25px;\n  text-align: center;\n`;\n\nconst StyledContentWrapper = styled.div<InvertItProps>`\n  padding-top: ${spacing.normal};\n  border-left: 6px solid ${colors.brand.light};\n  color: ${colors.text.primary};\n  background-color: ${colors.white};\n`;\n\nconst ModalHeader = styled(ModalHeaderV2)`\n  padding: ${spacing.small} ${spacing.nsmall};\n`;\n\nconst icons: Record<VisualElementProps['type'], ComponentType> = {\n  image: ExpandTwoArrows,\n  video: PlayCircleFilled,\n  other: CursorClick,\n};\n\ntype VisualElementProps = {\n  type: 'image' | 'video' | 'other';\n  element: ReactNode;\n};\n\nexport type TopicProps = {\n  id?: string;\n  topic?: {\n    title: string;\n    introduction: string;\n    image?: {\n      url: string;\n      alt: string;\n      crop?: ImageCrop;\n      focalPoint?: ImageFocalPoint;\n    };\n    visualElement?: VisualElementProps;\n    resources?: ReactNode;\n  };\n  subTopics?: ItemProps[] | null | undefined;\n  onSubTopicSelected?: (event: MouseEvent<HTMLElement>, id?: string) => void;\n  isLoading?: boolean;\n  renderMarkdown?: (text: string) => string;\n  invertedStyle?: boolean;\n  onToggleShowContent?: () => void;\n  showContent?: boolean;\n  isAdditionalTopic?: boolean;\n  frame?: boolean;\n  messageBox?: string;\n  children?: ReactNode;\n};\n\nconst Topic = ({\n  id,\n  topic,\n  subTopics,\n  onSubTopicSelected,\n  isLoading,\n  renderMarkdown,\n  invertedStyle,\n  onToggleShowContent,\n  showContent,\n  isAdditionalTopic,\n  frame,\n  messageBox,\n  children,\n}: TopicProps) => {\n  const { t } = useTranslation();\n  const contentId = `expanded-description-${id}`;\n  const testId = 'nav-topic-about';\n  const VisualElementIcon = topic?.visualElement?.type ? icons[topic.visualElement.type] : null;\n  const wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];\n  if (isLoading || !topic) {\n    return (\n      <Wrapper css={wrapperStyle} data-testid={testId}>\n        {isLoading ? <Loader /> : null}\n      </Wrapper>\n    );\n  }\n\n  return (\n    <Wrapper css={wrapperStyle} data-testid={testId}>\n      <TopicIntroductionWrapper>\n        <div>\n          <HeadingWrapper>\n            <h1 id={id} tabIndex={-1}>\n              {topic.title}\n            </h1>\n            {isAdditionalTopic && (\n              <>\n                <AdditionalIcon aria-hidden=\"true\">T</AdditionalIcon>\n                <span>{t('navigation.additionalTopic')}</span>\n              </>\n            )}\n          </HeadingWrapper>\n          <TopicIntroduction>\n            {renderMarkdown ? parse(renderMarkdown(topic.introduction)) : topic.introduction}\n          </TopicIntroduction>\n        </div>\n        {topic.image && (\n          <TopicHeaderVisualElementWrapper>\n            {topic.visualElement ? (\n              <ModalV2\n                label={t('topicPage.imageModal')}\n                activateButton={\n                  <VisualElementButton\n                    variant=\"stripped\"\n                    title={topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show')}>\n                    <ShowVisualElementWrapper>\n                      <TopicHeaderImage\n                        src={`${topic.image.url}?${makeSrcQueryString(800, topic.image.crop, topic.image.focalPoint)}`}\n                        alt={topic.image.alt}\n                      />\n                      <TopicHeaderOverlay />\n                    </ShowVisualElementWrapper>\n                    <ExpandVisualElementButton>{VisualElementIcon && <VisualElementIcon />}</ExpandVisualElementButton>\n                  </VisualElementButton>\n                }\n                animation=\"subtle\"\n                animationDuration={50}\n                size=\"large\">\n                {(onClose: () => void) => (\n                  <>\n                    <ModalHeader>\n                      <ModalCloseButton onClick={onClose} title={t('modal.closeModal')} />\n                    </ModalHeader>\n                    {topic.visualElement && topic.visualElement.element}\n                  </>\n                )}\n              </ModalV2>\n            ) : (\n              <TopicHeaderImage\n                src={`${topic.image.url}?${makeSrcQueryString(400, topic.image.crop, topic.image.focalPoint)}`}\n                alt={topic.image.alt}\n              />\n            )}\n          </TopicHeaderVisualElementWrapper>\n        )}\n      </TopicIntroductionWrapper>\n      {messageBox && <MessageBox>{messageBox}</MessageBox>}\n      <div>\n        {onToggleShowContent && (\n          <StyledButtonWrapper invertedStyle={invertedStyle}>\n            <ButtonV2\n              aria-expanded={!!showContent}\n              aria-controls={contentId}\n              variant=\"link\"\n              onClick={() => onToggleShowContent()}>\n              {showContent ? (\n                <>\n                  {t('navigation.showShorterDescription')} <ChevronUp />\n                </>\n              ) : (\n                <>\n                  {t('navigation.showLongerDescription')} <ChevronDown />\n                </>\n              )}\n            </ButtonV2>\n          </StyledButtonWrapper>\n        )}\n        {showContent && (\n          <StyledContentWrapper id={contentId} invertedStyle={invertedStyle}>\n            {children}\n          </StyledContentWrapper>\n        )}\n      </div>\n      {subTopics && subTopics.length !== 0 && (\n        <NavigationBox\n          colorMode=\"light\"\n          heading={t('navigation.topics')}\n          items={subTopics}\n          onClick={onSubTopicSelected}\n          invertedStyle={invertedStyle}\n        />\n      )}\n      {topic.resources}\n    </Wrapper>\n  );\n};\nexport default Topic;\n"]} */"));
|
|
123
|
+
var icons = {
|
|
124
|
+
image: _action.ExpandTwoArrows,
|
|
125
|
+
video: _common.PlayCircleFilled,
|
|
126
|
+
other: _action.CursorClick
|
|
127
|
+
};
|
|
128
|
+
var Topic = function Topic(_ref2) {
|
|
129
|
+
var _topic$visualElement;
|
|
130
|
+
var id = _ref2.id,
|
|
131
|
+
topic = _ref2.topic,
|
|
132
|
+
subTopics = _ref2.subTopics,
|
|
133
|
+
onSubTopicSelected = _ref2.onSubTopicSelected,
|
|
134
|
+
isLoading = _ref2.isLoading,
|
|
135
|
+
renderMarkdown = _ref2.renderMarkdown,
|
|
136
|
+
invertedStyle = _ref2.invertedStyle,
|
|
137
|
+
onToggleShowContent = _ref2.onToggleShowContent,
|
|
138
|
+
showContent = _ref2.showContent,
|
|
139
|
+
isAdditionalTopic = _ref2.isAdditionalTopic,
|
|
140
|
+
frame = _ref2.frame,
|
|
141
|
+
messageBox = _ref2.messageBox,
|
|
142
|
+
children = _ref2.children;
|
|
181
143
|
var _useTranslation = (0, _reactI18next.useTranslation)(),
|
|
182
144
|
t = _useTranslation.t;
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
width: '24px',
|
|
204
|
-
height: '24px'
|
|
205
|
-
}
|
|
206
|
-
}), topic.visualElement.type === 'video' && (0, _jsxRuntime.jsx)(_common.PlayCircleFilled, {
|
|
207
|
-
style: {
|
|
208
|
-
width: '24px',
|
|
209
|
-
height: '24px'
|
|
210
|
-
}
|
|
211
|
-
}), topic.visualElement.type === 'other' && (0, _jsxRuntime.jsx)(_action.CursorClick, {
|
|
212
|
-
style: {
|
|
213
|
-
width: '24px',
|
|
214
|
-
height: '24px'
|
|
215
|
-
}
|
|
216
|
-
})]
|
|
217
|
-
})]
|
|
218
|
-
}),
|
|
219
|
-
animation: "subtle",
|
|
220
|
-
animationDuration: 50,
|
|
221
|
-
backgroundColor: "white",
|
|
222
|
-
size: "large",
|
|
223
|
-
children: function children(onClose) {
|
|
224
|
-
return (0, _jsxRuntime.jsxs)(_jsxRuntime.Fragment, {
|
|
225
|
-
children: [(0, _jsxRuntime.jsx)(_modal.ModalHeader, {
|
|
226
|
-
children: (0, _jsxRuntime.jsx)(_modal.ModalCloseButton, {
|
|
227
|
-
onClick: onClose,
|
|
228
|
-
title: t('modal.closeModal')
|
|
229
|
-
})
|
|
230
|
-
}), (0, _jsxRuntime.jsx)(_modal.ModalBody, {
|
|
231
|
-
modifier: "no-side-padding-mobile",
|
|
232
|
-
children: topic.visualElement && topic.visualElement.element
|
|
233
|
-
})]
|
|
234
|
-
});
|
|
235
|
-
}
|
|
236
|
-
})
|
|
237
|
-
}) : (0, _jsxRuntime.jsx)(TopicHeaderImage, {
|
|
238
|
-
src: "".concat(topic.image.url, "?").concat((0, _Image.makeSrcQueryString)(400, topic.image.crop, topic.image.focalPoint)),
|
|
239
|
-
alt: topic.image.alt
|
|
240
|
-
})
|
|
241
|
-
}), (0, _jsxRuntime.jsxs)(TopicHeading, {
|
|
242
|
-
invertedStyle: invertedStyle,
|
|
243
|
-
id: id,
|
|
244
|
-
tabIndex: -1,
|
|
245
|
-
children: [(0, _jsxRuntime.jsx)(StyledHeadingText, {
|
|
145
|
+
var contentId = "expanded-description-".concat(id);
|
|
146
|
+
var testId = 'nav-topic-about';
|
|
147
|
+
var VisualElementIcon = topic !== null && topic !== void 0 && (_topic$visualElement = topic.visualElement) !== null && _topic$visualElement !== void 0 && _topic$visualElement.type ? icons[topic.visualElement.type] : null;
|
|
148
|
+
var wrapperStyle = [frame ? frameStyle : undefined, invertedStyle ? _invertedStyle : undefined];
|
|
149
|
+
if (isLoading || !topic) {
|
|
150
|
+
return (0, _jsxRuntime.jsx)(Wrapper, {
|
|
151
|
+
css: wrapperStyle,
|
|
152
|
+
"data-testid": testId,
|
|
153
|
+
children: isLoading ? (0, _jsxRuntime.jsx)(_Loader["default"], {}) : null
|
|
154
|
+
});
|
|
155
|
+
}
|
|
156
|
+
return (0, _jsxRuntime.jsxs)(Wrapper, {
|
|
157
|
+
css: wrapperStyle,
|
|
158
|
+
"data-testid": testId,
|
|
159
|
+
children: [(0, _jsxRuntime.jsxs)(TopicIntroductionWrapper, {
|
|
160
|
+
children: [(0, _jsxRuntime.jsxs)("div", {
|
|
161
|
+
children: [(0, _jsxRuntime.jsxs)(HeadingWrapper, {
|
|
162
|
+
children: [(0, _jsxRuntime.jsx)("h1", {
|
|
163
|
+
id: id,
|
|
164
|
+
tabIndex: -1,
|
|
246
165
|
children: topic.title
|
|
247
|
-
}), isAdditionalTopic && (0, _jsxRuntime.jsxs)(
|
|
248
|
-
children: [(0, _jsxRuntime.jsx)(
|
|
166
|
+
}), isAdditionalTopic && (0, _jsxRuntime.jsxs)(_jsxRuntime.Fragment, {
|
|
167
|
+
children: [(0, _jsxRuntime.jsx)(AdditionalIcon, {
|
|
168
|
+
"aria-hidden": "true",
|
|
249
169
|
children: "T"
|
|
250
|
-
}),
|
|
170
|
+
}), (0, _jsxRuntime.jsx)("span", {
|
|
171
|
+
children: t('navigation.additionalTopic')
|
|
172
|
+
})]
|
|
251
173
|
})]
|
|
252
|
-
}), messageBox && (0, _jsxRuntime.jsx)(_Messages.MessageBox, {
|
|
253
|
-
children: messageBox
|
|
254
174
|
}), (0, _jsxRuntime.jsx)(TopicIntroduction, {
|
|
255
|
-
invertedStyle: invertedStyle,
|
|
256
175
|
children: renderMarkdown ? (0, _htmlReactParser["default"])(renderMarkdown(topic.introduction)) : topic.introduction
|
|
257
|
-
})
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
})
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
176
|
+
})]
|
|
177
|
+
}), topic.image && (0, _jsxRuntime.jsx)(TopicHeaderVisualElementWrapper, {
|
|
178
|
+
children: topic.visualElement ? (0, _jsxRuntime.jsx)(_modal.ModalV2, {
|
|
179
|
+
label: t('topicPage.imageModal'),
|
|
180
|
+
activateButton: (0, _jsxRuntime.jsxs)(VisualElementButton, {
|
|
181
|
+
variant: "stripped",
|
|
182
|
+
title: topic.visualElement.type === 'image' ? t('image.largeSize') : t('visualElement.show'),
|
|
183
|
+
children: [(0, _jsxRuntime.jsxs)(ShowVisualElementWrapper, {
|
|
184
|
+
children: [(0, _jsxRuntime.jsx)(TopicHeaderImage, {
|
|
185
|
+
src: "".concat(topic.image.url, "?").concat((0, _Image.makeSrcQueryString)(800, topic.image.crop, topic.image.focalPoint)),
|
|
186
|
+
alt: topic.image.alt
|
|
187
|
+
}), (0, _jsxRuntime.jsx)(TopicHeaderOverlay, {})]
|
|
188
|
+
}), (0, _jsxRuntime.jsx)(ExpandVisualElementButton, {
|
|
189
|
+
children: VisualElementIcon && (0, _jsxRuntime.jsx)(VisualElementIcon, {})
|
|
190
|
+
})]
|
|
191
|
+
}),
|
|
192
|
+
animation: "subtle",
|
|
193
|
+
animationDuration: 50,
|
|
194
|
+
size: "large",
|
|
195
|
+
children: function children(onClose) {
|
|
196
|
+
return (0, _jsxRuntime.jsxs)(_jsxRuntime.Fragment, {
|
|
197
|
+
children: [(0, _jsxRuntime.jsx)(ModalHeader, {
|
|
198
|
+
children: (0, _jsxRuntime.jsx)(_modal.ModalCloseButton, {
|
|
199
|
+
onClick: onClose,
|
|
200
|
+
title: t('modal.closeModal')
|
|
201
|
+
})
|
|
202
|
+
}), topic.visualElement && topic.visualElement.element]
|
|
203
|
+
});
|
|
204
|
+
}
|
|
205
|
+
}) : (0, _jsxRuntime.jsx)(TopicHeaderImage, {
|
|
206
|
+
src: "".concat(topic.image.url, "?").concat((0, _Image.makeSrcQueryString)(400, topic.image.crop, topic.image.focalPoint)),
|
|
207
|
+
alt: topic.image.alt
|
|
208
|
+
})
|
|
209
|
+
})]
|
|
210
|
+
}), messageBox && (0, _jsxRuntime.jsx)(_Messages.MessageBox, {
|
|
211
|
+
children: messageBox
|
|
212
|
+
}), (0, _jsxRuntime.jsxs)("div", {
|
|
213
|
+
children: [onToggleShowContent && (0, _jsxRuntime.jsx)(StyledButtonWrapper, {
|
|
214
|
+
invertedStyle: invertedStyle,
|
|
215
|
+
children: (0, _jsxRuntime.jsx)(_button.ButtonV2, {
|
|
216
|
+
"aria-expanded": !!showContent,
|
|
217
|
+
"aria-controls": contentId,
|
|
218
|
+
variant: "link",
|
|
219
|
+
onClick: function onClick() {
|
|
220
|
+
return onToggleShowContent();
|
|
221
|
+
},
|
|
222
|
+
children: showContent ? (0, _jsxRuntime.jsxs)(_jsxRuntime.Fragment, {
|
|
223
|
+
children: [t('navigation.showShorterDescription'), " ", (0, _jsxRuntime.jsx)(_common.ChevronUp, {})]
|
|
224
|
+
}) : (0, _jsxRuntime.jsxs)(_jsxRuntime.Fragment, {
|
|
225
|
+
children: [t('navigation.showLongerDescription'), " ", (0, _jsxRuntime.jsx)(_common.ChevronDown, {})]
|
|
281
226
|
})
|
|
282
|
-
})
|
|
283
|
-
})
|
|
284
|
-
|
|
227
|
+
})
|
|
228
|
+
}), showContent && (0, _jsxRuntime.jsx)(StyledContentWrapper, {
|
|
229
|
+
id: contentId,
|
|
230
|
+
invertedStyle: invertedStyle,
|
|
231
|
+
children: children
|
|
232
|
+
})]
|
|
233
|
+
}), subTopics && subTopics.length !== 0 && (0, _jsxRuntime.jsx)(_Navigation.NavigationBox, {
|
|
234
|
+
colorMode: "light",
|
|
235
|
+
heading: t('navigation.topics'),
|
|
236
|
+
items: subTopics,
|
|
237
|
+
onClick: onSubTopicSelected,
|
|
238
|
+
invertedStyle: invertedStyle
|
|
239
|
+
}), topic.resources]
|
|
285
240
|
});
|
|
286
241
|
};
|
|
287
242
|
var _default = Topic;
|