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