@ndla/ui 35.1.2 → 36.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (50) hide show
  1. package/es/Embed/RelatedContentEmbed.js +2 -2
  2. package/es/Messages/MessageBox.js +9 -9
  3. package/es/NDLAFilm/FilmContentCard.js +21 -11
  4. package/es/NDLAFilm/FilmMovieList.js +2 -2
  5. package/es/NDLAFilm/FilmSlideshow.js +169 -335
  6. package/es/RelatedArticleList/RelatedArticleV2.js +27 -6
  7. package/es/Search/ToggleSearchButton.js +7 -2
  8. package/es/locale/messages-en.js +1 -0
  9. package/es/locale/messages-nb.js +1 -0
  10. package/es/locale/messages-nn.js +1 -0
  11. package/es/locale/messages-se.js +1 -0
  12. package/es/locale/messages-sma.js +1 -0
  13. package/lib/Embed/RelatedContentEmbed.js +2 -2
  14. package/lib/Messages/MessageBox.js +9 -9
  15. package/lib/NDLAFilm/FilmContentCard.d.ts +4 -2
  16. package/lib/NDLAFilm/FilmContentCard.js +21 -12
  17. package/lib/NDLAFilm/FilmMovieList.d.ts +1 -1
  18. package/lib/NDLAFilm/FilmMovieList.js +2 -2
  19. package/lib/NDLAFilm/FilmSlideshow.d.ts +11 -5
  20. package/lib/NDLAFilm/FilmSlideshow.js +169 -333
  21. package/lib/RelatedArticleList/RelatedArticleV2.d.ts +4 -3
  22. package/lib/RelatedArticleList/RelatedArticleV2.js +35 -14
  23. package/lib/Search/ToggleSearchButton.js +7 -2
  24. package/lib/locale/messages-en.d.ts +1 -0
  25. package/lib/locale/messages-en.js +1 -0
  26. package/lib/locale/messages-nb.d.ts +1 -0
  27. package/lib/locale/messages-nb.js +1 -0
  28. package/lib/locale/messages-nn.d.ts +1 -0
  29. package/lib/locale/messages-nn.js +1 -0
  30. package/lib/locale/messages-se.d.ts +1 -0
  31. package/lib/locale/messages-se.js +1 -0
  32. package/lib/locale/messages-sma.d.ts +1 -0
  33. package/lib/locale/messages-sma.js +1 -0
  34. package/package.json +5 -5
  35. package/src/Embed/AudioEmbed.stories.tsx +226 -0
  36. package/src/Embed/BrightcoveEmbed.stories.tsx +209 -0
  37. package/src/Embed/ConceptEmbed.stories.tsx +190 -0
  38. package/src/Embed/ImageEmbed.stories.tsx +106 -0
  39. package/src/Embed/RelatedContentEmbed.tsx +1 -1
  40. package/src/Messages/MessageBox.tsx +1 -1
  41. package/src/NDLAFilm/FilmContentCard.tsx +11 -9
  42. package/src/NDLAFilm/FilmMovieList.tsx +2 -2
  43. package/src/NDLAFilm/FilmSlideshow.tsx +178 -387
  44. package/src/RelatedArticleList/RelatedArticleV2.tsx +24 -7
  45. package/src/Search/ToggleSearchButton.tsx +5 -1
  46. package/src/locale/messages-en.ts +1 -0
  47. package/src/locale/messages-nb.ts +1 -0
  48. package/src/locale/messages-nn.ts +1 -0
  49. package/src/locale/messages-se.ts +1 -0
  50. package/src/locale/messages-sma.ts +1 -0
@@ -4,17 +4,15 @@ function _typeof(obj) { "@babel/helpers - typeof"; return _typeof = "function" =
4
4
  Object.defineProperty(exports, "__esModule", {
5
5
  value: true
6
6
  });
7
- exports["default"] = void 0;
7
+ exports.slideshowBreakpoints = exports["default"] = void 0;
8
8
  var _base = _interopRequireDefault(require("@emotion/styled/base"));
9
9
  var _react = _interopRequireWildcard(require("react"));
10
- var _reactSwipeable = require("react-swipeable");
11
- var _react2 = require("@emotion/react");
10
+ var _carousel = require("@ndla/carousel");
12
11
  var _core = require("@ndla/core");
13
12
  var _safelink = _interopRequireDefault(require("@ndla/safelink"));
14
- var _icons = require("@ndla/icons");
15
- var _Layout = require("../Layout");
16
- var _NavigationArrow = _interopRequireWildcard(require("./NavigationArrow"));
17
- var _SlideshowIndicator = _interopRequireDefault(require("./SlideshowIndicator"));
13
+ var _button = require("@ndla/button");
14
+ var _common = require("@ndla/icons/common");
15
+ var _FilmContentCard = _interopRequireDefault(require("./FilmContentCard"));
18
16
  var _jsxRuntime = require("@emotion/react/jsx-runtime");
19
17
  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); }
20
18
  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; }
@@ -31,350 +29,188 @@ function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len
31
29
  function _iterableToArrayLimit(arr, i) { var _i = null == arr ? null : "undefined" != typeof Symbol && arr[Symbol.iterator] || arr["@@iterator"]; if (null != _i) { var _s, _e, _x, _r, _arr = [], _n = !0, _d = !1; try { if (_x = (_i = _i.call(arr)).next, 0 === i) { if (Object(_i) !== _i) return; _n = !1; } else for (; !(_n = (_s = _x.call(_i)).done) && (_arr.push(_s.value), _arr.length !== i); _n = !0); } catch (err) { _d = !0, _e = err; } finally { try { if (!_n && null != _i["return"] && (_r = _i["return"](), Object(_r) !== _r)) return; } finally { if (_d) throw _e; } } return _arr; } }
32
30
  function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
33
31
  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)."; } /**
34
- * Copyright (c) 2016-present, NDLA.
32
+ * Copyright (c) 2023-present, NDLA.
35
33
  *
36
34
  * This source code is licensed under the GPLv3 license found in the
37
35
  * LICENSE file in the root directory of this source tree.
38
36
  *
39
37
  */
40
- var SlideLinkWrapper = /*#__PURE__*/(0, _base["default"])("div", {
41
- target: "essc37o7",
42
- label: "SlideLinkWrapper"
43
- })("margin:0 auto;display:flex;justify-content:flex-start;align-items:flex-end;position:absolute;z-index:2;height:100vw;width:100%;", _core.mq.range({
44
- from: _core.breakpoints.mobileWide
45
- }), "{height:100vw;}", _core.mq.range({
46
- from: _core.breakpoints.tablet
47
- }), "{height:75vw;}", _core.mq.range({
48
- from: _core.breakpoints.desktop
49
- }), "{height:55vw;}", _core.mq.range({
50
- from: _core.breakpoints.wide
51
- }), "{height:40vw;}", _core.mq.range({
52
- from: _core.breakpoints.ultraWide
53
- }), "{height:36vw;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AA2BmC","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2016-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, { useCallback, useEffect, useRef, useState } from 'react';\nimport { SwipeDirections, SwipeEventData, useSwipeable } from 'react-swipeable';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { breakpoints, mq, spacing, spacingUnit, fonts, colors } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { Spinner } from '@ndla/icons';\nimport { OneColumn } from '../Layout';\nimport NavigationArrow, { StyledNavigationArrow } from './NavigationArrow';\nimport SlideshowIndicator from './SlideshowIndicator';\nimport { MovieType } from './types';\n\ninterface Props {\n  autoSlide?: boolean;\n  randomStart?: boolean;\n  slideshow: MovieType[];\n  slideInterval?: number;\n}\n\nconst SlideLinkWrapper = styled.div`\n  margin: 0 auto;\n  display: flex;\n  justify-content: flex-start;\n  align-items: flex-end;\n  position: absolute;\n  z-index: 2;\n  height: 100vw;\n  width: 100%;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n`;\n\nconst itemWrapperCSS = css`\n  display: flex;\n  box-shadow: none;\n`;\n\ninterface SlideshowItemProps {\n  fadeOver?: boolean;\n}\n\nconst SlideshowItem = styled.div<SlideshowItemProps>`\n  width: 100vw;\n  height: 100vw;\n  /* aspect ratios */\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n  background-color: '#222';\n  background-size: cover;\n  background-position-x: center;\n  background-position-y: center;\n  border: 0;\n  position: ${(props) => (props.fadeOver ? 'absolute' : 'relative')};\n  animation: ${(props) => props.fadeOver && 'fadeIn 400ms ease'};\n  z-index: ${(props) => props.fadeOver && 1};\n  &:before {\n    content: '';\n    opacity: 0.4;\n    background: #091a2a;\n    top: 0;\n    left: 0;\n    bottom: 0;\n    right: 0;\n    position: absolute;\n    z-index: 1;\n  }\n`;\n\ninterface SlideshowLinkProps {\n  out?: boolean;\n}\n\nconst shouldForwardProp = (p: string) => p !== 'out';\n\nconst SlideshowLink = styled(SafeLink, { shouldForwardProp })<SlideshowLinkProps>`\n  display: flex;\n  box-shadow: none;\n  transition: all 400ms ease;\n  opacity: ${(props) => props.out && 0};\n  animation: ${(props) => !props.out && 'fadeInBottomFixed 600ms ease'};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    padding-bottom: ${spacing.medium};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding-bottom: ${spacing.large};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding-bottom: ${spacingUnit * 3}px;\n  }\n  &:hover {\n    ${() => SlideshowName} {\n      text-decoration: underline;\n      text-decoration-color: white;\n    }\n  }\n`;\n\nconst SlideshowWrapper = styled.section`\n  width: 100%;\n  overflow: hidden;\n  &:hover {\n    ${StyledNavigationArrow} {\n      opacity: 1;\n      transform: translate(0, 0);\n    }\n  }\n`;\n\nconst SlideshowInfo = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n  background-color: rgba(3, 23, 43, 0.7);\n  border-radius: 4px;\n  padding: ${spacing.medium} ${spacing.medium} ${spacing.medium} ${spacing.normal};\n  margin: 0 -20px;\n  width: 100vw;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin: 0;\n    width: 100%;\n    padding: ${spacing.medium} ${spacingUnit * 2}px ${spacing.medium} ${spacing.normal};\n  }\n`;\n\nconst SlideshowName = styled.p`\n  ${fonts.sizes('22px', '30px')};\n  color: ${colors.white};\n  text-shadow: 0 2px 4px rgba(0, 0, 0, 0.25);\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('26px', '30px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('40px', '44px')};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    ${fonts.sizes('48px', '54px')};\n  }\n`;\n\nconst SlideshowDescription = styled.p`\n  color: ${colors.white};\n  margin: 0;\n  padding: 0;\n  ${fonts.sizes('12px', '18px')};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('15px', '20px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('18px', '24px')};\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${fonts.sizes('20px', '32px')};\n  }\n`;\n\nconst EmptySlideshow = styled.div`\n  background: rgba(255, 255, 255, 0.08);\n  margin-bottom: $spacing--large * 4;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  height: 40vw;\n`;\n\nconst defaultTransitionSwipeEnd = 'transform 600ms cubic-bezier(0, 0.76, 0.09, 1)';\nconst defaultTransitionText = 'opacity 600ms ease';\n\nconst renderSlideItem = (slide: MovieType) => (\n  <SlideshowItem\n    key={slide.id}\n    role=\"img\"\n    aria-label={(slide.metaImage && slide.metaImage.alt) || ''}\n    style={{\n      backgroundImage: `url(${(slide.metaImage && slide.metaImage.url) || ''})`,\n    }}\n  />\n);\n\nconst FilmSlideshow = ({ autoSlide = false, slideshow = [], slideInterval = 5000 }: Props) => {\n  const [swipeDistance, setSwipeDistance] = useState(0);\n  const [slideIndex, setSlideIndex] = useState(0);\n  const [slideIndexTarget, setSlideIndexTarget] = useState(0);\n  const [animationComplete, setAnimationComplete] = useState(true);\n  const [swipeDirection, setSwipeDirection] = useState<SwipeDirections | undefined>(undefined);\n  const slideRef = useRef<HTMLDivElement>(null);\n  const slideText = useRef<HTMLDivElement>(null);\n  const timer = useRef<ReturnType<typeof setTimeout> | null>(null);\n\n  const gotoSlide = useCallback((indexTarget: number, useAnimation = false) => {\n    setSwipeDistance(0);\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSlideIndexTarget(indexTarget);\n    setAnimationComplete(!useAnimation);\n  }, []);\n\n  const onChangedSlide = () => {\n    if (!animationComplete) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideIndexTarget * 100}vw))`;\n      }\n\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    } else if (slideIndexTarget === -1) {\n      if (slideRef.current) {\n        // Go to last slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideshow.length * 100}vw))`;\n      }\n\n      setSlideIndex(slideshow.length - 1);\n      setSlideIndexTarget(slideshow.length - 1);\n      setAnimationComplete(true);\n    } else if (slideIndexTarget === slideshow.length) {\n      if (slideRef.current) {\n        // Go to first slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(100vw))`;\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n      setAnimationComplete(true);\n    } else {\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    }\n  };\n\n  const onSwipeEnd = () => {\n    setSwipeDirection(undefined);\n    let slide;\n    if (swipeDistance > 40) {\n      slide = -1;\n    } else if (swipeDistance < -40) {\n      slide = 1;\n    } else {\n      slide = 0;\n    }\n    if (slideRef.current && slideText.current) {\n      slideRef.current.style.transition = defaultTransitionSwipeEnd;\n      slideText.current.style.transition = defaultTransitionText;\n      slideText.current.style.opacity = '1';\n    }\n    setSwipeDistance(0);\n\n    initTimer();\n\n    if (slide !== 0) {\n      setSlideIndex(slideIndex + slide);\n      setSlideIndexTarget(slideIndex + slide);\n    } else {\n      // Reset transfrom\n      if (slideRef.current) {\n        slideRef.current.style.transform = getSlidePosition(slideIndex + slide);\n      }\n    }\n  };\n\n  const onSwipe = (eventData: SwipeEventData) => {\n    if (eventData.initial) {\n      setSwipeDirection(eventData.dir);\n    }\n    const dir = eventData.initial ? eventData.dir : swipeDirection;\n    if (dir === 'Up' || dir === 'Down') {\n      return;\n    }\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSwipeDistance(eventData.deltaX);\n    if (slideRef && slideRef.current) {\n      slideRef.current.style.transition = 'none';\n      slideRef.current.style.transform = getSlidePosition(slideIndexTarget);\n    }\n    const opacityText = 1 - Math.min(100, Math.abs(swipeDistance)) / 100;\n    if (slideText && slideText.current) {\n      slideText.current.style.transition = 'none';\n      slideText.current.style.opacity = opacityText.toString();\n    }\n  };\n\n  const onTransitionEnd = () => {\n    const slideshowLength = slideshow.length;\n    if (slideIndex === -1) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(slideshowLength - 1);\n      }\n      setSlideIndex(slideshowLength - 1);\n      setSlideIndexTarget(slideshowLength - 1);\n    } else if (slideIndex >= slideshowLength) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(0);\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n    }\n  };\n\n  const getSlidePosition = (target: number) => {\n    if (swipeDistance !== 0) {\n      return `translateX(calc(${swipeDistance}px -\n        ${(target + 1) * 100}vw))`;\n    }\n    return `translateX(-${(target + 1) * 100}vw)`;\n  };\n\n  const initTimer = useCallback(() => {\n    if (autoSlide) {\n      timer.current = setTimeout(() => {\n        gotoSlide(slideIndex + 1);\n      }, slideInterval);\n    }\n  }, [autoSlide, gotoSlide, slideInterval, slideIndex]);\n\n  useEffect(() => {\n    initTimer();\n  }, [initTimer]);\n\n  const handlers = useSwipeable({\n    onSwiped: onSwipeEnd,\n    onSwiping: onSwipe,\n    preventScrollOnSwipe: swipeDirection === 'Left' || swipeDirection === 'Right',\n  });\n\n  if (slideshow.length === 0) {\n    return (\n      <div>\n        <EmptySlideshow>\n          <Spinner inverted />\n        </EmptySlideshow>\n      </div>\n    );\n  }\n\n  const slideshowWidth = `${(slideshow.length + 2) * 100}vw`;\n  let activeSlide = slideIndex;\n  if (activeSlide < 0) {\n    activeSlide = slideshow.length - 1;\n  } else if (activeSlide >= slideshow.length) {\n    activeSlide = 0;\n  }\n\n  const backgroundImage = slideshow[activeSlide].metaImage;\n\n  return (\n    <SlideshowWrapper {...handlers}>\n      <>\n        <SlideLinkWrapper>\n          <OneColumn>\n            <SlideshowLink to={slideshow[activeSlide].path} out={!animationComplete}>\n              <SlideshowInfo ref={slideText}>\n                <SlideshowName>{slideshow[activeSlide].title}</SlideshowName>\n                <SlideshowDescription>{slideshow[activeSlide].metaDescription}</SlideshowDescription>\n              </SlideshowInfo>\n            </SlideshowLink>\n          </OneColumn>\n        </SlideLinkWrapper>\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget > 0 ? slideIndexTarget - 1 : slideshow.length - 1}\n          gotoSlide={gotoSlide}\n        />\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget < slideshow.length - 1 ? slideIndexTarget + 1 : 0}\n          gotoSlide={gotoSlide}\n          rightArrow\n        />\n        {!animationComplete && (\n          <SlideshowItem\n            fadeOver\n            role=\"img\"\n            onAnimationEnd={onChangedSlide}\n            style={{\n              backgroundImage: `url(${(backgroundImage && backgroundImage.url) || ''})`,\n            }}\n          />\n        )}\n        <div\n          ref={slideRef}\n          css={itemWrapperCSS}\n          onTransitionEnd={onTransitionEnd}\n          style={{\n            width: slideshowWidth,\n            transform: getSlidePosition(slideIndex),\n          }}\n        >\n          {renderSlideItem(slideshow[slideshow.length - 1])}\n          {slideshow.map(renderSlideItem)}\n          {renderSlideItem(slideshow[0])}\n        </div>\n        <SlideshowIndicator slideshow={slideshow} activeSlide={activeSlide} gotoSlide={gotoSlide} />\n      </>\n    </SlideshowWrapper>\n  );\n};\n\nexport default FilmSlideshow;\n"]} */"));
54
- var itemWrapperCSS = process.env.NODE_ENV === "production" ? {
55
- name: "195eag2-itemWrapperCSS",
56
- styles: "display:flex;box-shadow:none;label:itemWrapperCSS;"
57
- } : {
58
- name: "195eag2-itemWrapperCSS",
59
- styles: "display:flex;box-shadow:none;label:itemWrapperCSS;",
60
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AAqD0B","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2016-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, { useCallback, useEffect, useRef, useState } from 'react';\nimport { SwipeDirections, SwipeEventData, useSwipeable } from 'react-swipeable';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { breakpoints, mq, spacing, spacingUnit, fonts, colors } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { Spinner } from '@ndla/icons';\nimport { OneColumn } from '../Layout';\nimport NavigationArrow, { StyledNavigationArrow } from './NavigationArrow';\nimport SlideshowIndicator from './SlideshowIndicator';\nimport { MovieType } from './types';\n\ninterface Props {\n  autoSlide?: boolean;\n  randomStart?: boolean;\n  slideshow: MovieType[];\n  slideInterval?: number;\n}\n\nconst SlideLinkWrapper = styled.div`\n  margin: 0 auto;\n  display: flex;\n  justify-content: flex-start;\n  align-items: flex-end;\n  position: absolute;\n  z-index: 2;\n  height: 100vw;\n  width: 100%;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n`;\n\nconst itemWrapperCSS = css`\n  display: flex;\n  box-shadow: none;\n`;\n\ninterface SlideshowItemProps {\n  fadeOver?: boolean;\n}\n\nconst SlideshowItem = styled.div<SlideshowItemProps>`\n  width: 100vw;\n  height: 100vw;\n  /* aspect ratios */\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n  background-color: '#222';\n  background-size: cover;\n  background-position-x: center;\n  background-position-y: center;\n  border: 0;\n  position: ${(props) => (props.fadeOver ? 'absolute' : 'relative')};\n  animation: ${(props) => props.fadeOver && 'fadeIn 400ms ease'};\n  z-index: ${(props) => props.fadeOver && 1};\n  &:before {\n    content: '';\n    opacity: 0.4;\n    background: #091a2a;\n    top: 0;\n    left: 0;\n    bottom: 0;\n    right: 0;\n    position: absolute;\n    z-index: 1;\n  }\n`;\n\ninterface SlideshowLinkProps {\n  out?: boolean;\n}\n\nconst shouldForwardProp = (p: string) => p !== 'out';\n\nconst SlideshowLink = styled(SafeLink, { shouldForwardProp })<SlideshowLinkProps>`\n  display: flex;\n  box-shadow: none;\n  transition: all 400ms ease;\n  opacity: ${(props) => props.out && 0};\n  animation: ${(props) => !props.out && 'fadeInBottomFixed 600ms ease'};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    padding-bottom: ${spacing.medium};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding-bottom: ${spacing.large};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding-bottom: ${spacingUnit * 3}px;\n  }\n  &:hover {\n    ${() => SlideshowName} {\n      text-decoration: underline;\n      text-decoration-color: white;\n    }\n  }\n`;\n\nconst SlideshowWrapper = styled.section`\n  width: 100%;\n  overflow: hidden;\n  &:hover {\n    ${StyledNavigationArrow} {\n      opacity: 1;\n      transform: translate(0, 0);\n    }\n  }\n`;\n\nconst SlideshowInfo = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n  background-color: rgba(3, 23, 43, 0.7);\n  border-radius: 4px;\n  padding: ${spacing.medium} ${spacing.medium} ${spacing.medium} ${spacing.normal};\n  margin: 0 -20px;\n  width: 100vw;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin: 0;\n    width: 100%;\n    padding: ${spacing.medium} ${spacingUnit * 2}px ${spacing.medium} ${spacing.normal};\n  }\n`;\n\nconst SlideshowName = styled.p`\n  ${fonts.sizes('22px', '30px')};\n  color: ${colors.white};\n  text-shadow: 0 2px 4px rgba(0, 0, 0, 0.25);\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('26px', '30px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('40px', '44px')};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    ${fonts.sizes('48px', '54px')};\n  }\n`;\n\nconst SlideshowDescription = styled.p`\n  color: ${colors.white};\n  margin: 0;\n  padding: 0;\n  ${fonts.sizes('12px', '18px')};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('15px', '20px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('18px', '24px')};\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${fonts.sizes('20px', '32px')};\n  }\n`;\n\nconst EmptySlideshow = styled.div`\n  background: rgba(255, 255, 255, 0.08);\n  margin-bottom: $spacing--large * 4;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  height: 40vw;\n`;\n\nconst defaultTransitionSwipeEnd = 'transform 600ms cubic-bezier(0, 0.76, 0.09, 1)';\nconst defaultTransitionText = 'opacity 600ms ease';\n\nconst renderSlideItem = (slide: MovieType) => (\n  <SlideshowItem\n    key={slide.id}\n    role=\"img\"\n    aria-label={(slide.metaImage && slide.metaImage.alt) || ''}\n    style={{\n      backgroundImage: `url(${(slide.metaImage && slide.metaImage.url) || ''})`,\n    }}\n  />\n);\n\nconst FilmSlideshow = ({ autoSlide = false, slideshow = [], slideInterval = 5000 }: Props) => {\n  const [swipeDistance, setSwipeDistance] = useState(0);\n  const [slideIndex, setSlideIndex] = useState(0);\n  const [slideIndexTarget, setSlideIndexTarget] = useState(0);\n  const [animationComplete, setAnimationComplete] = useState(true);\n  const [swipeDirection, setSwipeDirection] = useState<SwipeDirections | undefined>(undefined);\n  const slideRef = useRef<HTMLDivElement>(null);\n  const slideText = useRef<HTMLDivElement>(null);\n  const timer = useRef<ReturnType<typeof setTimeout> | null>(null);\n\n  const gotoSlide = useCallback((indexTarget: number, useAnimation = false) => {\n    setSwipeDistance(0);\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSlideIndexTarget(indexTarget);\n    setAnimationComplete(!useAnimation);\n  }, []);\n\n  const onChangedSlide = () => {\n    if (!animationComplete) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideIndexTarget * 100}vw))`;\n      }\n\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    } else if (slideIndexTarget === -1) {\n      if (slideRef.current) {\n        // Go to last slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideshow.length * 100}vw))`;\n      }\n\n      setSlideIndex(slideshow.length - 1);\n      setSlideIndexTarget(slideshow.length - 1);\n      setAnimationComplete(true);\n    } else if (slideIndexTarget === slideshow.length) {\n      if (slideRef.current) {\n        // Go to first slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(100vw))`;\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n      setAnimationComplete(true);\n    } else {\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    }\n  };\n\n  const onSwipeEnd = () => {\n    setSwipeDirection(undefined);\n    let slide;\n    if (swipeDistance > 40) {\n      slide = -1;\n    } else if (swipeDistance < -40) {\n      slide = 1;\n    } else {\n      slide = 0;\n    }\n    if (slideRef.current && slideText.current) {\n      slideRef.current.style.transition = defaultTransitionSwipeEnd;\n      slideText.current.style.transition = defaultTransitionText;\n      slideText.current.style.opacity = '1';\n    }\n    setSwipeDistance(0);\n\n    initTimer();\n\n    if (slide !== 0) {\n      setSlideIndex(slideIndex + slide);\n      setSlideIndexTarget(slideIndex + slide);\n    } else {\n      // Reset transfrom\n      if (slideRef.current) {\n        slideRef.current.style.transform = getSlidePosition(slideIndex + slide);\n      }\n    }\n  };\n\n  const onSwipe = (eventData: SwipeEventData) => {\n    if (eventData.initial) {\n      setSwipeDirection(eventData.dir);\n    }\n    const dir = eventData.initial ? eventData.dir : swipeDirection;\n    if (dir === 'Up' || dir === 'Down') {\n      return;\n    }\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSwipeDistance(eventData.deltaX);\n    if (slideRef && slideRef.current) {\n      slideRef.current.style.transition = 'none';\n      slideRef.current.style.transform = getSlidePosition(slideIndexTarget);\n    }\n    const opacityText = 1 - Math.min(100, Math.abs(swipeDistance)) / 100;\n    if (slideText && slideText.current) {\n      slideText.current.style.transition = 'none';\n      slideText.current.style.opacity = opacityText.toString();\n    }\n  };\n\n  const onTransitionEnd = () => {\n    const slideshowLength = slideshow.length;\n    if (slideIndex === -1) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(slideshowLength - 1);\n      }\n      setSlideIndex(slideshowLength - 1);\n      setSlideIndexTarget(slideshowLength - 1);\n    } else if (slideIndex >= slideshowLength) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(0);\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n    }\n  };\n\n  const getSlidePosition = (target: number) => {\n    if (swipeDistance !== 0) {\n      return `translateX(calc(${swipeDistance}px -\n        ${(target + 1) * 100}vw))`;\n    }\n    return `translateX(-${(target + 1) * 100}vw)`;\n  };\n\n  const initTimer = useCallback(() => {\n    if (autoSlide) {\n      timer.current = setTimeout(() => {\n        gotoSlide(slideIndex + 1);\n      }, slideInterval);\n    }\n  }, [autoSlide, gotoSlide, slideInterval, slideIndex]);\n\n  useEffect(() => {\n    initTimer();\n  }, [initTimer]);\n\n  const handlers = useSwipeable({\n    onSwiped: onSwipeEnd,\n    onSwiping: onSwipe,\n    preventScrollOnSwipe: swipeDirection === 'Left' || swipeDirection === 'Right',\n  });\n\n  if (slideshow.length === 0) {\n    return (\n      <div>\n        <EmptySlideshow>\n          <Spinner inverted />\n        </EmptySlideshow>\n      </div>\n    );\n  }\n\n  const slideshowWidth = `${(slideshow.length + 2) * 100}vw`;\n  let activeSlide = slideIndex;\n  if (activeSlide < 0) {\n    activeSlide = slideshow.length - 1;\n  } else if (activeSlide >= slideshow.length) {\n    activeSlide = 0;\n  }\n\n  const backgroundImage = slideshow[activeSlide].metaImage;\n\n  return (\n    <SlideshowWrapper {...handlers}>\n      <>\n        <SlideLinkWrapper>\n          <OneColumn>\n            <SlideshowLink to={slideshow[activeSlide].path} out={!animationComplete}>\n              <SlideshowInfo ref={slideText}>\n                <SlideshowName>{slideshow[activeSlide].title}</SlideshowName>\n                <SlideshowDescription>{slideshow[activeSlide].metaDescription}</SlideshowDescription>\n              </SlideshowInfo>\n            </SlideshowLink>\n          </OneColumn>\n        </SlideLinkWrapper>\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget > 0 ? slideIndexTarget - 1 : slideshow.length - 1}\n          gotoSlide={gotoSlide}\n        />\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget < slideshow.length - 1 ? slideIndexTarget + 1 : 0}\n          gotoSlide={gotoSlide}\n          rightArrow\n        />\n        {!animationComplete && (\n          <SlideshowItem\n            fadeOver\n            role=\"img\"\n            onAnimationEnd={onChangedSlide}\n            style={{\n              backgroundImage: `url(${(backgroundImage && backgroundImage.url) || ''})`,\n            }}\n          />\n        )}\n        <div\n          ref={slideRef}\n          css={itemWrapperCSS}\n          onTransitionEnd={onTransitionEnd}\n          style={{\n            width: slideshowWidth,\n            transform: getSlidePosition(slideIndex),\n          }}\n        >\n          {renderSlideItem(slideshow[slideshow.length - 1])}\n          {slideshow.map(renderSlideItem)}\n          {renderSlideItem(slideshow[0])}\n        </div>\n        <SlideshowIndicator slideshow={slideshow} activeSlide={activeSlide} gotoSlide={gotoSlide} />\n      </>\n    </SlideshowWrapper>\n  );\n};\n\nexport default FilmSlideshow;\n"]} */",
61
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
62
- };
63
- var SlideshowItem = /*#__PURE__*/(0, _base["default"])("div", {
38
+ var slideshowBreakpoints = [{
39
+ until: 'mobileWide',
40
+ columnsPrSlide: 2,
41
+ distanceBetweenItems: _core.spacingUnit / 2,
42
+ margin: _core.spacingUnit,
43
+ arrowOffset: 13
44
+ }, {
45
+ until: 'tabletWide',
46
+ columnsPrSlide: 3,
47
+ distanceBetweenItems: _core.spacingUnit / 2,
48
+ margin: _core.spacingUnit,
49
+ arrowOffset: 13
50
+ }, {
51
+ until: 'desktop',
52
+ columnsPrSlide: 3,
53
+ distanceBetweenItems: _core.spacingUnit,
54
+ margin: _core.spacingUnit * 2,
55
+ arrowOffset: 0
56
+ }, {
57
+ until: 'wide',
58
+ columnsPrSlide: 3,
59
+ distanceBetweenItems: _core.spacingUnit,
60
+ margin: _core.spacingUnit * 2,
61
+ arrowOffset: 0
62
+ }, {
63
+ until: 'ultraWide',
64
+ columnsPrSlide: 3,
65
+ distanceBetweenItems: _core.spacingUnit,
66
+ margin: _core.spacingUnit * 3.5,
67
+ arrowOffset: 0
68
+ }, {
69
+ columnsPrSlide: 3,
70
+ distanceBetweenItems: _core.spacingUnit,
71
+ margin: _core.spacingUnit * 3.5,
72
+ arrowOffset: 0
73
+ }];
74
+ exports.slideshowBreakpoints = slideshowBreakpoints;
75
+ var SlideInfoWrapper = /*#__PURE__*/(0, _base["default"])("div", {
64
76
  target: "essc37o6",
65
- label: "SlideshowItem"
66
- })("width:100vw;height:100vw;", _core.mq.range({
67
- from: _core.breakpoints.mobileWide
68
- }), "{height:100vw;}", _core.mq.range({
69
- from: _core.breakpoints.tablet
70
- }), "{height:75vw;}", _core.mq.range({
71
- from: _core.breakpoints.desktop
72
- }), "{height:55vw;}", _core.mq.range({
73
- from: _core.breakpoints.wide
74
- }), "{height:40vw;}", _core.mq.range({
75
- from: _core.breakpoints.ultraWide
76
- }), "{height:36vw;}background-color:'#222';background-size:cover;background-position-x:center;background-position-y:center;border:0;position:", function (props) {
77
- return props.fadeOver ? 'absolute' : 'relative';
78
- }, ";animation:", function (props) {
79
- return props.fadeOver && 'fadeIn 400ms ease';
80
- }, ";z-index:", function (props) {
81
- return props.fadeOver && 1;
82
- }, ";&:before{content:'';opacity:0.4;background:#091a2a;top:0;left:0;bottom:0;right:0;position:absolute;z-index:1;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AA8DoD","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2016-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, { useCallback, useEffect, useRef, useState } from 'react';\nimport { SwipeDirections, SwipeEventData, useSwipeable } from 'react-swipeable';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { breakpoints, mq, spacing, spacingUnit, fonts, colors } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { Spinner } from '@ndla/icons';\nimport { OneColumn } from '../Layout';\nimport NavigationArrow, { StyledNavigationArrow } from './NavigationArrow';\nimport SlideshowIndicator from './SlideshowIndicator';\nimport { MovieType } from './types';\n\ninterface Props {\n  autoSlide?: boolean;\n  randomStart?: boolean;\n  slideshow: MovieType[];\n  slideInterval?: number;\n}\n\nconst SlideLinkWrapper = styled.div`\n  margin: 0 auto;\n  display: flex;\n  justify-content: flex-start;\n  align-items: flex-end;\n  position: absolute;\n  z-index: 2;\n  height: 100vw;\n  width: 100%;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n`;\n\nconst itemWrapperCSS = css`\n  display: flex;\n  box-shadow: none;\n`;\n\ninterface SlideshowItemProps {\n  fadeOver?: boolean;\n}\n\nconst SlideshowItem = styled.div<SlideshowItemProps>`\n  width: 100vw;\n  height: 100vw;\n  /* aspect ratios */\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n  background-color: '#222';\n  background-size: cover;\n  background-position-x: center;\n  background-position-y: center;\n  border: 0;\n  position: ${(props) => (props.fadeOver ? 'absolute' : 'relative')};\n  animation: ${(props) => props.fadeOver && 'fadeIn 400ms ease'};\n  z-index: ${(props) => props.fadeOver && 1};\n  &:before {\n    content: '';\n    opacity: 0.4;\n    background: #091a2a;\n    top: 0;\n    left: 0;\n    bottom: 0;\n    right: 0;\n    position: absolute;\n    z-index: 1;\n  }\n`;\n\ninterface SlideshowLinkProps {\n  out?: boolean;\n}\n\nconst shouldForwardProp = (p: string) => p !== 'out';\n\nconst SlideshowLink = styled(SafeLink, { shouldForwardProp })<SlideshowLinkProps>`\n  display: flex;\n  box-shadow: none;\n  transition: all 400ms ease;\n  opacity: ${(props) => props.out && 0};\n  animation: ${(props) => !props.out && 'fadeInBottomFixed 600ms ease'};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    padding-bottom: ${spacing.medium};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding-bottom: ${spacing.large};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding-bottom: ${spacingUnit * 3}px;\n  }\n  &:hover {\n    ${() => SlideshowName} {\n      text-decoration: underline;\n      text-decoration-color: white;\n    }\n  }\n`;\n\nconst SlideshowWrapper = styled.section`\n  width: 100%;\n  overflow: hidden;\n  &:hover {\n    ${StyledNavigationArrow} {\n      opacity: 1;\n      transform: translate(0, 0);\n    }\n  }\n`;\n\nconst SlideshowInfo = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n  background-color: rgba(3, 23, 43, 0.7);\n  border-radius: 4px;\n  padding: ${spacing.medium} ${spacing.medium} ${spacing.medium} ${spacing.normal};\n  margin: 0 -20px;\n  width: 100vw;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin: 0;\n    width: 100%;\n    padding: ${spacing.medium} ${spacingUnit * 2}px ${spacing.medium} ${spacing.normal};\n  }\n`;\n\nconst SlideshowName = styled.p`\n  ${fonts.sizes('22px', '30px')};\n  color: ${colors.white};\n  text-shadow: 0 2px 4px rgba(0, 0, 0, 0.25);\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('26px', '30px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('40px', '44px')};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    ${fonts.sizes('48px', '54px')};\n  }\n`;\n\nconst SlideshowDescription = styled.p`\n  color: ${colors.white};\n  margin: 0;\n  padding: 0;\n  ${fonts.sizes('12px', '18px')};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('15px', '20px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('18px', '24px')};\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${fonts.sizes('20px', '32px')};\n  }\n`;\n\nconst EmptySlideshow = styled.div`\n  background: rgba(255, 255, 255, 0.08);\n  margin-bottom: $spacing--large * 4;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  height: 40vw;\n`;\n\nconst defaultTransitionSwipeEnd = 'transform 600ms cubic-bezier(0, 0.76, 0.09, 1)';\nconst defaultTransitionText = 'opacity 600ms ease';\n\nconst renderSlideItem = (slide: MovieType) => (\n  <SlideshowItem\n    key={slide.id}\n    role=\"img\"\n    aria-label={(slide.metaImage && slide.metaImage.alt) || ''}\n    style={{\n      backgroundImage: `url(${(slide.metaImage && slide.metaImage.url) || ''})`,\n    }}\n  />\n);\n\nconst FilmSlideshow = ({ autoSlide = false, slideshow = [], slideInterval = 5000 }: Props) => {\n  const [swipeDistance, setSwipeDistance] = useState(0);\n  const [slideIndex, setSlideIndex] = useState(0);\n  const [slideIndexTarget, setSlideIndexTarget] = useState(0);\n  const [animationComplete, setAnimationComplete] = useState(true);\n  const [swipeDirection, setSwipeDirection] = useState<SwipeDirections | undefined>(undefined);\n  const slideRef = useRef<HTMLDivElement>(null);\n  const slideText = useRef<HTMLDivElement>(null);\n  const timer = useRef<ReturnType<typeof setTimeout> | null>(null);\n\n  const gotoSlide = useCallback((indexTarget: number, useAnimation = false) => {\n    setSwipeDistance(0);\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSlideIndexTarget(indexTarget);\n    setAnimationComplete(!useAnimation);\n  }, []);\n\n  const onChangedSlide = () => {\n    if (!animationComplete) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideIndexTarget * 100}vw))`;\n      }\n\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    } else if (slideIndexTarget === -1) {\n      if (slideRef.current) {\n        // Go to last slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideshow.length * 100}vw))`;\n      }\n\n      setSlideIndex(slideshow.length - 1);\n      setSlideIndexTarget(slideshow.length - 1);\n      setAnimationComplete(true);\n    } else if (slideIndexTarget === slideshow.length) {\n      if (slideRef.current) {\n        // Go to first slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(100vw))`;\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n      setAnimationComplete(true);\n    } else {\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    }\n  };\n\n  const onSwipeEnd = () => {\n    setSwipeDirection(undefined);\n    let slide;\n    if (swipeDistance > 40) {\n      slide = -1;\n    } else if (swipeDistance < -40) {\n      slide = 1;\n    } else {\n      slide = 0;\n    }\n    if (slideRef.current && slideText.current) {\n      slideRef.current.style.transition = defaultTransitionSwipeEnd;\n      slideText.current.style.transition = defaultTransitionText;\n      slideText.current.style.opacity = '1';\n    }\n    setSwipeDistance(0);\n\n    initTimer();\n\n    if (slide !== 0) {\n      setSlideIndex(slideIndex + slide);\n      setSlideIndexTarget(slideIndex + slide);\n    } else {\n      // Reset transfrom\n      if (slideRef.current) {\n        slideRef.current.style.transform = getSlidePosition(slideIndex + slide);\n      }\n    }\n  };\n\n  const onSwipe = (eventData: SwipeEventData) => {\n    if (eventData.initial) {\n      setSwipeDirection(eventData.dir);\n    }\n    const dir = eventData.initial ? eventData.dir : swipeDirection;\n    if (dir === 'Up' || dir === 'Down') {\n      return;\n    }\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSwipeDistance(eventData.deltaX);\n    if (slideRef && slideRef.current) {\n      slideRef.current.style.transition = 'none';\n      slideRef.current.style.transform = getSlidePosition(slideIndexTarget);\n    }\n    const opacityText = 1 - Math.min(100, Math.abs(swipeDistance)) / 100;\n    if (slideText && slideText.current) {\n      slideText.current.style.transition = 'none';\n      slideText.current.style.opacity = opacityText.toString();\n    }\n  };\n\n  const onTransitionEnd = () => {\n    const slideshowLength = slideshow.length;\n    if (slideIndex === -1) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(slideshowLength - 1);\n      }\n      setSlideIndex(slideshowLength - 1);\n      setSlideIndexTarget(slideshowLength - 1);\n    } else if (slideIndex >= slideshowLength) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(0);\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n    }\n  };\n\n  const getSlidePosition = (target: number) => {\n    if (swipeDistance !== 0) {\n      return `translateX(calc(${swipeDistance}px -\n        ${(target + 1) * 100}vw))`;\n    }\n    return `translateX(-${(target + 1) * 100}vw)`;\n  };\n\n  const initTimer = useCallback(() => {\n    if (autoSlide) {\n      timer.current = setTimeout(() => {\n        gotoSlide(slideIndex + 1);\n      }, slideInterval);\n    }\n  }, [autoSlide, gotoSlide, slideInterval, slideIndex]);\n\n  useEffect(() => {\n    initTimer();\n  }, [initTimer]);\n\n  const handlers = useSwipeable({\n    onSwiped: onSwipeEnd,\n    onSwiping: onSwipe,\n    preventScrollOnSwipe: swipeDirection === 'Left' || swipeDirection === 'Right',\n  });\n\n  if (slideshow.length === 0) {\n    return (\n      <div>\n        <EmptySlideshow>\n          <Spinner inverted />\n        </EmptySlideshow>\n      </div>\n    );\n  }\n\n  const slideshowWidth = `${(slideshow.length + 2) * 100}vw`;\n  let activeSlide = slideIndex;\n  if (activeSlide < 0) {\n    activeSlide = slideshow.length - 1;\n  } else if (activeSlide >= slideshow.length) {\n    activeSlide = 0;\n  }\n\n  const backgroundImage = slideshow[activeSlide].metaImage;\n\n  return (\n    <SlideshowWrapper {...handlers}>\n      <>\n        <SlideLinkWrapper>\n          <OneColumn>\n            <SlideshowLink to={slideshow[activeSlide].path} out={!animationComplete}>\n              <SlideshowInfo ref={slideText}>\n                <SlideshowName>{slideshow[activeSlide].title}</SlideshowName>\n                <SlideshowDescription>{slideshow[activeSlide].metaDescription}</SlideshowDescription>\n              </SlideshowInfo>\n            </SlideshowLink>\n          </OneColumn>\n        </SlideLinkWrapper>\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget > 0 ? slideIndexTarget - 1 : slideshow.length - 1}\n          gotoSlide={gotoSlide}\n        />\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget < slideshow.length - 1 ? slideIndexTarget + 1 : 0}\n          gotoSlide={gotoSlide}\n          rightArrow\n        />\n        {!animationComplete && (\n          <SlideshowItem\n            fadeOver\n            role=\"img\"\n            onAnimationEnd={onChangedSlide}\n            style={{\n              backgroundImage: `url(${(backgroundImage && backgroundImage.url) || ''})`,\n            }}\n          />\n        )}\n        <div\n          ref={slideRef}\n          css={itemWrapperCSS}\n          onTransitionEnd={onTransitionEnd}\n          style={{\n            width: slideshowWidth,\n            transform: getSlidePosition(slideIndex),\n          }}\n        >\n          {renderSlideItem(slideshow[slideshow.length - 1])}\n          {slideshow.map(renderSlideItem)}\n          {renderSlideItem(slideshow[0])}\n        </div>\n        <SlideshowIndicator slideshow={slideshow} activeSlide={activeSlide} gotoSlide={gotoSlide} />\n      </>\n    </SlideshowWrapper>\n  );\n};\n\nexport default FilmSlideshow;\n"]} */"));
83
- var shouldForwardProp = function shouldForwardProp(p) {
84
- return p !== 'out';
85
- };
86
- var SlideshowLink = /*#__PURE__*/(0, _base["default"])(_safelink["default"], {
87
- shouldForwardProp: shouldForwardProp,
77
+ label: "SlideInfoWrapper"
78
+ })("position:absolute;color:", _core.colors.white, ";max-width:40%;min-width:40%;top:40%;right:5%;", _core.mq.range({
79
+ until: _core.breakpoints.desktop
80
+ }), "{top:30%;max-width:60%;min-width:60%;}", _core.mq.range({
81
+ until: _core.breakpoints.tablet
82
+ }), "{max-width:90%;min-width:90%;left:5%;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AAyEmC","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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, { useCallback, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Carousel, CarouselAutosize } from '@ndla/carousel';\nimport { breakpoints, colors, misc, mq, spacing, spacingUnit } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { IconButtonV2 } from '@ndla/button';\nimport { ChevronLeft, ChevronRight } from '@ndla/icons/common';\nimport FilmContentCard from './FilmContentCard';\nimport { MovieType } from './types';\n\nexport const slideshowBreakpoints: {\n  until?: keyof typeof breakpoints;\n  columnsPrSlide: number;\n  distanceBetweenItems: number;\n  arrowOffset: number;\n  margin?: number;\n  maxColumnWidth?: number;\n}[] = [\n  {\n    until: 'mobileWide',\n    columnsPrSlide: 2,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'tabletWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'desktop',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'wide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'ultraWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n  {\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n];\n\ninterface Props {\n  slideshow: MovieType[];\n}\n\nconst SlideInfoWrapper = styled.div`\n  position: absolute;\n  color: ${colors.white};\n  max-width: 40%;\n  min-width: 40%;\n  top: 40%;\n  right: 5%;\n  ${mq.range({ until: breakpoints.desktop })} {\n    top: 30%;\n    max-width: 60%;\n    min-width: 60%;\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    max-width: 90%;\n    min-width: 90%;\n    left: 5%;\n  }\n`;\n\nconst StyledSafeLink = styled(SafeLink)`\n  position: relative;\n  display: block;\n  box-shadow: none;\n`;\n\nconst InfoWrapper = styled.div`\n  padding: ${spacing.normal};\n  border-radius: ${misc.borderRadius};\n  border: 0.5px solid ${colors.brand.primary};\n  background-color: rgba(11, 29, 45, 0.8);\n  h3 {\n    margin: 0px;\n  }\n`;\n\nconst StyledImg = styled.img`\n  max-height: 600px;\n  object-position: top;\n  width: 100%;\n  aspect-ratio: 16/9;\n  ${mq.range({ until: breakpoints.tablet })} {\n    min-height: 440px;\n    max-height: 440px;\n  }\n  object-fit: cover;\n`;\n\nconst CarouselContainer = styled.div`\n  margin-top: -50px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    margin-top: -70px;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin-top: -150px;\n  }\n`;\n\ninterface StyledFilmContentCardProps {\n  current?: boolean;\n}\n\nconst SlideshowButton = styled(IconButtonV2)`\n  margin-top: ${spacing.normal};\n`;\n\nconst shouldForwardProp = (p: string) => p !== 'current';\n\nconst StyledFilmContentCard = styled(FilmContentCard, { shouldForwardProp })<StyledFilmContentCardProps>`\n  margin-bottom: 2%;\n  transform: ${(p) => (p.current ? 'translateY(0%)' : 'translateY(10%)')};\n  transition: all 200ms;\n`;\n\nconst FilmSlideshow = ({ slideshow }: Props) => {\n  const [currentSlide, setCurrentSlide] = useState(slideshow[0]);\n\n  return (\n    <CarouselAutosize breakpoints={slideshowBreakpoints} itemsLength={slideshow.length}>\n      {(autoSizedProps) => (\n        <section>\n          <StyledSafeLink to={currentSlide.path} tabIndex={-1} aria-hidden>\n            <StyledImg src={currentSlide.metaImage?.url ?? ''} alt={currentSlide.metaImage?.alt ?? ''} />\n            <SlideInfoWrapper>\n              <InfoWrapper>\n                <h3>{currentSlide.title}</h3>\n                <span id=\"currentMovieDescription\">{currentSlide.metaDescription}</span>\n              </InfoWrapper>\n            </SlideInfoWrapper>\n          </StyledSafeLink>\n          <CarouselContainer>\n            <Carousel\n              leftButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronLeft />\n                </SlideshowButton>\n              }\n              rightButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronRight />\n                </SlideshowButton>\n              }\n              items={slideshow.map((movie) => (\n                <FilmCard\n                  key={movie.id}\n                  current={movie.id === currentSlide.id}\n                  movie={movie}\n                  columnWidth={autoSizedProps.columnWidth}\n                  setCurrentSlide={() => setCurrentSlide(movie)}\n                />\n              ))}\n              {...autoSizedProps}\n            />\n          </CarouselContainer>\n        </section>\n      )}\n    </CarouselAutosize>\n  );\n};\n\ninterface FilmCardProps {\n  setCurrentSlide: () => void;\n  movie: MovieType;\n  current: boolean;\n  columnWidth: number;\n}\n\nconst FilmCard = ({ setCurrentSlide, movie, current, columnWidth }: FilmCardProps) => {\n  const [hoverCallback, setHoverCallback] = useState<ReturnType<typeof setTimeout> | undefined>(undefined);\n\n  const onHover = useCallback(() => {\n    const timeout = setTimeout(() => setCurrentSlide(), 500);\n    setHoverCallback(timeout);\n  }, [setCurrentSlide]);\n\n  return (\n    <StyledFilmContentCard\n      onMouseEnter={onHover}\n      onMouseLeave={() => {\n        if (hoverCallback) {\n          clearTimeout(hoverCallback);\n          setHoverCallback(undefined);\n        }\n      }}\n      onFocus={() => setCurrentSlide()}\n      current={current}\n      aria-describedby={'currentMovieDescription'}\n      key={movie.id}\n      movie={movie}\n      columnWidth={columnWidth}\n      resourceTypes={[]}\n    />\n  );\n};\n\nexport default FilmSlideshow;\n"]} */"));
83
+ var StyledSafeLink = /*#__PURE__*/(0, _base["default"])(_safelink["default"], {
88
84
  target: "essc37o5",
89
- label: "SlideshowLink"
90
- })("display:flex;box-shadow:none;transition:all 400ms ease;opacity:", function (props) {
91
- return props.out && 0;
92
- }, ";animation:", function (props) {
93
- return !props.out && 'fadeInBottomFixed 600ms ease';
94
- }, ";", _core.mq.range({
95
- from: _core.breakpoints.mobileWide
96
- }), "{padding-bottom:", _core.spacing.medium, ";}", _core.mq.range({
97
- from: _core.breakpoints.tablet
98
- }), "{padding-bottom:", _core.spacing.large, ";}", _core.mq.range({
99
- from: _core.breakpoints.desktop
100
- }), "{padding-bottom:", _core.spacingUnit * 3, "px;}&:hover{", function () {
101
- return SlideshowName;
102
- }, "{text-decoration:underline;text-decoration-color:white;}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AA4GiF","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2016-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, { useCallback, useEffect, useRef, useState } from 'react';\nimport { SwipeDirections, SwipeEventData, useSwipeable } from 'react-swipeable';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { breakpoints, mq, spacing, spacingUnit, fonts, colors } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { Spinner } from '@ndla/icons';\nimport { OneColumn } from '../Layout';\nimport NavigationArrow, { StyledNavigationArrow } from './NavigationArrow';\nimport SlideshowIndicator from './SlideshowIndicator';\nimport { MovieType } from './types';\n\ninterface Props {\n  autoSlide?: boolean;\n  randomStart?: boolean;\n  slideshow: MovieType[];\n  slideInterval?: number;\n}\n\nconst SlideLinkWrapper = styled.div`\n  margin: 0 auto;\n  display: flex;\n  justify-content: flex-start;\n  align-items: flex-end;\n  position: absolute;\n  z-index: 2;\n  height: 100vw;\n  width: 100%;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n`;\n\nconst itemWrapperCSS = css`\n  display: flex;\n  box-shadow: none;\n`;\n\ninterface SlideshowItemProps {\n  fadeOver?: boolean;\n}\n\nconst SlideshowItem = styled.div<SlideshowItemProps>`\n  width: 100vw;\n  height: 100vw;\n  /* aspect ratios */\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n  background-color: '#222';\n  background-size: cover;\n  background-position-x: center;\n  background-position-y: center;\n  border: 0;\n  position: ${(props) => (props.fadeOver ? 'absolute' : 'relative')};\n  animation: ${(props) => props.fadeOver && 'fadeIn 400ms ease'};\n  z-index: ${(props) => props.fadeOver && 1};\n  &:before {\n    content: '';\n    opacity: 0.4;\n    background: #091a2a;\n    top: 0;\n    left: 0;\n    bottom: 0;\n    right: 0;\n    position: absolute;\n    z-index: 1;\n  }\n`;\n\ninterface SlideshowLinkProps {\n  out?: boolean;\n}\n\nconst shouldForwardProp = (p: string) => p !== 'out';\n\nconst SlideshowLink = styled(SafeLink, { shouldForwardProp })<SlideshowLinkProps>`\n  display: flex;\n  box-shadow: none;\n  transition: all 400ms ease;\n  opacity: ${(props) => props.out && 0};\n  animation: ${(props) => !props.out && 'fadeInBottomFixed 600ms ease'};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    padding-bottom: ${spacing.medium};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding-bottom: ${spacing.large};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding-bottom: ${spacingUnit * 3}px;\n  }\n  &:hover {\n    ${() => SlideshowName} {\n      text-decoration: underline;\n      text-decoration-color: white;\n    }\n  }\n`;\n\nconst SlideshowWrapper = styled.section`\n  width: 100%;\n  overflow: hidden;\n  &:hover {\n    ${StyledNavigationArrow} {\n      opacity: 1;\n      transform: translate(0, 0);\n    }\n  }\n`;\n\nconst SlideshowInfo = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n  background-color: rgba(3, 23, 43, 0.7);\n  border-radius: 4px;\n  padding: ${spacing.medium} ${spacing.medium} ${spacing.medium} ${spacing.normal};\n  margin: 0 -20px;\n  width: 100vw;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin: 0;\n    width: 100%;\n    padding: ${spacing.medium} ${spacingUnit * 2}px ${spacing.medium} ${spacing.normal};\n  }\n`;\n\nconst SlideshowName = styled.p`\n  ${fonts.sizes('22px', '30px')};\n  color: ${colors.white};\n  text-shadow: 0 2px 4px rgba(0, 0, 0, 0.25);\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('26px', '30px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('40px', '44px')};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    ${fonts.sizes('48px', '54px')};\n  }\n`;\n\nconst SlideshowDescription = styled.p`\n  color: ${colors.white};\n  margin: 0;\n  padding: 0;\n  ${fonts.sizes('12px', '18px')};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('15px', '20px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('18px', '24px')};\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${fonts.sizes('20px', '32px')};\n  }\n`;\n\nconst EmptySlideshow = styled.div`\n  background: rgba(255, 255, 255, 0.08);\n  margin-bottom: $spacing--large * 4;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  height: 40vw;\n`;\n\nconst defaultTransitionSwipeEnd = 'transform 600ms cubic-bezier(0, 0.76, 0.09, 1)';\nconst defaultTransitionText = 'opacity 600ms ease';\n\nconst renderSlideItem = (slide: MovieType) => (\n  <SlideshowItem\n    key={slide.id}\n    role=\"img\"\n    aria-label={(slide.metaImage && slide.metaImage.alt) || ''}\n    style={{\n      backgroundImage: `url(${(slide.metaImage && slide.metaImage.url) || ''})`,\n    }}\n  />\n);\n\nconst FilmSlideshow = ({ autoSlide = false, slideshow = [], slideInterval = 5000 }: Props) => {\n  const [swipeDistance, setSwipeDistance] = useState(0);\n  const [slideIndex, setSlideIndex] = useState(0);\n  const [slideIndexTarget, setSlideIndexTarget] = useState(0);\n  const [animationComplete, setAnimationComplete] = useState(true);\n  const [swipeDirection, setSwipeDirection] = useState<SwipeDirections | undefined>(undefined);\n  const slideRef = useRef<HTMLDivElement>(null);\n  const slideText = useRef<HTMLDivElement>(null);\n  const timer = useRef<ReturnType<typeof setTimeout> | null>(null);\n\n  const gotoSlide = useCallback((indexTarget: number, useAnimation = false) => {\n    setSwipeDistance(0);\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSlideIndexTarget(indexTarget);\n    setAnimationComplete(!useAnimation);\n  }, []);\n\n  const onChangedSlide = () => {\n    if (!animationComplete) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideIndexTarget * 100}vw))`;\n      }\n\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    } else if (slideIndexTarget === -1) {\n      if (slideRef.current) {\n        // Go to last slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideshow.length * 100}vw))`;\n      }\n\n      setSlideIndex(slideshow.length - 1);\n      setSlideIndexTarget(slideshow.length - 1);\n      setAnimationComplete(true);\n    } else if (slideIndexTarget === slideshow.length) {\n      if (slideRef.current) {\n        // Go to first slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(100vw))`;\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n      setAnimationComplete(true);\n    } else {\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    }\n  };\n\n  const onSwipeEnd = () => {\n    setSwipeDirection(undefined);\n    let slide;\n    if (swipeDistance > 40) {\n      slide = -1;\n    } else if (swipeDistance < -40) {\n      slide = 1;\n    } else {\n      slide = 0;\n    }\n    if (slideRef.current && slideText.current) {\n      slideRef.current.style.transition = defaultTransitionSwipeEnd;\n      slideText.current.style.transition = defaultTransitionText;\n      slideText.current.style.opacity = '1';\n    }\n    setSwipeDistance(0);\n\n    initTimer();\n\n    if (slide !== 0) {\n      setSlideIndex(slideIndex + slide);\n      setSlideIndexTarget(slideIndex + slide);\n    } else {\n      // Reset transfrom\n      if (slideRef.current) {\n        slideRef.current.style.transform = getSlidePosition(slideIndex + slide);\n      }\n    }\n  };\n\n  const onSwipe = (eventData: SwipeEventData) => {\n    if (eventData.initial) {\n      setSwipeDirection(eventData.dir);\n    }\n    const dir = eventData.initial ? eventData.dir : swipeDirection;\n    if (dir === 'Up' || dir === 'Down') {\n      return;\n    }\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSwipeDistance(eventData.deltaX);\n    if (slideRef && slideRef.current) {\n      slideRef.current.style.transition = 'none';\n      slideRef.current.style.transform = getSlidePosition(slideIndexTarget);\n    }\n    const opacityText = 1 - Math.min(100, Math.abs(swipeDistance)) / 100;\n    if (slideText && slideText.current) {\n      slideText.current.style.transition = 'none';\n      slideText.current.style.opacity = opacityText.toString();\n    }\n  };\n\n  const onTransitionEnd = () => {\n    const slideshowLength = slideshow.length;\n    if (slideIndex === -1) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(slideshowLength - 1);\n      }\n      setSlideIndex(slideshowLength - 1);\n      setSlideIndexTarget(slideshowLength - 1);\n    } else if (slideIndex >= slideshowLength) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(0);\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n    }\n  };\n\n  const getSlidePosition = (target: number) => {\n    if (swipeDistance !== 0) {\n      return `translateX(calc(${swipeDistance}px -\n        ${(target + 1) * 100}vw))`;\n    }\n    return `translateX(-${(target + 1) * 100}vw)`;\n  };\n\n  const initTimer = useCallback(() => {\n    if (autoSlide) {\n      timer.current = setTimeout(() => {\n        gotoSlide(slideIndex + 1);\n      }, slideInterval);\n    }\n  }, [autoSlide, gotoSlide, slideInterval, slideIndex]);\n\n  useEffect(() => {\n    initTimer();\n  }, [initTimer]);\n\n  const handlers = useSwipeable({\n    onSwiped: onSwipeEnd,\n    onSwiping: onSwipe,\n    preventScrollOnSwipe: swipeDirection === 'Left' || swipeDirection === 'Right',\n  });\n\n  if (slideshow.length === 0) {\n    return (\n      <div>\n        <EmptySlideshow>\n          <Spinner inverted />\n        </EmptySlideshow>\n      </div>\n    );\n  }\n\n  const slideshowWidth = `${(slideshow.length + 2) * 100}vw`;\n  let activeSlide = slideIndex;\n  if (activeSlide < 0) {\n    activeSlide = slideshow.length - 1;\n  } else if (activeSlide >= slideshow.length) {\n    activeSlide = 0;\n  }\n\n  const backgroundImage = slideshow[activeSlide].metaImage;\n\n  return (\n    <SlideshowWrapper {...handlers}>\n      <>\n        <SlideLinkWrapper>\n          <OneColumn>\n            <SlideshowLink to={slideshow[activeSlide].path} out={!animationComplete}>\n              <SlideshowInfo ref={slideText}>\n                <SlideshowName>{slideshow[activeSlide].title}</SlideshowName>\n                <SlideshowDescription>{slideshow[activeSlide].metaDescription}</SlideshowDescription>\n              </SlideshowInfo>\n            </SlideshowLink>\n          </OneColumn>\n        </SlideLinkWrapper>\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget > 0 ? slideIndexTarget - 1 : slideshow.length - 1}\n          gotoSlide={gotoSlide}\n        />\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget < slideshow.length - 1 ? slideIndexTarget + 1 : 0}\n          gotoSlide={gotoSlide}\n          rightArrow\n        />\n        {!animationComplete && (\n          <SlideshowItem\n            fadeOver\n            role=\"img\"\n            onAnimationEnd={onChangedSlide}\n            style={{\n              backgroundImage: `url(${(backgroundImage && backgroundImage.url) || ''})`,\n            }}\n          />\n        )}\n        <div\n          ref={slideRef}\n          css={itemWrapperCSS}\n          onTransitionEnd={onTransitionEnd}\n          style={{\n            width: slideshowWidth,\n            transform: getSlidePosition(slideIndex),\n          }}\n        >\n          {renderSlideItem(slideshow[slideshow.length - 1])}\n          {slideshow.map(renderSlideItem)}\n          {renderSlideItem(slideshow[0])}\n        </div>\n        <SlideshowIndicator slideshow={slideshow} activeSlide={activeSlide} gotoSlide={gotoSlide} />\n      </>\n    </SlideshowWrapper>\n  );\n};\n\nexport default FilmSlideshow;\n"]} */"));
103
- var SlideshowWrapper = /*#__PURE__*/(0, _base["default"])("section", {
85
+ label: "StyledSafeLink"
86
+ })(process.env.NODE_ENV === "production" ? {
87
+ name: "rmwyk3",
88
+ styles: "position:relative;display:block;box-shadow:none"
89
+ } : {
90
+ name: "rmwyk3",
91
+ styles: "position:relative;display:block;box-shadow:none",
92
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AA4FuC","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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, { useCallback, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Carousel, CarouselAutosize } from '@ndla/carousel';\nimport { breakpoints, colors, misc, mq, spacing, spacingUnit } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { IconButtonV2 } from '@ndla/button';\nimport { ChevronLeft, ChevronRight } from '@ndla/icons/common';\nimport FilmContentCard from './FilmContentCard';\nimport { MovieType } from './types';\n\nexport const slideshowBreakpoints: {\n  until?: keyof typeof breakpoints;\n  columnsPrSlide: number;\n  distanceBetweenItems: number;\n  arrowOffset: number;\n  margin?: number;\n  maxColumnWidth?: number;\n}[] = [\n  {\n    until: 'mobileWide',\n    columnsPrSlide: 2,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'tabletWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'desktop',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'wide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'ultraWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n  {\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n];\n\ninterface Props {\n  slideshow: MovieType[];\n}\n\nconst SlideInfoWrapper = styled.div`\n  position: absolute;\n  color: ${colors.white};\n  max-width: 40%;\n  min-width: 40%;\n  top: 40%;\n  right: 5%;\n  ${mq.range({ until: breakpoints.desktop })} {\n    top: 30%;\n    max-width: 60%;\n    min-width: 60%;\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    max-width: 90%;\n    min-width: 90%;\n    left: 5%;\n  }\n`;\n\nconst StyledSafeLink = styled(SafeLink)`\n  position: relative;\n  display: block;\n  box-shadow: none;\n`;\n\nconst InfoWrapper = styled.div`\n  padding: ${spacing.normal};\n  border-radius: ${misc.borderRadius};\n  border: 0.5px solid ${colors.brand.primary};\n  background-color: rgba(11, 29, 45, 0.8);\n  h3 {\n    margin: 0px;\n  }\n`;\n\nconst StyledImg = styled.img`\n  max-height: 600px;\n  object-position: top;\n  width: 100%;\n  aspect-ratio: 16/9;\n  ${mq.range({ until: breakpoints.tablet })} {\n    min-height: 440px;\n    max-height: 440px;\n  }\n  object-fit: cover;\n`;\n\nconst CarouselContainer = styled.div`\n  margin-top: -50px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    margin-top: -70px;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin-top: -150px;\n  }\n`;\n\ninterface StyledFilmContentCardProps {\n  current?: boolean;\n}\n\nconst SlideshowButton = styled(IconButtonV2)`\n  margin-top: ${spacing.normal};\n`;\n\nconst shouldForwardProp = (p: string) => p !== 'current';\n\nconst StyledFilmContentCard = styled(FilmContentCard, { shouldForwardProp })<StyledFilmContentCardProps>`\n  margin-bottom: 2%;\n  transform: ${(p) => (p.current ? 'translateY(0%)' : 'translateY(10%)')};\n  transition: all 200ms;\n`;\n\nconst FilmSlideshow = ({ slideshow }: Props) => {\n  const [currentSlide, setCurrentSlide] = useState(slideshow[0]);\n\n  return (\n    <CarouselAutosize breakpoints={slideshowBreakpoints} itemsLength={slideshow.length}>\n      {(autoSizedProps) => (\n        <section>\n          <StyledSafeLink to={currentSlide.path} tabIndex={-1} aria-hidden>\n            <StyledImg src={currentSlide.metaImage?.url ?? ''} alt={currentSlide.metaImage?.alt ?? ''} />\n            <SlideInfoWrapper>\n              <InfoWrapper>\n                <h3>{currentSlide.title}</h3>\n                <span id=\"currentMovieDescription\">{currentSlide.metaDescription}</span>\n              </InfoWrapper>\n            </SlideInfoWrapper>\n          </StyledSafeLink>\n          <CarouselContainer>\n            <Carousel\n              leftButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronLeft />\n                </SlideshowButton>\n              }\n              rightButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronRight />\n                </SlideshowButton>\n              }\n              items={slideshow.map((movie) => (\n                <FilmCard\n                  key={movie.id}\n                  current={movie.id === currentSlide.id}\n                  movie={movie}\n                  columnWidth={autoSizedProps.columnWidth}\n                  setCurrentSlide={() => setCurrentSlide(movie)}\n                />\n              ))}\n              {...autoSizedProps}\n            />\n          </CarouselContainer>\n        </section>\n      )}\n    </CarouselAutosize>\n  );\n};\n\ninterface FilmCardProps {\n  setCurrentSlide: () => void;\n  movie: MovieType;\n  current: boolean;\n  columnWidth: number;\n}\n\nconst FilmCard = ({ setCurrentSlide, movie, current, columnWidth }: FilmCardProps) => {\n  const [hoverCallback, setHoverCallback] = useState<ReturnType<typeof setTimeout> | undefined>(undefined);\n\n  const onHover = useCallback(() => {\n    const timeout = setTimeout(() => setCurrentSlide(), 500);\n    setHoverCallback(timeout);\n  }, [setCurrentSlide]);\n\n  return (\n    <StyledFilmContentCard\n      onMouseEnter={onHover}\n      onMouseLeave={() => {\n        if (hoverCallback) {\n          clearTimeout(hoverCallback);\n          setHoverCallback(undefined);\n        }\n      }}\n      onFocus={() => setCurrentSlide()}\n      current={current}\n      aria-describedby={'currentMovieDescription'}\n      key={movie.id}\n      movie={movie}\n      columnWidth={columnWidth}\n      resourceTypes={[]}\n    />\n  );\n};\n\nexport default FilmSlideshow;\n"]} */",
93
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
94
+ });
95
+ var InfoWrapper = /*#__PURE__*/(0, _base["default"])("div", {
104
96
  target: "essc37o4",
105
- label: "SlideshowWrapper"
106
- })("width:100%;overflow:hidden;&:hover{", _NavigationArrow.StyledNavigationArrow, "{opacity:1;transform:translate(0, 0);}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AAmIuC","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2016-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, { useCallback, useEffect, useRef, useState } from 'react';\nimport { SwipeDirections, SwipeEventData, useSwipeable } from 'react-swipeable';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { breakpoints, mq, spacing, spacingUnit, fonts, colors } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { Spinner } from '@ndla/icons';\nimport { OneColumn } from '../Layout';\nimport NavigationArrow, { StyledNavigationArrow } from './NavigationArrow';\nimport SlideshowIndicator from './SlideshowIndicator';\nimport { MovieType } from './types';\n\ninterface Props {\n  autoSlide?: boolean;\n  randomStart?: boolean;\n  slideshow: MovieType[];\n  slideInterval?: number;\n}\n\nconst SlideLinkWrapper = styled.div`\n  margin: 0 auto;\n  display: flex;\n  justify-content: flex-start;\n  align-items: flex-end;\n  position: absolute;\n  z-index: 2;\n  height: 100vw;\n  width: 100%;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n`;\n\nconst itemWrapperCSS = css`\n  display: flex;\n  box-shadow: none;\n`;\n\ninterface SlideshowItemProps {\n  fadeOver?: boolean;\n}\n\nconst SlideshowItem = styled.div<SlideshowItemProps>`\n  width: 100vw;\n  height: 100vw;\n  /* aspect ratios */\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n  background-color: '#222';\n  background-size: cover;\n  background-position-x: center;\n  background-position-y: center;\n  border: 0;\n  position: ${(props) => (props.fadeOver ? 'absolute' : 'relative')};\n  animation: ${(props) => props.fadeOver && 'fadeIn 400ms ease'};\n  z-index: ${(props) => props.fadeOver && 1};\n  &:before {\n    content: '';\n    opacity: 0.4;\n    background: #091a2a;\n    top: 0;\n    left: 0;\n    bottom: 0;\n    right: 0;\n    position: absolute;\n    z-index: 1;\n  }\n`;\n\ninterface SlideshowLinkProps {\n  out?: boolean;\n}\n\nconst shouldForwardProp = (p: string) => p !== 'out';\n\nconst SlideshowLink = styled(SafeLink, { shouldForwardProp })<SlideshowLinkProps>`\n  display: flex;\n  box-shadow: none;\n  transition: all 400ms ease;\n  opacity: ${(props) => props.out && 0};\n  animation: ${(props) => !props.out && 'fadeInBottomFixed 600ms ease'};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    padding-bottom: ${spacing.medium};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding-bottom: ${spacing.large};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding-bottom: ${spacingUnit * 3}px;\n  }\n  &:hover {\n    ${() => SlideshowName} {\n      text-decoration: underline;\n      text-decoration-color: white;\n    }\n  }\n`;\n\nconst SlideshowWrapper = styled.section`\n  width: 100%;\n  overflow: hidden;\n  &:hover {\n    ${StyledNavigationArrow} {\n      opacity: 1;\n      transform: translate(0, 0);\n    }\n  }\n`;\n\nconst SlideshowInfo = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n  background-color: rgba(3, 23, 43, 0.7);\n  border-radius: 4px;\n  padding: ${spacing.medium} ${spacing.medium} ${spacing.medium} ${spacing.normal};\n  margin: 0 -20px;\n  width: 100vw;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin: 0;\n    width: 100%;\n    padding: ${spacing.medium} ${spacingUnit * 2}px ${spacing.medium} ${spacing.normal};\n  }\n`;\n\nconst SlideshowName = styled.p`\n  ${fonts.sizes('22px', '30px')};\n  color: ${colors.white};\n  text-shadow: 0 2px 4px rgba(0, 0, 0, 0.25);\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('26px', '30px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('40px', '44px')};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    ${fonts.sizes('48px', '54px')};\n  }\n`;\n\nconst SlideshowDescription = styled.p`\n  color: ${colors.white};\n  margin: 0;\n  padding: 0;\n  ${fonts.sizes('12px', '18px')};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('15px', '20px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('18px', '24px')};\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${fonts.sizes('20px', '32px')};\n  }\n`;\n\nconst EmptySlideshow = styled.div`\n  background: rgba(255, 255, 255, 0.08);\n  margin-bottom: $spacing--large * 4;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  height: 40vw;\n`;\n\nconst defaultTransitionSwipeEnd = 'transform 600ms cubic-bezier(0, 0.76, 0.09, 1)';\nconst defaultTransitionText = 'opacity 600ms ease';\n\nconst renderSlideItem = (slide: MovieType) => (\n  <SlideshowItem\n    key={slide.id}\n    role=\"img\"\n    aria-label={(slide.metaImage && slide.metaImage.alt) || ''}\n    style={{\n      backgroundImage: `url(${(slide.metaImage && slide.metaImage.url) || ''})`,\n    }}\n  />\n);\n\nconst FilmSlideshow = ({ autoSlide = false, slideshow = [], slideInterval = 5000 }: Props) => {\n  const [swipeDistance, setSwipeDistance] = useState(0);\n  const [slideIndex, setSlideIndex] = useState(0);\n  const [slideIndexTarget, setSlideIndexTarget] = useState(0);\n  const [animationComplete, setAnimationComplete] = useState(true);\n  const [swipeDirection, setSwipeDirection] = useState<SwipeDirections | undefined>(undefined);\n  const slideRef = useRef<HTMLDivElement>(null);\n  const slideText = useRef<HTMLDivElement>(null);\n  const timer = useRef<ReturnType<typeof setTimeout> | null>(null);\n\n  const gotoSlide = useCallback((indexTarget: number, useAnimation = false) => {\n    setSwipeDistance(0);\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSlideIndexTarget(indexTarget);\n    setAnimationComplete(!useAnimation);\n  }, []);\n\n  const onChangedSlide = () => {\n    if (!animationComplete) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideIndexTarget * 100}vw))`;\n      }\n\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    } else if (slideIndexTarget === -1) {\n      if (slideRef.current) {\n        // Go to last slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideshow.length * 100}vw))`;\n      }\n\n      setSlideIndex(slideshow.length - 1);\n      setSlideIndexTarget(slideshow.length - 1);\n      setAnimationComplete(true);\n    } else if (slideIndexTarget === slideshow.length) {\n      if (slideRef.current) {\n        // Go to first slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(100vw))`;\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n      setAnimationComplete(true);\n    } else {\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    }\n  };\n\n  const onSwipeEnd = () => {\n    setSwipeDirection(undefined);\n    let slide;\n    if (swipeDistance > 40) {\n      slide = -1;\n    } else if (swipeDistance < -40) {\n      slide = 1;\n    } else {\n      slide = 0;\n    }\n    if (slideRef.current && slideText.current) {\n      slideRef.current.style.transition = defaultTransitionSwipeEnd;\n      slideText.current.style.transition = defaultTransitionText;\n      slideText.current.style.opacity = '1';\n    }\n    setSwipeDistance(0);\n\n    initTimer();\n\n    if (slide !== 0) {\n      setSlideIndex(slideIndex + slide);\n      setSlideIndexTarget(slideIndex + slide);\n    } else {\n      // Reset transfrom\n      if (slideRef.current) {\n        slideRef.current.style.transform = getSlidePosition(slideIndex + slide);\n      }\n    }\n  };\n\n  const onSwipe = (eventData: SwipeEventData) => {\n    if (eventData.initial) {\n      setSwipeDirection(eventData.dir);\n    }\n    const dir = eventData.initial ? eventData.dir : swipeDirection;\n    if (dir === 'Up' || dir === 'Down') {\n      return;\n    }\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSwipeDistance(eventData.deltaX);\n    if (slideRef && slideRef.current) {\n      slideRef.current.style.transition = 'none';\n      slideRef.current.style.transform = getSlidePosition(slideIndexTarget);\n    }\n    const opacityText = 1 - Math.min(100, Math.abs(swipeDistance)) / 100;\n    if (slideText && slideText.current) {\n      slideText.current.style.transition = 'none';\n      slideText.current.style.opacity = opacityText.toString();\n    }\n  };\n\n  const onTransitionEnd = () => {\n    const slideshowLength = slideshow.length;\n    if (slideIndex === -1) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(slideshowLength - 1);\n      }\n      setSlideIndex(slideshowLength - 1);\n      setSlideIndexTarget(slideshowLength - 1);\n    } else if (slideIndex >= slideshowLength) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(0);\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n    }\n  };\n\n  const getSlidePosition = (target: number) => {\n    if (swipeDistance !== 0) {\n      return `translateX(calc(${swipeDistance}px -\n        ${(target + 1) * 100}vw))`;\n    }\n    return `translateX(-${(target + 1) * 100}vw)`;\n  };\n\n  const initTimer = useCallback(() => {\n    if (autoSlide) {\n      timer.current = setTimeout(() => {\n        gotoSlide(slideIndex + 1);\n      }, slideInterval);\n    }\n  }, [autoSlide, gotoSlide, slideInterval, slideIndex]);\n\n  useEffect(() => {\n    initTimer();\n  }, [initTimer]);\n\n  const handlers = useSwipeable({\n    onSwiped: onSwipeEnd,\n    onSwiping: onSwipe,\n    preventScrollOnSwipe: swipeDirection === 'Left' || swipeDirection === 'Right',\n  });\n\n  if (slideshow.length === 0) {\n    return (\n      <div>\n        <EmptySlideshow>\n          <Spinner inverted />\n        </EmptySlideshow>\n      </div>\n    );\n  }\n\n  const slideshowWidth = `${(slideshow.length + 2) * 100}vw`;\n  let activeSlide = slideIndex;\n  if (activeSlide < 0) {\n    activeSlide = slideshow.length - 1;\n  } else if (activeSlide >= slideshow.length) {\n    activeSlide = 0;\n  }\n\n  const backgroundImage = slideshow[activeSlide].metaImage;\n\n  return (\n    <SlideshowWrapper {...handlers}>\n      <>\n        <SlideLinkWrapper>\n          <OneColumn>\n            <SlideshowLink to={slideshow[activeSlide].path} out={!animationComplete}>\n              <SlideshowInfo ref={slideText}>\n                <SlideshowName>{slideshow[activeSlide].title}</SlideshowName>\n                <SlideshowDescription>{slideshow[activeSlide].metaDescription}</SlideshowDescription>\n              </SlideshowInfo>\n            </SlideshowLink>\n          </OneColumn>\n        </SlideLinkWrapper>\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget > 0 ? slideIndexTarget - 1 : slideshow.length - 1}\n          gotoSlide={gotoSlide}\n        />\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget < slideshow.length - 1 ? slideIndexTarget + 1 : 0}\n          gotoSlide={gotoSlide}\n          rightArrow\n        />\n        {!animationComplete && (\n          <SlideshowItem\n            fadeOver\n            role=\"img\"\n            onAnimationEnd={onChangedSlide}\n            style={{\n              backgroundImage: `url(${(backgroundImage && backgroundImage.url) || ''})`,\n            }}\n          />\n        )}\n        <div\n          ref={slideRef}\n          css={itemWrapperCSS}\n          onTransitionEnd={onTransitionEnd}\n          style={{\n            width: slideshowWidth,\n            transform: getSlidePosition(slideIndex),\n          }}\n        >\n          {renderSlideItem(slideshow[slideshow.length - 1])}\n          {slideshow.map(renderSlideItem)}\n          {renderSlideItem(slideshow[0])}\n        </div>\n        <SlideshowIndicator slideshow={slideshow} activeSlide={activeSlide} gotoSlide={gotoSlide} />\n      </>\n    </SlideshowWrapper>\n  );\n};\n\nexport default FilmSlideshow;\n"]} */"));
107
- var SlideshowInfo = /*#__PURE__*/(0, _base["default"])("div", {
97
+ label: "InfoWrapper"
98
+ })("padding:", _core.spacing.normal, ";border-radius:", _core.misc.borderRadius, ";border:0.5px solid ", _core.colors.brand.primary, ";background-color:rgba(11, 29, 45, 0.8);h3{margin:0px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AAkG8B","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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, { useCallback, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Carousel, CarouselAutosize } from '@ndla/carousel';\nimport { breakpoints, colors, misc, mq, spacing, spacingUnit } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { IconButtonV2 } from '@ndla/button';\nimport { ChevronLeft, ChevronRight } from '@ndla/icons/common';\nimport FilmContentCard from './FilmContentCard';\nimport { MovieType } from './types';\n\nexport const slideshowBreakpoints: {\n  until?: keyof typeof breakpoints;\n  columnsPrSlide: number;\n  distanceBetweenItems: number;\n  arrowOffset: number;\n  margin?: number;\n  maxColumnWidth?: number;\n}[] = [\n  {\n    until: 'mobileWide',\n    columnsPrSlide: 2,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'tabletWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'desktop',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'wide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'ultraWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n  {\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n];\n\ninterface Props {\n  slideshow: MovieType[];\n}\n\nconst SlideInfoWrapper = styled.div`\n  position: absolute;\n  color: ${colors.white};\n  max-width: 40%;\n  min-width: 40%;\n  top: 40%;\n  right: 5%;\n  ${mq.range({ until: breakpoints.desktop })} {\n    top: 30%;\n    max-width: 60%;\n    min-width: 60%;\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    max-width: 90%;\n    min-width: 90%;\n    left: 5%;\n  }\n`;\n\nconst StyledSafeLink = styled(SafeLink)`\n  position: relative;\n  display: block;\n  box-shadow: none;\n`;\n\nconst InfoWrapper = styled.div`\n  padding: ${spacing.normal};\n  border-radius: ${misc.borderRadius};\n  border: 0.5px solid ${colors.brand.primary};\n  background-color: rgba(11, 29, 45, 0.8);\n  h3 {\n    margin: 0px;\n  }\n`;\n\nconst StyledImg = styled.img`\n  max-height: 600px;\n  object-position: top;\n  width: 100%;\n  aspect-ratio: 16/9;\n  ${mq.range({ until: breakpoints.tablet })} {\n    min-height: 440px;\n    max-height: 440px;\n  }\n  object-fit: cover;\n`;\n\nconst CarouselContainer = styled.div`\n  margin-top: -50px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    margin-top: -70px;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin-top: -150px;\n  }\n`;\n\ninterface StyledFilmContentCardProps {\n  current?: boolean;\n}\n\nconst SlideshowButton = styled(IconButtonV2)`\n  margin-top: ${spacing.normal};\n`;\n\nconst shouldForwardProp = (p: string) => p !== 'current';\n\nconst StyledFilmContentCard = styled(FilmContentCard, { shouldForwardProp })<StyledFilmContentCardProps>`\n  margin-bottom: 2%;\n  transform: ${(p) => (p.current ? 'translateY(0%)' : 'translateY(10%)')};\n  transition: all 200ms;\n`;\n\nconst FilmSlideshow = ({ slideshow }: Props) => {\n  const [currentSlide, setCurrentSlide] = useState(slideshow[0]);\n\n  return (\n    <CarouselAutosize breakpoints={slideshowBreakpoints} itemsLength={slideshow.length}>\n      {(autoSizedProps) => (\n        <section>\n          <StyledSafeLink to={currentSlide.path} tabIndex={-1} aria-hidden>\n            <StyledImg src={currentSlide.metaImage?.url ?? ''} alt={currentSlide.metaImage?.alt ?? ''} />\n            <SlideInfoWrapper>\n              <InfoWrapper>\n                <h3>{currentSlide.title}</h3>\n                <span id=\"currentMovieDescription\">{currentSlide.metaDescription}</span>\n              </InfoWrapper>\n            </SlideInfoWrapper>\n          </StyledSafeLink>\n          <CarouselContainer>\n            <Carousel\n              leftButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronLeft />\n                </SlideshowButton>\n              }\n              rightButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronRight />\n                </SlideshowButton>\n              }\n              items={slideshow.map((movie) => (\n                <FilmCard\n                  key={movie.id}\n                  current={movie.id === currentSlide.id}\n                  movie={movie}\n                  columnWidth={autoSizedProps.columnWidth}\n                  setCurrentSlide={() => setCurrentSlide(movie)}\n                />\n              ))}\n              {...autoSizedProps}\n            />\n          </CarouselContainer>\n        </section>\n      )}\n    </CarouselAutosize>\n  );\n};\n\ninterface FilmCardProps {\n  setCurrentSlide: () => void;\n  movie: MovieType;\n  current: boolean;\n  columnWidth: number;\n}\n\nconst FilmCard = ({ setCurrentSlide, movie, current, columnWidth }: FilmCardProps) => {\n  const [hoverCallback, setHoverCallback] = useState<ReturnType<typeof setTimeout> | undefined>(undefined);\n\n  const onHover = useCallback(() => {\n    const timeout = setTimeout(() => setCurrentSlide(), 500);\n    setHoverCallback(timeout);\n  }, [setCurrentSlide]);\n\n  return (\n    <StyledFilmContentCard\n      onMouseEnter={onHover}\n      onMouseLeave={() => {\n        if (hoverCallback) {\n          clearTimeout(hoverCallback);\n          setHoverCallback(undefined);\n        }\n      }}\n      onFocus={() => setCurrentSlide()}\n      current={current}\n      aria-describedby={'currentMovieDescription'}\n      key={movie.id}\n      movie={movie}\n      columnWidth={columnWidth}\n      resourceTypes={[]}\n    />\n  );\n};\n\nexport default FilmSlideshow;\n"]} */"));
99
+ var StyledImg = /*#__PURE__*/(0, _base["default"])("img", {
108
100
  target: "essc37o3",
109
- label: "SlideshowInfo"
110
- })("display:flex;flex-direction:column;gap:", _core.spacing.small, ";background-color:rgba(3, 23, 43, 0.7);border-radius:4px;padding:", _core.spacing.medium, " ", _core.spacing.medium, " ", _core.spacing.medium, " ", _core.spacing.normal, ";margin:0 -20px;width:100vw;", _core.mq.range({
111
- from: _core.breakpoints.mobileWide
112
- }), "{margin:0;width:100%;padding:", _core.spacing.medium, " ", _core.spacingUnit * 2, "px ", _core.spacing.medium, " ", _core.spacing.normal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AA8IgC","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2016-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, { useCallback, useEffect, useRef, useState } from 'react';\nimport { SwipeDirections, SwipeEventData, useSwipeable } from 'react-swipeable';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { breakpoints, mq, spacing, spacingUnit, fonts, colors } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { Spinner } from '@ndla/icons';\nimport { OneColumn } from '../Layout';\nimport NavigationArrow, { StyledNavigationArrow } from './NavigationArrow';\nimport SlideshowIndicator from './SlideshowIndicator';\nimport { MovieType } from './types';\n\ninterface Props {\n  autoSlide?: boolean;\n  randomStart?: boolean;\n  slideshow: MovieType[];\n  slideInterval?: number;\n}\n\nconst SlideLinkWrapper = styled.div`\n  margin: 0 auto;\n  display: flex;\n  justify-content: flex-start;\n  align-items: flex-end;\n  position: absolute;\n  z-index: 2;\n  height: 100vw;\n  width: 100%;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n`;\n\nconst itemWrapperCSS = css`\n  display: flex;\n  box-shadow: none;\n`;\n\ninterface SlideshowItemProps {\n  fadeOver?: boolean;\n}\n\nconst SlideshowItem = styled.div<SlideshowItemProps>`\n  width: 100vw;\n  height: 100vw;\n  /* aspect ratios */\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n  background-color: '#222';\n  background-size: cover;\n  background-position-x: center;\n  background-position-y: center;\n  border: 0;\n  position: ${(props) => (props.fadeOver ? 'absolute' : 'relative')};\n  animation: ${(props) => props.fadeOver && 'fadeIn 400ms ease'};\n  z-index: ${(props) => props.fadeOver && 1};\n  &:before {\n    content: '';\n    opacity: 0.4;\n    background: #091a2a;\n    top: 0;\n    left: 0;\n    bottom: 0;\n    right: 0;\n    position: absolute;\n    z-index: 1;\n  }\n`;\n\ninterface SlideshowLinkProps {\n  out?: boolean;\n}\n\nconst shouldForwardProp = (p: string) => p !== 'out';\n\nconst SlideshowLink = styled(SafeLink, { shouldForwardProp })<SlideshowLinkProps>`\n  display: flex;\n  box-shadow: none;\n  transition: all 400ms ease;\n  opacity: ${(props) => props.out && 0};\n  animation: ${(props) => !props.out && 'fadeInBottomFixed 600ms ease'};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    padding-bottom: ${spacing.medium};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding-bottom: ${spacing.large};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding-bottom: ${spacingUnit * 3}px;\n  }\n  &:hover {\n    ${() => SlideshowName} {\n      text-decoration: underline;\n      text-decoration-color: white;\n    }\n  }\n`;\n\nconst SlideshowWrapper = styled.section`\n  width: 100%;\n  overflow: hidden;\n  &:hover {\n    ${StyledNavigationArrow} {\n      opacity: 1;\n      transform: translate(0, 0);\n    }\n  }\n`;\n\nconst SlideshowInfo = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n  background-color: rgba(3, 23, 43, 0.7);\n  border-radius: 4px;\n  padding: ${spacing.medium} ${spacing.medium} ${spacing.medium} ${spacing.normal};\n  margin: 0 -20px;\n  width: 100vw;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin: 0;\n    width: 100%;\n    padding: ${spacing.medium} ${spacingUnit * 2}px ${spacing.medium} ${spacing.normal};\n  }\n`;\n\nconst SlideshowName = styled.p`\n  ${fonts.sizes('22px', '30px')};\n  color: ${colors.white};\n  text-shadow: 0 2px 4px rgba(0, 0, 0, 0.25);\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('26px', '30px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('40px', '44px')};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    ${fonts.sizes('48px', '54px')};\n  }\n`;\n\nconst SlideshowDescription = styled.p`\n  color: ${colors.white};\n  margin: 0;\n  padding: 0;\n  ${fonts.sizes('12px', '18px')};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('15px', '20px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('18px', '24px')};\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${fonts.sizes('20px', '32px')};\n  }\n`;\n\nconst EmptySlideshow = styled.div`\n  background: rgba(255, 255, 255, 0.08);\n  margin-bottom: $spacing--large * 4;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  height: 40vw;\n`;\n\nconst defaultTransitionSwipeEnd = 'transform 600ms cubic-bezier(0, 0.76, 0.09, 1)';\nconst defaultTransitionText = 'opacity 600ms ease';\n\nconst renderSlideItem = (slide: MovieType) => (\n  <SlideshowItem\n    key={slide.id}\n    role=\"img\"\n    aria-label={(slide.metaImage && slide.metaImage.alt) || ''}\n    style={{\n      backgroundImage: `url(${(slide.metaImage && slide.metaImage.url) || ''})`,\n    }}\n  />\n);\n\nconst FilmSlideshow = ({ autoSlide = false, slideshow = [], slideInterval = 5000 }: Props) => {\n  const [swipeDistance, setSwipeDistance] = useState(0);\n  const [slideIndex, setSlideIndex] = useState(0);\n  const [slideIndexTarget, setSlideIndexTarget] = useState(0);\n  const [animationComplete, setAnimationComplete] = useState(true);\n  const [swipeDirection, setSwipeDirection] = useState<SwipeDirections | undefined>(undefined);\n  const slideRef = useRef<HTMLDivElement>(null);\n  const slideText = useRef<HTMLDivElement>(null);\n  const timer = useRef<ReturnType<typeof setTimeout> | null>(null);\n\n  const gotoSlide = useCallback((indexTarget: number, useAnimation = false) => {\n    setSwipeDistance(0);\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSlideIndexTarget(indexTarget);\n    setAnimationComplete(!useAnimation);\n  }, []);\n\n  const onChangedSlide = () => {\n    if (!animationComplete) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideIndexTarget * 100}vw))`;\n      }\n\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    } else if (slideIndexTarget === -1) {\n      if (slideRef.current) {\n        // Go to last slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideshow.length * 100}vw))`;\n      }\n\n      setSlideIndex(slideshow.length - 1);\n      setSlideIndexTarget(slideshow.length - 1);\n      setAnimationComplete(true);\n    } else if (slideIndexTarget === slideshow.length) {\n      if (slideRef.current) {\n        // Go to first slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(100vw))`;\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n      setAnimationComplete(true);\n    } else {\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    }\n  };\n\n  const onSwipeEnd = () => {\n    setSwipeDirection(undefined);\n    let slide;\n    if (swipeDistance > 40) {\n      slide = -1;\n    } else if (swipeDistance < -40) {\n      slide = 1;\n    } else {\n      slide = 0;\n    }\n    if (slideRef.current && slideText.current) {\n      slideRef.current.style.transition = defaultTransitionSwipeEnd;\n      slideText.current.style.transition = defaultTransitionText;\n      slideText.current.style.opacity = '1';\n    }\n    setSwipeDistance(0);\n\n    initTimer();\n\n    if (slide !== 0) {\n      setSlideIndex(slideIndex + slide);\n      setSlideIndexTarget(slideIndex + slide);\n    } else {\n      // Reset transfrom\n      if (slideRef.current) {\n        slideRef.current.style.transform = getSlidePosition(slideIndex + slide);\n      }\n    }\n  };\n\n  const onSwipe = (eventData: SwipeEventData) => {\n    if (eventData.initial) {\n      setSwipeDirection(eventData.dir);\n    }\n    const dir = eventData.initial ? eventData.dir : swipeDirection;\n    if (dir === 'Up' || dir === 'Down') {\n      return;\n    }\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSwipeDistance(eventData.deltaX);\n    if (slideRef && slideRef.current) {\n      slideRef.current.style.transition = 'none';\n      slideRef.current.style.transform = getSlidePosition(slideIndexTarget);\n    }\n    const opacityText = 1 - Math.min(100, Math.abs(swipeDistance)) / 100;\n    if (slideText && slideText.current) {\n      slideText.current.style.transition = 'none';\n      slideText.current.style.opacity = opacityText.toString();\n    }\n  };\n\n  const onTransitionEnd = () => {\n    const slideshowLength = slideshow.length;\n    if (slideIndex === -1) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(slideshowLength - 1);\n      }\n      setSlideIndex(slideshowLength - 1);\n      setSlideIndexTarget(slideshowLength - 1);\n    } else if (slideIndex >= slideshowLength) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(0);\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n    }\n  };\n\n  const getSlidePosition = (target: number) => {\n    if (swipeDistance !== 0) {\n      return `translateX(calc(${swipeDistance}px -\n        ${(target + 1) * 100}vw))`;\n    }\n    return `translateX(-${(target + 1) * 100}vw)`;\n  };\n\n  const initTimer = useCallback(() => {\n    if (autoSlide) {\n      timer.current = setTimeout(() => {\n        gotoSlide(slideIndex + 1);\n      }, slideInterval);\n    }\n  }, [autoSlide, gotoSlide, slideInterval, slideIndex]);\n\n  useEffect(() => {\n    initTimer();\n  }, [initTimer]);\n\n  const handlers = useSwipeable({\n    onSwiped: onSwipeEnd,\n    onSwiping: onSwipe,\n    preventScrollOnSwipe: swipeDirection === 'Left' || swipeDirection === 'Right',\n  });\n\n  if (slideshow.length === 0) {\n    return (\n      <div>\n        <EmptySlideshow>\n          <Spinner inverted />\n        </EmptySlideshow>\n      </div>\n    );\n  }\n\n  const slideshowWidth = `${(slideshow.length + 2) * 100}vw`;\n  let activeSlide = slideIndex;\n  if (activeSlide < 0) {\n    activeSlide = slideshow.length - 1;\n  } else if (activeSlide >= slideshow.length) {\n    activeSlide = 0;\n  }\n\n  const backgroundImage = slideshow[activeSlide].metaImage;\n\n  return (\n    <SlideshowWrapper {...handlers}>\n      <>\n        <SlideLinkWrapper>\n          <OneColumn>\n            <SlideshowLink to={slideshow[activeSlide].path} out={!animationComplete}>\n              <SlideshowInfo ref={slideText}>\n                <SlideshowName>{slideshow[activeSlide].title}</SlideshowName>\n                <SlideshowDescription>{slideshow[activeSlide].metaDescription}</SlideshowDescription>\n              </SlideshowInfo>\n            </SlideshowLink>\n          </OneColumn>\n        </SlideLinkWrapper>\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget > 0 ? slideIndexTarget - 1 : slideshow.length - 1}\n          gotoSlide={gotoSlide}\n        />\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget < slideshow.length - 1 ? slideIndexTarget + 1 : 0}\n          gotoSlide={gotoSlide}\n          rightArrow\n        />\n        {!animationComplete && (\n          <SlideshowItem\n            fadeOver\n            role=\"img\"\n            onAnimationEnd={onChangedSlide}\n            style={{\n              backgroundImage: `url(${(backgroundImage && backgroundImage.url) || ''})`,\n            }}\n          />\n        )}\n        <div\n          ref={slideRef}\n          css={itemWrapperCSS}\n          onTransitionEnd={onTransitionEnd}\n          style={{\n            width: slideshowWidth,\n            transform: getSlidePosition(slideIndex),\n          }}\n        >\n          {renderSlideItem(slideshow[slideshow.length - 1])}\n          {slideshow.map(renderSlideItem)}\n          {renderSlideItem(slideshow[0])}\n        </div>\n        <SlideshowIndicator slideshow={slideshow} activeSlide={activeSlide} gotoSlide={gotoSlide} />\n      </>\n    </SlideshowWrapper>\n  );\n};\n\nexport default FilmSlideshow;\n"]} */"));
113
- var SlideshowName = /*#__PURE__*/(0, _base["default"])("p", {
101
+ label: "StyledImg"
102
+ })("max-height:600px;object-position:top;width:100%;aspect-ratio:16/9;", _core.mq.range({
103
+ until: _core.breakpoints.tablet
104
+ }), "{min-height:440px;max-height:440px;}object-fit:cover;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AA4G4B","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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, { useCallback, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Carousel, CarouselAutosize } from '@ndla/carousel';\nimport { breakpoints, colors, misc, mq, spacing, spacingUnit } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { IconButtonV2 } from '@ndla/button';\nimport { ChevronLeft, ChevronRight } from '@ndla/icons/common';\nimport FilmContentCard from './FilmContentCard';\nimport { MovieType } from './types';\n\nexport const slideshowBreakpoints: {\n  until?: keyof typeof breakpoints;\n  columnsPrSlide: number;\n  distanceBetweenItems: number;\n  arrowOffset: number;\n  margin?: number;\n  maxColumnWidth?: number;\n}[] = [\n  {\n    until: 'mobileWide',\n    columnsPrSlide: 2,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'tabletWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'desktop',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'wide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'ultraWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n  {\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n];\n\ninterface Props {\n  slideshow: MovieType[];\n}\n\nconst SlideInfoWrapper = styled.div`\n  position: absolute;\n  color: ${colors.white};\n  max-width: 40%;\n  min-width: 40%;\n  top: 40%;\n  right: 5%;\n  ${mq.range({ until: breakpoints.desktop })} {\n    top: 30%;\n    max-width: 60%;\n    min-width: 60%;\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    max-width: 90%;\n    min-width: 90%;\n    left: 5%;\n  }\n`;\n\nconst StyledSafeLink = styled(SafeLink)`\n  position: relative;\n  display: block;\n  box-shadow: none;\n`;\n\nconst InfoWrapper = styled.div`\n  padding: ${spacing.normal};\n  border-radius: ${misc.borderRadius};\n  border: 0.5px solid ${colors.brand.primary};\n  background-color: rgba(11, 29, 45, 0.8);\n  h3 {\n    margin: 0px;\n  }\n`;\n\nconst StyledImg = styled.img`\n  max-height: 600px;\n  object-position: top;\n  width: 100%;\n  aspect-ratio: 16/9;\n  ${mq.range({ until: breakpoints.tablet })} {\n    min-height: 440px;\n    max-height: 440px;\n  }\n  object-fit: cover;\n`;\n\nconst CarouselContainer = styled.div`\n  margin-top: -50px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    margin-top: -70px;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin-top: -150px;\n  }\n`;\n\ninterface StyledFilmContentCardProps {\n  current?: boolean;\n}\n\nconst SlideshowButton = styled(IconButtonV2)`\n  margin-top: ${spacing.normal};\n`;\n\nconst shouldForwardProp = (p: string) => p !== 'current';\n\nconst StyledFilmContentCard = styled(FilmContentCard, { shouldForwardProp })<StyledFilmContentCardProps>`\n  margin-bottom: 2%;\n  transform: ${(p) => (p.current ? 'translateY(0%)' : 'translateY(10%)')};\n  transition: all 200ms;\n`;\n\nconst FilmSlideshow = ({ slideshow }: Props) => {\n  const [currentSlide, setCurrentSlide] = useState(slideshow[0]);\n\n  return (\n    <CarouselAutosize breakpoints={slideshowBreakpoints} itemsLength={slideshow.length}>\n      {(autoSizedProps) => (\n        <section>\n          <StyledSafeLink to={currentSlide.path} tabIndex={-1} aria-hidden>\n            <StyledImg src={currentSlide.metaImage?.url ?? ''} alt={currentSlide.metaImage?.alt ?? ''} />\n            <SlideInfoWrapper>\n              <InfoWrapper>\n                <h3>{currentSlide.title}</h3>\n                <span id=\"currentMovieDescription\">{currentSlide.metaDescription}</span>\n              </InfoWrapper>\n            </SlideInfoWrapper>\n          </StyledSafeLink>\n          <CarouselContainer>\n            <Carousel\n              leftButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronLeft />\n                </SlideshowButton>\n              }\n              rightButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronRight />\n                </SlideshowButton>\n              }\n              items={slideshow.map((movie) => (\n                <FilmCard\n                  key={movie.id}\n                  current={movie.id === currentSlide.id}\n                  movie={movie}\n                  columnWidth={autoSizedProps.columnWidth}\n                  setCurrentSlide={() => setCurrentSlide(movie)}\n                />\n              ))}\n              {...autoSizedProps}\n            />\n          </CarouselContainer>\n        </section>\n      )}\n    </CarouselAutosize>\n  );\n};\n\ninterface FilmCardProps {\n  setCurrentSlide: () => void;\n  movie: MovieType;\n  current: boolean;\n  columnWidth: number;\n}\n\nconst FilmCard = ({ setCurrentSlide, movie, current, columnWidth }: FilmCardProps) => {\n  const [hoverCallback, setHoverCallback] = useState<ReturnType<typeof setTimeout> | undefined>(undefined);\n\n  const onHover = useCallback(() => {\n    const timeout = setTimeout(() => setCurrentSlide(), 500);\n    setHoverCallback(timeout);\n  }, [setCurrentSlide]);\n\n  return (\n    <StyledFilmContentCard\n      onMouseEnter={onHover}\n      onMouseLeave={() => {\n        if (hoverCallback) {\n          clearTimeout(hoverCallback);\n          setHoverCallback(undefined);\n        }\n      }}\n      onFocus={() => setCurrentSlide()}\n      current={current}\n      aria-describedby={'currentMovieDescription'}\n      key={movie.id}\n      movie={movie}\n      columnWidth={columnWidth}\n      resourceTypes={[]}\n    />\n  );\n};\n\nexport default FilmSlideshow;\n"]} */"));
105
+ var CarouselContainer = /*#__PURE__*/(0, _base["default"])("div", {
114
106
  target: "essc37o2",
115
- label: "SlideshowName"
116
- })(_core.fonts.sizes('22px', '30px'), ";color:", _core.colors.white, ";text-shadow:0 2px 4px rgba(0, 0, 0, 0.25);margin:0;font-weight:", _core.fonts.weight.semibold, ";", _core.mq.range({
117
- from: _core.breakpoints.mobileWide
118
- }), "{", _core.fonts.sizes('26px', '30px'), ";}", _core.mq.range({
107
+ label: "CarouselContainer"
108
+ })("margin-top:-50px;", _core.mq.range({
119
109
  from: _core.breakpoints.tablet
120
- }), "{", _core.fonts.sizes('40px', '44px'), ";}", _core.mq.range({
110
+ }), "{margin-top:-70px;}", _core.mq.range({
121
111
  from: _core.breakpoints.desktop
122
- }), "{", _core.fonts.sizes('48px', '54px'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AA8J8B","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2016-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, { useCallback, useEffect, useRef, useState } from 'react';\nimport { SwipeDirections, SwipeEventData, useSwipeable } from 'react-swipeable';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { breakpoints, mq, spacing, spacingUnit, fonts, colors } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { Spinner } from '@ndla/icons';\nimport { OneColumn } from '../Layout';\nimport NavigationArrow, { StyledNavigationArrow } from './NavigationArrow';\nimport SlideshowIndicator from './SlideshowIndicator';\nimport { MovieType } from './types';\n\ninterface Props {\n  autoSlide?: boolean;\n  randomStart?: boolean;\n  slideshow: MovieType[];\n  slideInterval?: number;\n}\n\nconst SlideLinkWrapper = styled.div`\n  margin: 0 auto;\n  display: flex;\n  justify-content: flex-start;\n  align-items: flex-end;\n  position: absolute;\n  z-index: 2;\n  height: 100vw;\n  width: 100%;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n`;\n\nconst itemWrapperCSS = css`\n  display: flex;\n  box-shadow: none;\n`;\n\ninterface SlideshowItemProps {\n  fadeOver?: boolean;\n}\n\nconst SlideshowItem = styled.div<SlideshowItemProps>`\n  width: 100vw;\n  height: 100vw;\n  /* aspect ratios */\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n  background-color: '#222';\n  background-size: cover;\n  background-position-x: center;\n  background-position-y: center;\n  border: 0;\n  position: ${(props) => (props.fadeOver ? 'absolute' : 'relative')};\n  animation: ${(props) => props.fadeOver && 'fadeIn 400ms ease'};\n  z-index: ${(props) => props.fadeOver && 1};\n  &:before {\n    content: '';\n    opacity: 0.4;\n    background: #091a2a;\n    top: 0;\n    left: 0;\n    bottom: 0;\n    right: 0;\n    position: absolute;\n    z-index: 1;\n  }\n`;\n\ninterface SlideshowLinkProps {\n  out?: boolean;\n}\n\nconst shouldForwardProp = (p: string) => p !== 'out';\n\nconst SlideshowLink = styled(SafeLink, { shouldForwardProp })<SlideshowLinkProps>`\n  display: flex;\n  box-shadow: none;\n  transition: all 400ms ease;\n  opacity: ${(props) => props.out && 0};\n  animation: ${(props) => !props.out && 'fadeInBottomFixed 600ms ease'};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    padding-bottom: ${spacing.medium};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding-bottom: ${spacing.large};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding-bottom: ${spacingUnit * 3}px;\n  }\n  &:hover {\n    ${() => SlideshowName} {\n      text-decoration: underline;\n      text-decoration-color: white;\n    }\n  }\n`;\n\nconst SlideshowWrapper = styled.section`\n  width: 100%;\n  overflow: hidden;\n  &:hover {\n    ${StyledNavigationArrow} {\n      opacity: 1;\n      transform: translate(0, 0);\n    }\n  }\n`;\n\nconst SlideshowInfo = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n  background-color: rgba(3, 23, 43, 0.7);\n  border-radius: 4px;\n  padding: ${spacing.medium} ${spacing.medium} ${spacing.medium} ${spacing.normal};\n  margin: 0 -20px;\n  width: 100vw;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin: 0;\n    width: 100%;\n    padding: ${spacing.medium} ${spacingUnit * 2}px ${spacing.medium} ${spacing.normal};\n  }\n`;\n\nconst SlideshowName = styled.p`\n  ${fonts.sizes('22px', '30px')};\n  color: ${colors.white};\n  text-shadow: 0 2px 4px rgba(0, 0, 0, 0.25);\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('26px', '30px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('40px', '44px')};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    ${fonts.sizes('48px', '54px')};\n  }\n`;\n\nconst SlideshowDescription = styled.p`\n  color: ${colors.white};\n  margin: 0;\n  padding: 0;\n  ${fonts.sizes('12px', '18px')};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('15px', '20px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('18px', '24px')};\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${fonts.sizes('20px', '32px')};\n  }\n`;\n\nconst EmptySlideshow = styled.div`\n  background: rgba(255, 255, 255, 0.08);\n  margin-bottom: $spacing--large * 4;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  height: 40vw;\n`;\n\nconst defaultTransitionSwipeEnd = 'transform 600ms cubic-bezier(0, 0.76, 0.09, 1)';\nconst defaultTransitionText = 'opacity 600ms ease';\n\nconst renderSlideItem = (slide: MovieType) => (\n  <SlideshowItem\n    key={slide.id}\n    role=\"img\"\n    aria-label={(slide.metaImage && slide.metaImage.alt) || ''}\n    style={{\n      backgroundImage: `url(${(slide.metaImage && slide.metaImage.url) || ''})`,\n    }}\n  />\n);\n\nconst FilmSlideshow = ({ autoSlide = false, slideshow = [], slideInterval = 5000 }: Props) => {\n  const [swipeDistance, setSwipeDistance] = useState(0);\n  const [slideIndex, setSlideIndex] = useState(0);\n  const [slideIndexTarget, setSlideIndexTarget] = useState(0);\n  const [animationComplete, setAnimationComplete] = useState(true);\n  const [swipeDirection, setSwipeDirection] = useState<SwipeDirections | undefined>(undefined);\n  const slideRef = useRef<HTMLDivElement>(null);\n  const slideText = useRef<HTMLDivElement>(null);\n  const timer = useRef<ReturnType<typeof setTimeout> | null>(null);\n\n  const gotoSlide = useCallback((indexTarget: number, useAnimation = false) => {\n    setSwipeDistance(0);\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSlideIndexTarget(indexTarget);\n    setAnimationComplete(!useAnimation);\n  }, []);\n\n  const onChangedSlide = () => {\n    if (!animationComplete) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideIndexTarget * 100}vw))`;\n      }\n\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    } else if (slideIndexTarget === -1) {\n      if (slideRef.current) {\n        // Go to last slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideshow.length * 100}vw))`;\n      }\n\n      setSlideIndex(slideshow.length - 1);\n      setSlideIndexTarget(slideshow.length - 1);\n      setAnimationComplete(true);\n    } else if (slideIndexTarget === slideshow.length) {\n      if (slideRef.current) {\n        // Go to first slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(100vw))`;\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n      setAnimationComplete(true);\n    } else {\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    }\n  };\n\n  const onSwipeEnd = () => {\n    setSwipeDirection(undefined);\n    let slide;\n    if (swipeDistance > 40) {\n      slide = -1;\n    } else if (swipeDistance < -40) {\n      slide = 1;\n    } else {\n      slide = 0;\n    }\n    if (slideRef.current && slideText.current) {\n      slideRef.current.style.transition = defaultTransitionSwipeEnd;\n      slideText.current.style.transition = defaultTransitionText;\n      slideText.current.style.opacity = '1';\n    }\n    setSwipeDistance(0);\n\n    initTimer();\n\n    if (slide !== 0) {\n      setSlideIndex(slideIndex + slide);\n      setSlideIndexTarget(slideIndex + slide);\n    } else {\n      // Reset transfrom\n      if (slideRef.current) {\n        slideRef.current.style.transform = getSlidePosition(slideIndex + slide);\n      }\n    }\n  };\n\n  const onSwipe = (eventData: SwipeEventData) => {\n    if (eventData.initial) {\n      setSwipeDirection(eventData.dir);\n    }\n    const dir = eventData.initial ? eventData.dir : swipeDirection;\n    if (dir === 'Up' || dir === 'Down') {\n      return;\n    }\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSwipeDistance(eventData.deltaX);\n    if (slideRef && slideRef.current) {\n      slideRef.current.style.transition = 'none';\n      slideRef.current.style.transform = getSlidePosition(slideIndexTarget);\n    }\n    const opacityText = 1 - Math.min(100, Math.abs(swipeDistance)) / 100;\n    if (slideText && slideText.current) {\n      slideText.current.style.transition = 'none';\n      slideText.current.style.opacity = opacityText.toString();\n    }\n  };\n\n  const onTransitionEnd = () => {\n    const slideshowLength = slideshow.length;\n    if (slideIndex === -1) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(slideshowLength - 1);\n      }\n      setSlideIndex(slideshowLength - 1);\n      setSlideIndexTarget(slideshowLength - 1);\n    } else if (slideIndex >= slideshowLength) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(0);\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n    }\n  };\n\n  const getSlidePosition = (target: number) => {\n    if (swipeDistance !== 0) {\n      return `translateX(calc(${swipeDistance}px -\n        ${(target + 1) * 100}vw))`;\n    }\n    return `translateX(-${(target + 1) * 100}vw)`;\n  };\n\n  const initTimer = useCallback(() => {\n    if (autoSlide) {\n      timer.current = setTimeout(() => {\n        gotoSlide(slideIndex + 1);\n      }, slideInterval);\n    }\n  }, [autoSlide, gotoSlide, slideInterval, slideIndex]);\n\n  useEffect(() => {\n    initTimer();\n  }, [initTimer]);\n\n  const handlers = useSwipeable({\n    onSwiped: onSwipeEnd,\n    onSwiping: onSwipe,\n    preventScrollOnSwipe: swipeDirection === 'Left' || swipeDirection === 'Right',\n  });\n\n  if (slideshow.length === 0) {\n    return (\n      <div>\n        <EmptySlideshow>\n          <Spinner inverted />\n        </EmptySlideshow>\n      </div>\n    );\n  }\n\n  const slideshowWidth = `${(slideshow.length + 2) * 100}vw`;\n  let activeSlide = slideIndex;\n  if (activeSlide < 0) {\n    activeSlide = slideshow.length - 1;\n  } else if (activeSlide >= slideshow.length) {\n    activeSlide = 0;\n  }\n\n  const backgroundImage = slideshow[activeSlide].metaImage;\n\n  return (\n    <SlideshowWrapper {...handlers}>\n      <>\n        <SlideLinkWrapper>\n          <OneColumn>\n            <SlideshowLink to={slideshow[activeSlide].path} out={!animationComplete}>\n              <SlideshowInfo ref={slideText}>\n                <SlideshowName>{slideshow[activeSlide].title}</SlideshowName>\n                <SlideshowDescription>{slideshow[activeSlide].metaDescription}</SlideshowDescription>\n              </SlideshowInfo>\n            </SlideshowLink>\n          </OneColumn>\n        </SlideLinkWrapper>\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget > 0 ? slideIndexTarget - 1 : slideshow.length - 1}\n          gotoSlide={gotoSlide}\n        />\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget < slideshow.length - 1 ? slideIndexTarget + 1 : 0}\n          gotoSlide={gotoSlide}\n          rightArrow\n        />\n        {!animationComplete && (\n          <SlideshowItem\n            fadeOver\n            role=\"img\"\n            onAnimationEnd={onChangedSlide}\n            style={{\n              backgroundImage: `url(${(backgroundImage && backgroundImage.url) || ''})`,\n            }}\n          />\n        )}\n        <div\n          ref={slideRef}\n          css={itemWrapperCSS}\n          onTransitionEnd={onTransitionEnd}\n          style={{\n            width: slideshowWidth,\n            transform: getSlidePosition(slideIndex),\n          }}\n        >\n          {renderSlideItem(slideshow[slideshow.length - 1])}\n          {slideshow.map(renderSlideItem)}\n          {renderSlideItem(slideshow[0])}\n        </div>\n        <SlideshowIndicator slideshow={slideshow} activeSlide={activeSlide} gotoSlide={gotoSlide} />\n      </>\n    </SlideshowWrapper>\n  );\n};\n\nexport default FilmSlideshow;\n"]} */"));
123
- var SlideshowDescription = /*#__PURE__*/(0, _base["default"])("p", {
112
+ }), "{margin-top:-150px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AAwHoC","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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, { useCallback, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Carousel, CarouselAutosize } from '@ndla/carousel';\nimport { breakpoints, colors, misc, mq, spacing, spacingUnit } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { IconButtonV2 } from '@ndla/button';\nimport { ChevronLeft, ChevronRight } from '@ndla/icons/common';\nimport FilmContentCard from './FilmContentCard';\nimport { MovieType } from './types';\n\nexport const slideshowBreakpoints: {\n  until?: keyof typeof breakpoints;\n  columnsPrSlide: number;\n  distanceBetweenItems: number;\n  arrowOffset: number;\n  margin?: number;\n  maxColumnWidth?: number;\n}[] = [\n  {\n    until: 'mobileWide',\n    columnsPrSlide: 2,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'tabletWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'desktop',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'wide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'ultraWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n  {\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n];\n\ninterface Props {\n  slideshow: MovieType[];\n}\n\nconst SlideInfoWrapper = styled.div`\n  position: absolute;\n  color: ${colors.white};\n  max-width: 40%;\n  min-width: 40%;\n  top: 40%;\n  right: 5%;\n  ${mq.range({ until: breakpoints.desktop })} {\n    top: 30%;\n    max-width: 60%;\n    min-width: 60%;\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    max-width: 90%;\n    min-width: 90%;\n    left: 5%;\n  }\n`;\n\nconst StyledSafeLink = styled(SafeLink)`\n  position: relative;\n  display: block;\n  box-shadow: none;\n`;\n\nconst InfoWrapper = styled.div`\n  padding: ${spacing.normal};\n  border-radius: ${misc.borderRadius};\n  border: 0.5px solid ${colors.brand.primary};\n  background-color: rgba(11, 29, 45, 0.8);\n  h3 {\n    margin: 0px;\n  }\n`;\n\nconst StyledImg = styled.img`\n  max-height: 600px;\n  object-position: top;\n  width: 100%;\n  aspect-ratio: 16/9;\n  ${mq.range({ until: breakpoints.tablet })} {\n    min-height: 440px;\n    max-height: 440px;\n  }\n  object-fit: cover;\n`;\n\nconst CarouselContainer = styled.div`\n  margin-top: -50px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    margin-top: -70px;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin-top: -150px;\n  }\n`;\n\ninterface StyledFilmContentCardProps {\n  current?: boolean;\n}\n\nconst SlideshowButton = styled(IconButtonV2)`\n  margin-top: ${spacing.normal};\n`;\n\nconst shouldForwardProp = (p: string) => p !== 'current';\n\nconst StyledFilmContentCard = styled(FilmContentCard, { shouldForwardProp })<StyledFilmContentCardProps>`\n  margin-bottom: 2%;\n  transform: ${(p) => (p.current ? 'translateY(0%)' : 'translateY(10%)')};\n  transition: all 200ms;\n`;\n\nconst FilmSlideshow = ({ slideshow }: Props) => {\n  const [currentSlide, setCurrentSlide] = useState(slideshow[0]);\n\n  return (\n    <CarouselAutosize breakpoints={slideshowBreakpoints} itemsLength={slideshow.length}>\n      {(autoSizedProps) => (\n        <section>\n          <StyledSafeLink to={currentSlide.path} tabIndex={-1} aria-hidden>\n            <StyledImg src={currentSlide.metaImage?.url ?? ''} alt={currentSlide.metaImage?.alt ?? ''} />\n            <SlideInfoWrapper>\n              <InfoWrapper>\n                <h3>{currentSlide.title}</h3>\n                <span id=\"currentMovieDescription\">{currentSlide.metaDescription}</span>\n              </InfoWrapper>\n            </SlideInfoWrapper>\n          </StyledSafeLink>\n          <CarouselContainer>\n            <Carousel\n              leftButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronLeft />\n                </SlideshowButton>\n              }\n              rightButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronRight />\n                </SlideshowButton>\n              }\n              items={slideshow.map((movie) => (\n                <FilmCard\n                  key={movie.id}\n                  current={movie.id === currentSlide.id}\n                  movie={movie}\n                  columnWidth={autoSizedProps.columnWidth}\n                  setCurrentSlide={() => setCurrentSlide(movie)}\n                />\n              ))}\n              {...autoSizedProps}\n            />\n          </CarouselContainer>\n        </section>\n      )}\n    </CarouselAutosize>\n  );\n};\n\ninterface FilmCardProps {\n  setCurrentSlide: () => void;\n  movie: MovieType;\n  current: boolean;\n  columnWidth: number;\n}\n\nconst FilmCard = ({ setCurrentSlide, movie, current, columnWidth }: FilmCardProps) => {\n  const [hoverCallback, setHoverCallback] = useState<ReturnType<typeof setTimeout> | undefined>(undefined);\n\n  const onHover = useCallback(() => {\n    const timeout = setTimeout(() => setCurrentSlide(), 500);\n    setHoverCallback(timeout);\n  }, [setCurrentSlide]);\n\n  return (\n    <StyledFilmContentCard\n      onMouseEnter={onHover}\n      onMouseLeave={() => {\n        if (hoverCallback) {\n          clearTimeout(hoverCallback);\n          setHoverCallback(undefined);\n        }\n      }}\n      onFocus={() => setCurrentSlide()}\n      current={current}\n      aria-describedby={'currentMovieDescription'}\n      key={movie.id}\n      movie={movie}\n      columnWidth={columnWidth}\n      resourceTypes={[]}\n    />\n  );\n};\n\nexport default FilmSlideshow;\n"]} */"));
113
+ var SlideshowButton = /*#__PURE__*/(0, _base["default"])(_button.IconButtonV2, {
124
114
  target: "essc37o1",
125
- label: "SlideshowDescription"
126
- })("color:", _core.colors.white, ";margin:0;padding:0;", _core.fonts.sizes('12px', '18px'), ";", _core.mq.range({
127
- from: _core.breakpoints.mobileWide
128
- }), "{", _core.fonts.sizes('15px', '20px'), ";}", _core.mq.range({
129
- from: _core.breakpoints.tablet
130
- }), "{", _core.fonts.sizes('18px', '24px'), ";}", _core.mq.range({
131
- from: _core.breakpoints.wide
132
- }), "{", _core.fonts.sizes('20px', '32px'), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AA+KqC","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2016-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, { useCallback, useEffect, useRef, useState } from 'react';\nimport { SwipeDirections, SwipeEventData, useSwipeable } from 'react-swipeable';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { breakpoints, mq, spacing, spacingUnit, fonts, colors } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { Spinner } from '@ndla/icons';\nimport { OneColumn } from '../Layout';\nimport NavigationArrow, { StyledNavigationArrow } from './NavigationArrow';\nimport SlideshowIndicator from './SlideshowIndicator';\nimport { MovieType } from './types';\n\ninterface Props {\n  autoSlide?: boolean;\n  randomStart?: boolean;\n  slideshow: MovieType[];\n  slideInterval?: number;\n}\n\nconst SlideLinkWrapper = styled.div`\n  margin: 0 auto;\n  display: flex;\n  justify-content: flex-start;\n  align-items: flex-end;\n  position: absolute;\n  z-index: 2;\n  height: 100vw;\n  width: 100%;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n`;\n\nconst itemWrapperCSS = css`\n  display: flex;\n  box-shadow: none;\n`;\n\ninterface SlideshowItemProps {\n  fadeOver?: boolean;\n}\n\nconst SlideshowItem = styled.div<SlideshowItemProps>`\n  width: 100vw;\n  height: 100vw;\n  /* aspect ratios */\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n  background-color: '#222';\n  background-size: cover;\n  background-position-x: center;\n  background-position-y: center;\n  border: 0;\n  position: ${(props) => (props.fadeOver ? 'absolute' : 'relative')};\n  animation: ${(props) => props.fadeOver && 'fadeIn 400ms ease'};\n  z-index: ${(props) => props.fadeOver && 1};\n  &:before {\n    content: '';\n    opacity: 0.4;\n    background: #091a2a;\n    top: 0;\n    left: 0;\n    bottom: 0;\n    right: 0;\n    position: absolute;\n    z-index: 1;\n  }\n`;\n\ninterface SlideshowLinkProps {\n  out?: boolean;\n}\n\nconst shouldForwardProp = (p: string) => p !== 'out';\n\nconst SlideshowLink = styled(SafeLink, { shouldForwardProp })<SlideshowLinkProps>`\n  display: flex;\n  box-shadow: none;\n  transition: all 400ms ease;\n  opacity: ${(props) => props.out && 0};\n  animation: ${(props) => !props.out && 'fadeInBottomFixed 600ms ease'};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    padding-bottom: ${spacing.medium};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding-bottom: ${spacing.large};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding-bottom: ${spacingUnit * 3}px;\n  }\n  &:hover {\n    ${() => SlideshowName} {\n      text-decoration: underline;\n      text-decoration-color: white;\n    }\n  }\n`;\n\nconst SlideshowWrapper = styled.section`\n  width: 100%;\n  overflow: hidden;\n  &:hover {\n    ${StyledNavigationArrow} {\n      opacity: 1;\n      transform: translate(0, 0);\n    }\n  }\n`;\n\nconst SlideshowInfo = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n  background-color: rgba(3, 23, 43, 0.7);\n  border-radius: 4px;\n  padding: ${spacing.medium} ${spacing.medium} ${spacing.medium} ${spacing.normal};\n  margin: 0 -20px;\n  width: 100vw;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin: 0;\n    width: 100%;\n    padding: ${spacing.medium} ${spacingUnit * 2}px ${spacing.medium} ${spacing.normal};\n  }\n`;\n\nconst SlideshowName = styled.p`\n  ${fonts.sizes('22px', '30px')};\n  color: ${colors.white};\n  text-shadow: 0 2px 4px rgba(0, 0, 0, 0.25);\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('26px', '30px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('40px', '44px')};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    ${fonts.sizes('48px', '54px')};\n  }\n`;\n\nconst SlideshowDescription = styled.p`\n  color: ${colors.white};\n  margin: 0;\n  padding: 0;\n  ${fonts.sizes('12px', '18px')};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('15px', '20px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('18px', '24px')};\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${fonts.sizes('20px', '32px')};\n  }\n`;\n\nconst EmptySlideshow = styled.div`\n  background: rgba(255, 255, 255, 0.08);\n  margin-bottom: $spacing--large * 4;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  height: 40vw;\n`;\n\nconst defaultTransitionSwipeEnd = 'transform 600ms cubic-bezier(0, 0.76, 0.09, 1)';\nconst defaultTransitionText = 'opacity 600ms ease';\n\nconst renderSlideItem = (slide: MovieType) => (\n  <SlideshowItem\n    key={slide.id}\n    role=\"img\"\n    aria-label={(slide.metaImage && slide.metaImage.alt) || ''}\n    style={{\n      backgroundImage: `url(${(slide.metaImage && slide.metaImage.url) || ''})`,\n    }}\n  />\n);\n\nconst FilmSlideshow = ({ autoSlide = false, slideshow = [], slideInterval = 5000 }: Props) => {\n  const [swipeDistance, setSwipeDistance] = useState(0);\n  const [slideIndex, setSlideIndex] = useState(0);\n  const [slideIndexTarget, setSlideIndexTarget] = useState(0);\n  const [animationComplete, setAnimationComplete] = useState(true);\n  const [swipeDirection, setSwipeDirection] = useState<SwipeDirections | undefined>(undefined);\n  const slideRef = useRef<HTMLDivElement>(null);\n  const slideText = useRef<HTMLDivElement>(null);\n  const timer = useRef<ReturnType<typeof setTimeout> | null>(null);\n\n  const gotoSlide = useCallback((indexTarget: number, useAnimation = false) => {\n    setSwipeDistance(0);\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSlideIndexTarget(indexTarget);\n    setAnimationComplete(!useAnimation);\n  }, []);\n\n  const onChangedSlide = () => {\n    if (!animationComplete) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideIndexTarget * 100}vw))`;\n      }\n\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    } else if (slideIndexTarget === -1) {\n      if (slideRef.current) {\n        // Go to last slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideshow.length * 100}vw))`;\n      }\n\n      setSlideIndex(slideshow.length - 1);\n      setSlideIndexTarget(slideshow.length - 1);\n      setAnimationComplete(true);\n    } else if (slideIndexTarget === slideshow.length) {\n      if (slideRef.current) {\n        // Go to first slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(100vw))`;\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n      setAnimationComplete(true);\n    } else {\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    }\n  };\n\n  const onSwipeEnd = () => {\n    setSwipeDirection(undefined);\n    let slide;\n    if (swipeDistance > 40) {\n      slide = -1;\n    } else if (swipeDistance < -40) {\n      slide = 1;\n    } else {\n      slide = 0;\n    }\n    if (slideRef.current && slideText.current) {\n      slideRef.current.style.transition = defaultTransitionSwipeEnd;\n      slideText.current.style.transition = defaultTransitionText;\n      slideText.current.style.opacity = '1';\n    }\n    setSwipeDistance(0);\n\n    initTimer();\n\n    if (slide !== 0) {\n      setSlideIndex(slideIndex + slide);\n      setSlideIndexTarget(slideIndex + slide);\n    } else {\n      // Reset transfrom\n      if (slideRef.current) {\n        slideRef.current.style.transform = getSlidePosition(slideIndex + slide);\n      }\n    }\n  };\n\n  const onSwipe = (eventData: SwipeEventData) => {\n    if (eventData.initial) {\n      setSwipeDirection(eventData.dir);\n    }\n    const dir = eventData.initial ? eventData.dir : swipeDirection;\n    if (dir === 'Up' || dir === 'Down') {\n      return;\n    }\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSwipeDistance(eventData.deltaX);\n    if (slideRef && slideRef.current) {\n      slideRef.current.style.transition = 'none';\n      slideRef.current.style.transform = getSlidePosition(slideIndexTarget);\n    }\n    const opacityText = 1 - Math.min(100, Math.abs(swipeDistance)) / 100;\n    if (slideText && slideText.current) {\n      slideText.current.style.transition = 'none';\n      slideText.current.style.opacity = opacityText.toString();\n    }\n  };\n\n  const onTransitionEnd = () => {\n    const slideshowLength = slideshow.length;\n    if (slideIndex === -1) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(slideshowLength - 1);\n      }\n      setSlideIndex(slideshowLength - 1);\n      setSlideIndexTarget(slideshowLength - 1);\n    } else if (slideIndex >= slideshowLength) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(0);\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n    }\n  };\n\n  const getSlidePosition = (target: number) => {\n    if (swipeDistance !== 0) {\n      return `translateX(calc(${swipeDistance}px -\n        ${(target + 1) * 100}vw))`;\n    }\n    return `translateX(-${(target + 1) * 100}vw)`;\n  };\n\n  const initTimer = useCallback(() => {\n    if (autoSlide) {\n      timer.current = setTimeout(() => {\n        gotoSlide(slideIndex + 1);\n      }, slideInterval);\n    }\n  }, [autoSlide, gotoSlide, slideInterval, slideIndex]);\n\n  useEffect(() => {\n    initTimer();\n  }, [initTimer]);\n\n  const handlers = useSwipeable({\n    onSwiped: onSwipeEnd,\n    onSwiping: onSwipe,\n    preventScrollOnSwipe: swipeDirection === 'Left' || swipeDirection === 'Right',\n  });\n\n  if (slideshow.length === 0) {\n    return (\n      <div>\n        <EmptySlideshow>\n          <Spinner inverted />\n        </EmptySlideshow>\n      </div>\n    );\n  }\n\n  const slideshowWidth = `${(slideshow.length + 2) * 100}vw`;\n  let activeSlide = slideIndex;\n  if (activeSlide < 0) {\n    activeSlide = slideshow.length - 1;\n  } else if (activeSlide >= slideshow.length) {\n    activeSlide = 0;\n  }\n\n  const backgroundImage = slideshow[activeSlide].metaImage;\n\n  return (\n    <SlideshowWrapper {...handlers}>\n      <>\n        <SlideLinkWrapper>\n          <OneColumn>\n            <SlideshowLink to={slideshow[activeSlide].path} out={!animationComplete}>\n              <SlideshowInfo ref={slideText}>\n                <SlideshowName>{slideshow[activeSlide].title}</SlideshowName>\n                <SlideshowDescription>{slideshow[activeSlide].metaDescription}</SlideshowDescription>\n              </SlideshowInfo>\n            </SlideshowLink>\n          </OneColumn>\n        </SlideLinkWrapper>\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget > 0 ? slideIndexTarget - 1 : slideshow.length - 1}\n          gotoSlide={gotoSlide}\n        />\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget < slideshow.length - 1 ? slideIndexTarget + 1 : 0}\n          gotoSlide={gotoSlide}\n          rightArrow\n        />\n        {!animationComplete && (\n          <SlideshowItem\n            fadeOver\n            role=\"img\"\n            onAnimationEnd={onChangedSlide}\n            style={{\n              backgroundImage: `url(${(backgroundImage && backgroundImage.url) || ''})`,\n            }}\n          />\n        )}\n        <div\n          ref={slideRef}\n          css={itemWrapperCSS}\n          onTransitionEnd={onTransitionEnd}\n          style={{\n            width: slideshowWidth,\n            transform: getSlidePosition(slideIndex),\n          }}\n        >\n          {renderSlideItem(slideshow[slideshow.length - 1])}\n          {slideshow.map(renderSlideItem)}\n          {renderSlideItem(slideshow[0])}\n        </div>\n        <SlideshowIndicator slideshow={slideshow} activeSlide={activeSlide} gotoSlide={gotoSlide} />\n      </>\n    </SlideshowWrapper>\n  );\n};\n\nexport default FilmSlideshow;\n"]} */"));
133
- var EmptySlideshow = /*#__PURE__*/(0, _base["default"])("div", {
134
- target: "essc37o0",
135
- label: "EmptySlideshow"
136
- })(process.env.NODE_ENV === "production" ? {
137
- name: "hilalp",
138
- styles: "background:rgba(255, 255, 255, 0.08);margin-bottom:$spacing--large * 4;display:flex;align-items:center;justify-content:center;height:40vw"
139
- } : {
140
- name: "hilalp",
141
- styles: "background:rgba(255, 255, 255, 0.08);margin-bottom:$spacing--large * 4;display:flex;align-items:center;justify-content:center;height:40vw",
142
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AA+LiC","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2016-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, { useCallback, useEffect, useRef, useState } from 'react';\nimport { SwipeDirections, SwipeEventData, useSwipeable } from 'react-swipeable';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { breakpoints, mq, spacing, spacingUnit, fonts, colors } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { Spinner } from '@ndla/icons';\nimport { OneColumn } from '../Layout';\nimport NavigationArrow, { StyledNavigationArrow } from './NavigationArrow';\nimport SlideshowIndicator from './SlideshowIndicator';\nimport { MovieType } from './types';\n\ninterface Props {\n  autoSlide?: boolean;\n  randomStart?: boolean;\n  slideshow: MovieType[];\n  slideInterval?: number;\n}\n\nconst SlideLinkWrapper = styled.div`\n  margin: 0 auto;\n  display: flex;\n  justify-content: flex-start;\n  align-items: flex-end;\n  position: absolute;\n  z-index: 2;\n  height: 100vw;\n  width: 100%;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n`;\n\nconst itemWrapperCSS = css`\n  display: flex;\n  box-shadow: none;\n`;\n\ninterface SlideshowItemProps {\n  fadeOver?: boolean;\n}\n\nconst SlideshowItem = styled.div<SlideshowItemProps>`\n  width: 100vw;\n  height: 100vw;\n  /* aspect ratios */\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    height: 100vw;\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    height: 75vw;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    height: 55vw;\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    height: 40vw;\n  }\n  ${mq.range({ from: breakpoints.ultraWide })} {\n    height: 36vw;\n  }\n  background-color: '#222';\n  background-size: cover;\n  background-position-x: center;\n  background-position-y: center;\n  border: 0;\n  position: ${(props) => (props.fadeOver ? 'absolute' : 'relative')};\n  animation: ${(props) => props.fadeOver && 'fadeIn 400ms ease'};\n  z-index: ${(props) => props.fadeOver && 1};\n  &:before {\n    content: '';\n    opacity: 0.4;\n    background: #091a2a;\n    top: 0;\n    left: 0;\n    bottom: 0;\n    right: 0;\n    position: absolute;\n    z-index: 1;\n  }\n`;\n\ninterface SlideshowLinkProps {\n  out?: boolean;\n}\n\nconst shouldForwardProp = (p: string) => p !== 'out';\n\nconst SlideshowLink = styled(SafeLink, { shouldForwardProp })<SlideshowLinkProps>`\n  display: flex;\n  box-shadow: none;\n  transition: all 400ms ease;\n  opacity: ${(props) => props.out && 0};\n  animation: ${(props) => !props.out && 'fadeInBottomFixed 600ms ease'};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    padding-bottom: ${spacing.medium};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    padding-bottom: ${spacing.large};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    padding-bottom: ${spacingUnit * 3}px;\n  }\n  &:hover {\n    ${() => SlideshowName} {\n      text-decoration: underline;\n      text-decoration-color: white;\n    }\n  }\n`;\n\nconst SlideshowWrapper = styled.section`\n  width: 100%;\n  overflow: hidden;\n  &:hover {\n    ${StyledNavigationArrow} {\n      opacity: 1;\n      transform: translate(0, 0);\n    }\n  }\n`;\n\nconst SlideshowInfo = styled.div`\n  display: flex;\n  flex-direction: column;\n  gap: ${spacing.small};\n  background-color: rgba(3, 23, 43, 0.7);\n  border-radius: 4px;\n  padding: ${spacing.medium} ${spacing.medium} ${spacing.medium} ${spacing.normal};\n  margin: 0 -20px;\n  width: 100vw;\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    margin: 0;\n    width: 100%;\n    padding: ${spacing.medium} ${spacingUnit * 2}px ${spacing.medium} ${spacing.normal};\n  }\n`;\n\nconst SlideshowName = styled.p`\n  ${fonts.sizes('22px', '30px')};\n  color: ${colors.white};\n  text-shadow: 0 2px 4px rgba(0, 0, 0, 0.25);\n  margin: 0;\n  font-weight: ${fonts.weight.semibold};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('26px', '30px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('40px', '44px')};\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    ${fonts.sizes('48px', '54px')};\n  }\n`;\n\nconst SlideshowDescription = styled.p`\n  color: ${colors.white};\n  margin: 0;\n  padding: 0;\n  ${fonts.sizes('12px', '18px')};\n  ${mq.range({ from: breakpoints.mobileWide })} {\n    ${fonts.sizes('15px', '20px')};\n  }\n  ${mq.range({ from: breakpoints.tablet })} {\n    ${fonts.sizes('18px', '24px')};\n  }\n  ${mq.range({ from: breakpoints.wide })} {\n    ${fonts.sizes('20px', '32px')};\n  }\n`;\n\nconst EmptySlideshow = styled.div`\n  background: rgba(255, 255, 255, 0.08);\n  margin-bottom: $spacing--large * 4;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  height: 40vw;\n`;\n\nconst defaultTransitionSwipeEnd = 'transform 600ms cubic-bezier(0, 0.76, 0.09, 1)';\nconst defaultTransitionText = 'opacity 600ms ease';\n\nconst renderSlideItem = (slide: MovieType) => (\n  <SlideshowItem\n    key={slide.id}\n    role=\"img\"\n    aria-label={(slide.metaImage && slide.metaImage.alt) || ''}\n    style={{\n      backgroundImage: `url(${(slide.metaImage && slide.metaImage.url) || ''})`,\n    }}\n  />\n);\n\nconst FilmSlideshow = ({ autoSlide = false, slideshow = [], slideInterval = 5000 }: Props) => {\n  const [swipeDistance, setSwipeDistance] = useState(0);\n  const [slideIndex, setSlideIndex] = useState(0);\n  const [slideIndexTarget, setSlideIndexTarget] = useState(0);\n  const [animationComplete, setAnimationComplete] = useState(true);\n  const [swipeDirection, setSwipeDirection] = useState<SwipeDirections | undefined>(undefined);\n  const slideRef = useRef<HTMLDivElement>(null);\n  const slideText = useRef<HTMLDivElement>(null);\n  const timer = useRef<ReturnType<typeof setTimeout> | null>(null);\n\n  const gotoSlide = useCallback((indexTarget: number, useAnimation = false) => {\n    setSwipeDistance(0);\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSlideIndexTarget(indexTarget);\n    setAnimationComplete(!useAnimation);\n  }, []);\n\n  const onChangedSlide = () => {\n    if (!animationComplete) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideIndexTarget * 100}vw))`;\n      }\n\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    } else if (slideIndexTarget === -1) {\n      if (slideRef.current) {\n        // Go to last slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(${slideshow.length * 100}vw))`;\n      }\n\n      setSlideIndex(slideshow.length - 1);\n      setSlideIndexTarget(slideshow.length - 1);\n      setAnimationComplete(true);\n    } else if (slideIndexTarget === slideshow.length) {\n      if (slideRef.current) {\n        // Go to first slide for continuous loop\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = `translateX(100vw))`;\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n      setAnimationComplete(true);\n    } else {\n      setAnimationComplete(true);\n      setSlideIndex(slideIndexTarget);\n    }\n  };\n\n  const onSwipeEnd = () => {\n    setSwipeDirection(undefined);\n    let slide;\n    if (swipeDistance > 40) {\n      slide = -1;\n    } else if (swipeDistance < -40) {\n      slide = 1;\n    } else {\n      slide = 0;\n    }\n    if (slideRef.current && slideText.current) {\n      slideRef.current.style.transition = defaultTransitionSwipeEnd;\n      slideText.current.style.transition = defaultTransitionText;\n      slideText.current.style.opacity = '1';\n    }\n    setSwipeDistance(0);\n\n    initTimer();\n\n    if (slide !== 0) {\n      setSlideIndex(slideIndex + slide);\n      setSlideIndexTarget(slideIndex + slide);\n    } else {\n      // Reset transfrom\n      if (slideRef.current) {\n        slideRef.current.style.transform = getSlidePosition(slideIndex + slide);\n      }\n    }\n  };\n\n  const onSwipe = (eventData: SwipeEventData) => {\n    if (eventData.initial) {\n      setSwipeDirection(eventData.dir);\n    }\n    const dir = eventData.initial ? eventData.dir : swipeDirection;\n    if (dir === 'Up' || dir === 'Down') {\n      return;\n    }\n    if (timer.current) {\n      clearTimeout(timer.current);\n    }\n    setSwipeDistance(eventData.deltaX);\n    if (slideRef && slideRef.current) {\n      slideRef.current.style.transition = 'none';\n      slideRef.current.style.transform = getSlidePosition(slideIndexTarget);\n    }\n    const opacityText = 1 - Math.min(100, Math.abs(swipeDistance)) / 100;\n    if (slideText && slideText.current) {\n      slideText.current.style.transition = 'none';\n      slideText.current.style.opacity = opacityText.toString();\n    }\n  };\n\n  const onTransitionEnd = () => {\n    const slideshowLength = slideshow.length;\n    if (slideIndex === -1) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(slideshowLength - 1);\n      }\n      setSlideIndex(slideshowLength - 1);\n      setSlideIndexTarget(slideshowLength - 1);\n    } else if (slideIndex >= slideshowLength) {\n      if (slideRef.current) {\n        slideRef.current.style.transition = 'none';\n        slideRef.current.style.transform = getSlidePosition(0);\n      }\n      setSlideIndex(0);\n      setSlideIndexTarget(0);\n    }\n  };\n\n  const getSlidePosition = (target: number) => {\n    if (swipeDistance !== 0) {\n      return `translateX(calc(${swipeDistance}px -\n        ${(target + 1) * 100}vw))`;\n    }\n    return `translateX(-${(target + 1) * 100}vw)`;\n  };\n\n  const initTimer = useCallback(() => {\n    if (autoSlide) {\n      timer.current = setTimeout(() => {\n        gotoSlide(slideIndex + 1);\n      }, slideInterval);\n    }\n  }, [autoSlide, gotoSlide, slideInterval, slideIndex]);\n\n  useEffect(() => {\n    initTimer();\n  }, [initTimer]);\n\n  const handlers = useSwipeable({\n    onSwiped: onSwipeEnd,\n    onSwiping: onSwipe,\n    preventScrollOnSwipe: swipeDirection === 'Left' || swipeDirection === 'Right',\n  });\n\n  if (slideshow.length === 0) {\n    return (\n      <div>\n        <EmptySlideshow>\n          <Spinner inverted />\n        </EmptySlideshow>\n      </div>\n    );\n  }\n\n  const slideshowWidth = `${(slideshow.length + 2) * 100}vw`;\n  let activeSlide = slideIndex;\n  if (activeSlide < 0) {\n    activeSlide = slideshow.length - 1;\n  } else if (activeSlide >= slideshow.length) {\n    activeSlide = 0;\n  }\n\n  const backgroundImage = slideshow[activeSlide].metaImage;\n\n  return (\n    <SlideshowWrapper {...handlers}>\n      <>\n        <SlideLinkWrapper>\n          <OneColumn>\n            <SlideshowLink to={slideshow[activeSlide].path} out={!animationComplete}>\n              <SlideshowInfo ref={slideText}>\n                <SlideshowName>{slideshow[activeSlide].title}</SlideshowName>\n                <SlideshowDescription>{slideshow[activeSlide].metaDescription}</SlideshowDescription>\n              </SlideshowInfo>\n            </SlideshowLink>\n          </OneColumn>\n        </SlideLinkWrapper>\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget > 0 ? slideIndexTarget - 1 : slideshow.length - 1}\n          gotoSlide={gotoSlide}\n        />\n        <NavigationArrow\n          slideIndexTarget={slideIndexTarget < slideshow.length - 1 ? slideIndexTarget + 1 : 0}\n          gotoSlide={gotoSlide}\n          rightArrow\n        />\n        {!animationComplete && (\n          <SlideshowItem\n            fadeOver\n            role=\"img\"\n            onAnimationEnd={onChangedSlide}\n            style={{\n              backgroundImage: `url(${(backgroundImage && backgroundImage.url) || ''})`,\n            }}\n          />\n        )}\n        <div\n          ref={slideRef}\n          css={itemWrapperCSS}\n          onTransitionEnd={onTransitionEnd}\n          style={{\n            width: slideshowWidth,\n            transform: getSlidePosition(slideIndex),\n          }}\n        >\n          {renderSlideItem(slideshow[slideshow.length - 1])}\n          {slideshow.map(renderSlideItem)}\n          {renderSlideItem(slideshow[0])}\n        </div>\n        <SlideshowIndicator slideshow={slideshow} activeSlide={activeSlide} gotoSlide={gotoSlide} />\n      </>\n    </SlideshowWrapper>\n  );\n};\n\nexport default FilmSlideshow;\n"]} */",
143
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
144
- });
145
- var defaultTransitionSwipeEnd = 'transform 600ms cubic-bezier(0, 0.76, 0.09, 1)';
146
- var defaultTransitionText = 'opacity 600ms ease';
147
- var renderSlideItem = function renderSlideItem(slide) {
148
- return (0, _jsxRuntime.jsx)(SlideshowItem, {
149
- role: "img",
150
- "aria-label": slide.metaImage && slide.metaImage.alt || '',
151
- style: {
152
- backgroundImage: "url(".concat(slide.metaImage && slide.metaImage.url || '', ")")
153
- }
154
- }, slide.id);
115
+ label: "SlideshowButton"
116
+ })("margin-top:", _core.spacing.normal, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AAsI4C","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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, { useCallback, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Carousel, CarouselAutosize } from '@ndla/carousel';\nimport { breakpoints, colors, misc, mq, spacing, spacingUnit } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { IconButtonV2 } from '@ndla/button';\nimport { ChevronLeft, ChevronRight } from '@ndla/icons/common';\nimport FilmContentCard from './FilmContentCard';\nimport { MovieType } from './types';\n\nexport const slideshowBreakpoints: {\n  until?: keyof typeof breakpoints;\n  columnsPrSlide: number;\n  distanceBetweenItems: number;\n  arrowOffset: number;\n  margin?: number;\n  maxColumnWidth?: number;\n}[] = [\n  {\n    until: 'mobileWide',\n    columnsPrSlide: 2,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'tabletWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'desktop',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'wide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'ultraWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n  {\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n];\n\ninterface Props {\n  slideshow: MovieType[];\n}\n\nconst SlideInfoWrapper = styled.div`\n  position: absolute;\n  color: ${colors.white};\n  max-width: 40%;\n  min-width: 40%;\n  top: 40%;\n  right: 5%;\n  ${mq.range({ until: breakpoints.desktop })} {\n    top: 30%;\n    max-width: 60%;\n    min-width: 60%;\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    max-width: 90%;\n    min-width: 90%;\n    left: 5%;\n  }\n`;\n\nconst StyledSafeLink = styled(SafeLink)`\n  position: relative;\n  display: block;\n  box-shadow: none;\n`;\n\nconst InfoWrapper = styled.div`\n  padding: ${spacing.normal};\n  border-radius: ${misc.borderRadius};\n  border: 0.5px solid ${colors.brand.primary};\n  background-color: rgba(11, 29, 45, 0.8);\n  h3 {\n    margin: 0px;\n  }\n`;\n\nconst StyledImg = styled.img`\n  max-height: 600px;\n  object-position: top;\n  width: 100%;\n  aspect-ratio: 16/9;\n  ${mq.range({ until: breakpoints.tablet })} {\n    min-height: 440px;\n    max-height: 440px;\n  }\n  object-fit: cover;\n`;\n\nconst CarouselContainer = styled.div`\n  margin-top: -50px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    margin-top: -70px;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin-top: -150px;\n  }\n`;\n\ninterface StyledFilmContentCardProps {\n  current?: boolean;\n}\n\nconst SlideshowButton = styled(IconButtonV2)`\n  margin-top: ${spacing.normal};\n`;\n\nconst shouldForwardProp = (p: string) => p !== 'current';\n\nconst StyledFilmContentCard = styled(FilmContentCard, { shouldForwardProp })<StyledFilmContentCardProps>`\n  margin-bottom: 2%;\n  transform: ${(p) => (p.current ? 'translateY(0%)' : 'translateY(10%)')};\n  transition: all 200ms;\n`;\n\nconst FilmSlideshow = ({ slideshow }: Props) => {\n  const [currentSlide, setCurrentSlide] = useState(slideshow[0]);\n\n  return (\n    <CarouselAutosize breakpoints={slideshowBreakpoints} itemsLength={slideshow.length}>\n      {(autoSizedProps) => (\n        <section>\n          <StyledSafeLink to={currentSlide.path} tabIndex={-1} aria-hidden>\n            <StyledImg src={currentSlide.metaImage?.url ?? ''} alt={currentSlide.metaImage?.alt ?? ''} />\n            <SlideInfoWrapper>\n              <InfoWrapper>\n                <h3>{currentSlide.title}</h3>\n                <span id=\"currentMovieDescription\">{currentSlide.metaDescription}</span>\n              </InfoWrapper>\n            </SlideInfoWrapper>\n          </StyledSafeLink>\n          <CarouselContainer>\n            <Carousel\n              leftButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronLeft />\n                </SlideshowButton>\n              }\n              rightButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronRight />\n                </SlideshowButton>\n              }\n              items={slideshow.map((movie) => (\n                <FilmCard\n                  key={movie.id}\n                  current={movie.id === currentSlide.id}\n                  movie={movie}\n                  columnWidth={autoSizedProps.columnWidth}\n                  setCurrentSlide={() => setCurrentSlide(movie)}\n                />\n              ))}\n              {...autoSizedProps}\n            />\n          </CarouselContainer>\n        </section>\n      )}\n    </CarouselAutosize>\n  );\n};\n\ninterface FilmCardProps {\n  setCurrentSlide: () => void;\n  movie: MovieType;\n  current: boolean;\n  columnWidth: number;\n}\n\nconst FilmCard = ({ setCurrentSlide, movie, current, columnWidth }: FilmCardProps) => {\n  const [hoverCallback, setHoverCallback] = useState<ReturnType<typeof setTimeout> | undefined>(undefined);\n\n  const onHover = useCallback(() => {\n    const timeout = setTimeout(() => setCurrentSlide(), 500);\n    setHoverCallback(timeout);\n  }, [setCurrentSlide]);\n\n  return (\n    <StyledFilmContentCard\n      onMouseEnter={onHover}\n      onMouseLeave={() => {\n        if (hoverCallback) {\n          clearTimeout(hoverCallback);\n          setHoverCallback(undefined);\n        }\n      }}\n      onFocus={() => setCurrentSlide()}\n      current={current}\n      aria-describedby={'currentMovieDescription'}\n      key={movie.id}\n      movie={movie}\n      columnWidth={columnWidth}\n      resourceTypes={[]}\n    />\n  );\n};\n\nexport default FilmSlideshow;\n"]} */"));
117
+ var shouldForwardProp = function shouldForwardProp(p) {
118
+ return p !== 'current';
155
119
  };
120
+ var StyledFilmContentCard = /*#__PURE__*/(0, _base["default"])(_FilmContentCard["default"], {
121
+ shouldForwardProp: shouldForwardProp,
122
+ target: "essc37o0",
123
+ label: "StyledFilmContentCard"
124
+ })("margin-bottom:2%;transform:", function (p) {
125
+ return p.current ? 'translateY(0%)' : 'translateY(10%)';
126
+ }, ";transition:all 200ms;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["FilmSlideshow.tsx"],"names":[],"mappings":"AA4IwG","file":"FilmSlideshow.tsx","sourcesContent":["/**\n * Copyright (c) 2023-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, { useCallback, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Carousel, CarouselAutosize } from '@ndla/carousel';\nimport { breakpoints, colors, misc, mq, spacing, spacingUnit } from '@ndla/core';\nimport SafeLink from '@ndla/safelink';\nimport { IconButtonV2 } from '@ndla/button';\nimport { ChevronLeft, ChevronRight } from '@ndla/icons/common';\nimport FilmContentCard from './FilmContentCard';\nimport { MovieType } from './types';\n\nexport const slideshowBreakpoints: {\n  until?: keyof typeof breakpoints;\n  columnsPrSlide: number;\n  distanceBetweenItems: number;\n  arrowOffset: number;\n  margin?: number;\n  maxColumnWidth?: number;\n}[] = [\n  {\n    until: 'mobileWide',\n    columnsPrSlide: 2,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'tabletWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit / 2,\n    margin: spacingUnit,\n    arrowOffset: 13,\n  },\n  {\n    until: 'desktop',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'wide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 2,\n    arrowOffset: 0,\n  },\n  {\n    until: 'ultraWide',\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n  {\n    columnsPrSlide: 3,\n    distanceBetweenItems: spacingUnit,\n    margin: spacingUnit * 3.5,\n    arrowOffset: 0,\n  },\n];\n\ninterface Props {\n  slideshow: MovieType[];\n}\n\nconst SlideInfoWrapper = styled.div`\n  position: absolute;\n  color: ${colors.white};\n  max-width: 40%;\n  min-width: 40%;\n  top: 40%;\n  right: 5%;\n  ${mq.range({ until: breakpoints.desktop })} {\n    top: 30%;\n    max-width: 60%;\n    min-width: 60%;\n  }\n  ${mq.range({ until: breakpoints.tablet })} {\n    max-width: 90%;\n    min-width: 90%;\n    left: 5%;\n  }\n`;\n\nconst StyledSafeLink = styled(SafeLink)`\n  position: relative;\n  display: block;\n  box-shadow: none;\n`;\n\nconst InfoWrapper = styled.div`\n  padding: ${spacing.normal};\n  border-radius: ${misc.borderRadius};\n  border: 0.5px solid ${colors.brand.primary};\n  background-color: rgba(11, 29, 45, 0.8);\n  h3 {\n    margin: 0px;\n  }\n`;\n\nconst StyledImg = styled.img`\n  max-height: 600px;\n  object-position: top;\n  width: 100%;\n  aspect-ratio: 16/9;\n  ${mq.range({ until: breakpoints.tablet })} {\n    min-height: 440px;\n    max-height: 440px;\n  }\n  object-fit: cover;\n`;\n\nconst CarouselContainer = styled.div`\n  margin-top: -50px;\n  ${mq.range({ from: breakpoints.tablet })} {\n    margin-top: -70px;\n  }\n  ${mq.range({ from: breakpoints.desktop })} {\n    margin-top: -150px;\n  }\n`;\n\ninterface StyledFilmContentCardProps {\n  current?: boolean;\n}\n\nconst SlideshowButton = styled(IconButtonV2)`\n  margin-top: ${spacing.normal};\n`;\n\nconst shouldForwardProp = (p: string) => p !== 'current';\n\nconst StyledFilmContentCard = styled(FilmContentCard, { shouldForwardProp })<StyledFilmContentCardProps>`\n  margin-bottom: 2%;\n  transform: ${(p) => (p.current ? 'translateY(0%)' : 'translateY(10%)')};\n  transition: all 200ms;\n`;\n\nconst FilmSlideshow = ({ slideshow }: Props) => {\n  const [currentSlide, setCurrentSlide] = useState(slideshow[0]);\n\n  return (\n    <CarouselAutosize breakpoints={slideshowBreakpoints} itemsLength={slideshow.length}>\n      {(autoSizedProps) => (\n        <section>\n          <StyledSafeLink to={currentSlide.path} tabIndex={-1} aria-hidden>\n            <StyledImg src={currentSlide.metaImage?.url ?? ''} alt={currentSlide.metaImage?.alt ?? ''} />\n            <SlideInfoWrapper>\n              <InfoWrapper>\n                <h3>{currentSlide.title}</h3>\n                <span id=\"currentMovieDescription\">{currentSlide.metaDescription}</span>\n              </InfoWrapper>\n            </SlideInfoWrapper>\n          </StyledSafeLink>\n          <CarouselContainer>\n            <Carousel\n              leftButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronLeft />\n                </SlideshowButton>\n              }\n              rightButton={\n                <SlideshowButton aria-label={''}>\n                  <ChevronRight />\n                </SlideshowButton>\n              }\n              items={slideshow.map((movie) => (\n                <FilmCard\n                  key={movie.id}\n                  current={movie.id === currentSlide.id}\n                  movie={movie}\n                  columnWidth={autoSizedProps.columnWidth}\n                  setCurrentSlide={() => setCurrentSlide(movie)}\n                />\n              ))}\n              {...autoSizedProps}\n            />\n          </CarouselContainer>\n        </section>\n      )}\n    </CarouselAutosize>\n  );\n};\n\ninterface FilmCardProps {\n  setCurrentSlide: () => void;\n  movie: MovieType;\n  current: boolean;\n  columnWidth: number;\n}\n\nconst FilmCard = ({ setCurrentSlide, movie, current, columnWidth }: FilmCardProps) => {\n  const [hoverCallback, setHoverCallback] = useState<ReturnType<typeof setTimeout> | undefined>(undefined);\n\n  const onHover = useCallback(() => {\n    const timeout = setTimeout(() => setCurrentSlide(), 500);\n    setHoverCallback(timeout);\n  }, [setCurrentSlide]);\n\n  return (\n    <StyledFilmContentCard\n      onMouseEnter={onHover}\n      onMouseLeave={() => {\n        if (hoverCallback) {\n          clearTimeout(hoverCallback);\n          setHoverCallback(undefined);\n        }\n      }}\n      onFocus={() => setCurrentSlide()}\n      current={current}\n      aria-describedby={'currentMovieDescription'}\n      key={movie.id}\n      movie={movie}\n      columnWidth={columnWidth}\n      resourceTypes={[]}\n    />\n  );\n};\n\nexport default FilmSlideshow;\n"]} */"));
156
127
  var FilmSlideshow = function FilmSlideshow(_ref) {
157
- var _ref$autoSlide = _ref.autoSlide,
158
- autoSlide = _ref$autoSlide === void 0 ? false : _ref$autoSlide,
159
- _ref$slideshow = _ref.slideshow,
160
- slideshow = _ref$slideshow === void 0 ? [] : _ref$slideshow,
161
- _ref$slideInterval = _ref.slideInterval,
162
- slideInterval = _ref$slideInterval === void 0 ? 5000 : _ref$slideInterval;
163
- var _useState = (0, _react.useState)(0),
128
+ var slideshow = _ref.slideshow;
129
+ var _useState = (0, _react.useState)(slideshow[0]),
164
130
  _useState2 = _slicedToArray(_useState, 2),
165
- swipeDistance = _useState2[0],
166
- setSwipeDistance = _useState2[1];
167
- var _useState3 = (0, _react.useState)(0),
168
- _useState4 = _slicedToArray(_useState3, 2),
169
- slideIndex = _useState4[0],
170
- setSlideIndex = _useState4[1];
171
- var _useState5 = (0, _react.useState)(0),
172
- _useState6 = _slicedToArray(_useState5, 2),
173
- slideIndexTarget = _useState6[0],
174
- setSlideIndexTarget = _useState6[1];
175
- var _useState7 = (0, _react.useState)(true),
176
- _useState8 = _slicedToArray(_useState7, 2),
177
- animationComplete = _useState8[0],
178
- setAnimationComplete = _useState8[1];
179
- var _useState9 = (0, _react.useState)(undefined),
180
- _useState10 = _slicedToArray(_useState9, 2),
181
- swipeDirection = _useState10[0],
182
- setSwipeDirection = _useState10[1];
183
- var slideRef = (0, _react.useRef)(null);
184
- var slideText = (0, _react.useRef)(null);
185
- var timer = (0, _react.useRef)(null);
186
- var gotoSlide = (0, _react.useCallback)(function (indexTarget) {
187
- var useAnimation = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false;
188
- setSwipeDistance(0);
189
- if (timer.current) {
190
- clearTimeout(timer.current);
191
- }
192
- setSlideIndexTarget(indexTarget);
193
- setAnimationComplete(!useAnimation);
194
- }, []);
195
- var onChangedSlide = function onChangedSlide() {
196
- if (!animationComplete) {
197
- if (slideRef.current) {
198
- slideRef.current.style.transition = 'none';
199
- slideRef.current.style.transform = "translateX(".concat(slideIndexTarget * 100, "vw))");
200
- }
201
- setAnimationComplete(true);
202
- setSlideIndex(slideIndexTarget);
203
- } else if (slideIndexTarget === -1) {
204
- if (slideRef.current) {
205
- // Go to last slide for continuous loop
206
- slideRef.current.style.transition = 'none';
207
- slideRef.current.style.transform = "translateX(".concat(slideshow.length * 100, "vw))");
208
- }
209
- setSlideIndex(slideshow.length - 1);
210
- setSlideIndexTarget(slideshow.length - 1);
211
- setAnimationComplete(true);
212
- } else if (slideIndexTarget === slideshow.length) {
213
- if (slideRef.current) {
214
- // Go to first slide for continuous loop
215
- slideRef.current.style.transition = 'none';
216
- slideRef.current.style.transform = "translateX(100vw))";
217
- }
218
- setSlideIndex(0);
219
- setSlideIndexTarget(0);
220
- setAnimationComplete(true);
221
- } else {
222
- setAnimationComplete(true);
223
- setSlideIndex(slideIndexTarget);
224
- }
225
- };
226
- var onSwipeEnd = function onSwipeEnd() {
227
- setSwipeDirection(undefined);
228
- var slide;
229
- if (swipeDistance > 40) {
230
- slide = -1;
231
- } else if (swipeDistance < -40) {
232
- slide = 1;
233
- } else {
234
- slide = 0;
235
- }
236
- if (slideRef.current && slideText.current) {
237
- slideRef.current.style.transition = defaultTransitionSwipeEnd;
238
- slideText.current.style.transition = defaultTransitionText;
239
- slideText.current.style.opacity = '1';
240
- }
241
- setSwipeDistance(0);
242
- initTimer();
243
- if (slide !== 0) {
244
- setSlideIndex(slideIndex + slide);
245
- setSlideIndexTarget(slideIndex + slide);
246
- } else {
247
- // Reset transfrom
248
- if (slideRef.current) {
249
- slideRef.current.style.transform = getSlidePosition(slideIndex + slide);
250
- }
251
- }
252
- };
253
- var onSwipe = function onSwipe(eventData) {
254
- if (eventData.initial) {
255
- setSwipeDirection(eventData.dir);
256
- }
257
- var dir = eventData.initial ? eventData.dir : swipeDirection;
258
- if (dir === 'Up' || dir === 'Down') {
259
- return;
260
- }
261
- if (timer.current) {
262
- clearTimeout(timer.current);
263
- }
264
- setSwipeDistance(eventData.deltaX);
265
- if (slideRef && slideRef.current) {
266
- slideRef.current.style.transition = 'none';
267
- slideRef.current.style.transform = getSlidePosition(slideIndexTarget);
268
- }
269
- var opacityText = 1 - Math.min(100, Math.abs(swipeDistance)) / 100;
270
- if (slideText && slideText.current) {
271
- slideText.current.style.transition = 'none';
272
- slideText.current.style.opacity = opacityText.toString();
273
- }
274
- };
275
- var onTransitionEnd = function onTransitionEnd() {
276
- var slideshowLength = slideshow.length;
277
- if (slideIndex === -1) {
278
- if (slideRef.current) {
279
- slideRef.current.style.transition = 'none';
280
- slideRef.current.style.transform = getSlidePosition(slideshowLength - 1);
281
- }
282
- setSlideIndex(slideshowLength - 1);
283
- setSlideIndexTarget(slideshowLength - 1);
284
- } else if (slideIndex >= slideshowLength) {
285
- if (slideRef.current) {
286
- slideRef.current.style.transition = 'none';
287
- slideRef.current.style.transform = getSlidePosition(0);
288
- }
289
- setSlideIndex(0);
290
- setSlideIndexTarget(0);
291
- }
292
- };
293
- var getSlidePosition = function getSlidePosition(target) {
294
- if (swipeDistance !== 0) {
295
- return "translateX(calc(".concat(swipeDistance, "px -\n ").concat((target + 1) * 100, "vw))");
296
- }
297
- return "translateX(-".concat((target + 1) * 100, "vw)");
298
- };
299
- var initTimer = (0, _react.useCallback)(function () {
300
- if (autoSlide) {
301
- timer.current = setTimeout(function () {
302
- gotoSlide(slideIndex + 1);
303
- }, slideInterval);
304
- }
305
- }, [autoSlide, gotoSlide, slideInterval, slideIndex]);
306
- (0, _react.useEffect)(function () {
307
- initTimer();
308
- }, [initTimer]);
309
- var handlers = (0, _reactSwipeable.useSwipeable)({
310
- onSwiped: onSwipeEnd,
311
- onSwiping: onSwipe,
312
- preventScrollOnSwipe: swipeDirection === 'Left' || swipeDirection === 'Right'
313
- });
314
- if (slideshow.length === 0) {
315
- return (0, _jsxRuntime.jsx)("div", {
316
- children: (0, _jsxRuntime.jsx)(EmptySlideshow, {
317
- children: (0, _jsxRuntime.jsx)(_icons.Spinner, {
318
- inverted: true
319
- })
320
- })
321
- });
322
- }
323
- var slideshowWidth = "".concat((slideshow.length + 2) * 100, "vw");
324
- var activeSlide = slideIndex;
325
- if (activeSlide < 0) {
326
- activeSlide = slideshow.length - 1;
327
- } else if (activeSlide >= slideshow.length) {
328
- activeSlide = 0;
329
- }
330
- var backgroundImage = slideshow[activeSlide].metaImage;
331
- return (0, _jsxRuntime.jsx)(SlideshowWrapper, _objectSpread(_objectSpread({}, handlers), {}, {
332
- children: (0, _jsxRuntime.jsxs)(_jsxRuntime.Fragment, {
333
- children: [(0, _jsxRuntime.jsx)(SlideLinkWrapper, {
334
- children: (0, _jsxRuntime.jsx)(_Layout.OneColumn, {
335
- children: (0, _jsxRuntime.jsx)(SlideshowLink, {
336
- to: slideshow[activeSlide].path,
337
- out: !animationComplete,
338
- children: (0, _jsxRuntime.jsxs)(SlideshowInfo, {
339
- ref: slideText,
340
- children: [(0, _jsxRuntime.jsx)(SlideshowName, {
341
- children: slideshow[activeSlide].title
342
- }), (0, _jsxRuntime.jsx)(SlideshowDescription, {
343
- children: slideshow[activeSlide].metaDescription
131
+ currentSlide = _useState2[0],
132
+ _setCurrentSlide = _useState2[1];
133
+ return (0, _jsxRuntime.jsx)(_carousel.CarouselAutosize, {
134
+ breakpoints: slideshowBreakpoints,
135
+ itemsLength: slideshow.length,
136
+ children: function children(autoSizedProps) {
137
+ var _currentSlide$metaIma, _currentSlide$metaIma2, _currentSlide$metaIma3, _currentSlide$metaIma4;
138
+ return (0, _jsxRuntime.jsxs)("section", {
139
+ children: [(0, _jsxRuntime.jsxs)(StyledSafeLink, {
140
+ to: currentSlide.path,
141
+ tabIndex: -1,
142
+ "aria-hidden": true,
143
+ children: [(0, _jsxRuntime.jsx)(StyledImg, {
144
+ src: (_currentSlide$metaIma = (_currentSlide$metaIma2 = currentSlide.metaImage) === null || _currentSlide$metaIma2 === void 0 ? void 0 : _currentSlide$metaIma2.url) !== null && _currentSlide$metaIma !== void 0 ? _currentSlide$metaIma : '',
145
+ alt: (_currentSlide$metaIma3 = (_currentSlide$metaIma4 = currentSlide.metaImage) === null || _currentSlide$metaIma4 === void 0 ? void 0 : _currentSlide$metaIma4.alt) !== null && _currentSlide$metaIma3 !== void 0 ? _currentSlide$metaIma3 : ''
146
+ }), (0, _jsxRuntime.jsx)(SlideInfoWrapper, {
147
+ children: (0, _jsxRuntime.jsxs)(InfoWrapper, {
148
+ children: [(0, _jsxRuntime.jsx)("h3", {
149
+ children: currentSlide.title
150
+ }), (0, _jsxRuntime.jsx)("span", {
151
+ id: "currentMovieDescription",
152
+ children: currentSlide.metaDescription
344
153
  })]
345
154
  })
346
- })
347
- })
348
- }), (0, _jsxRuntime.jsx)(_NavigationArrow["default"], {
349
- slideIndexTarget: slideIndexTarget > 0 ? slideIndexTarget - 1 : slideshow.length - 1,
350
- gotoSlide: gotoSlide
351
- }), (0, _jsxRuntime.jsx)(_NavigationArrow["default"], {
352
- slideIndexTarget: slideIndexTarget < slideshow.length - 1 ? slideIndexTarget + 1 : 0,
353
- gotoSlide: gotoSlide,
354
- rightArrow: true
355
- }), !animationComplete && (0, _jsxRuntime.jsx)(SlideshowItem, {
356
- fadeOver: true,
357
- role: "img",
358
- onAnimationEnd: onChangedSlide,
359
- style: {
360
- backgroundImage: "url(".concat(backgroundImage && backgroundImage.url || '', ")")
361
- }
362
- }), (0, _jsxRuntime.jsxs)("div", {
363
- ref: slideRef,
364
- css: itemWrapperCSS,
365
- onTransitionEnd: onTransitionEnd,
366
- style: {
367
- width: slideshowWidth,
368
- transform: getSlidePosition(slideIndex)
369
- },
370
- children: [renderSlideItem(slideshow[slideshow.length - 1]), slideshow.map(renderSlideItem), renderSlideItem(slideshow[0])]
371
- }), (0, _jsxRuntime.jsx)(_SlideshowIndicator["default"], {
372
- slideshow: slideshow,
373
- activeSlide: activeSlide,
374
- gotoSlide: gotoSlide
375
- })]
376
- })
377
- }));
155
+ })]
156
+ }), (0, _jsxRuntime.jsx)(CarouselContainer, {
157
+ children: (0, _jsxRuntime.jsx)(_carousel.Carousel, _objectSpread({
158
+ leftButton: (0, _jsxRuntime.jsx)(SlideshowButton, {
159
+ "aria-label": '',
160
+ children: (0, _jsxRuntime.jsx)(_common.ChevronLeft, {})
161
+ }),
162
+ rightButton: (0, _jsxRuntime.jsx)(SlideshowButton, {
163
+ "aria-label": '',
164
+ children: (0, _jsxRuntime.jsx)(_common.ChevronRight, {})
165
+ }),
166
+ items: slideshow.map(function (movie) {
167
+ return (0, _jsxRuntime.jsx)(FilmCard, {
168
+ current: movie.id === currentSlide.id,
169
+ movie: movie,
170
+ columnWidth: autoSizedProps.columnWidth,
171
+ setCurrentSlide: function setCurrentSlide() {
172
+ return _setCurrentSlide(movie);
173
+ }
174
+ }, movie.id);
175
+ })
176
+ }, autoSizedProps))
177
+ })]
178
+ });
179
+ }
180
+ });
181
+ };
182
+ var FilmCard = function FilmCard(_ref2) {
183
+ var setCurrentSlide = _ref2.setCurrentSlide,
184
+ movie = _ref2.movie,
185
+ current = _ref2.current,
186
+ columnWidth = _ref2.columnWidth;
187
+ var _useState3 = (0, _react.useState)(undefined),
188
+ _useState4 = _slicedToArray(_useState3, 2),
189
+ hoverCallback = _useState4[0],
190
+ setHoverCallback = _useState4[1];
191
+ var onHover = (0, _react.useCallback)(function () {
192
+ var timeout = setTimeout(function () {
193
+ return setCurrentSlide();
194
+ }, 500);
195
+ setHoverCallback(timeout);
196
+ }, [setCurrentSlide]);
197
+ return (0, _jsxRuntime.jsx)(StyledFilmContentCard, {
198
+ onMouseEnter: onHover,
199
+ onMouseLeave: function onMouseLeave() {
200
+ if (hoverCallback) {
201
+ clearTimeout(hoverCallback);
202
+ setHoverCallback(undefined);
203
+ }
204
+ },
205
+ onFocus: function onFocus() {
206
+ return setCurrentSlide();
207
+ },
208
+ current: current,
209
+ "aria-describedby": 'currentMovieDescription',
210
+ movie: movie,
211
+ columnWidth: columnWidth,
212
+ resourceTypes: []
213
+ }, movie.id);
378
214
  };
379
215
  var _default = FilmSlideshow;
380
216
  exports["default"] = _default;