@ndla/ui 35.1.1 → 36.0.0

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