@ndla/ui 44.0.22 → 44.0.24

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.
@@ -34,39 +34,39 @@ var ControlsWrapper = /*#__PURE__*/(0, _base.default)("div", {
34
34
  label: "ControlsWrapper"
35
35
  })("border:1px solid ", _core.colors.brand.lighter, ";display:flex;align-items:center;justify-content:center;background:", _core.colors.white, ";font-family:", _core.fonts.sans, ";gap:", _core.spacing.xsmall, ";padding:", _core.spacing.small, " ", _core.spacing.normal, ";", _core.mq.range({
36
36
  until: _core.breakpoints.tabletWide
37
- }), "{display:grid;padding:", _core.spacing.small, ";grid-template-columns:1fr repeat(5, auto) 1fr;grid-template-areas:'track track track track track track track' '. speed backwards play forwards volume .';}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmBkC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
37
+ }), "{display:grid;padding:", _core.spacing.small, ";grid-template-columns:1fr repeat(5, auto) 1fr;grid-template-areas:'track track track track track track track' '. speed backwards play forwards volume .';}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmBkC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
38
38
  var PlayButton = /*#__PURE__*/(0, _base.default)(_button.IconButtonV2, {
39
39
  target: "e1gaeajv19",
40
40
  label: "PlayButton"
41
41
  })(_core.mq.range({
42
42
  until: _core.breakpoints.tabletWide
43
- }), "{grid-area:play;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsCuC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
43
+ }), "{grid-area:play;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsCuC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
44
44
  var Forward15SecButton = /*#__PURE__*/(0, _base.default)(_button.IconButtonV2, {
45
45
  target: "e1gaeajv18",
46
46
  label: "Forward15SecButton"
47
47
  })(_core.mq.range({
48
48
  until: _core.breakpoints.tabletWide
49
- }), "{grid-area:forwards;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4C+C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
49
+ }), "{grid-area:forwards;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4C+C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
50
50
  var Back15SecButton = /*#__PURE__*/(0, _base.default)(_button.IconButtonV2, {
51
51
  target: "e1gaeajv17",
52
52
  label: "Back15SecButton"
53
53
  })(_core.mq.range({
54
54
  until: _core.breakpoints.tabletWide
55
- }), "{grid-area:backwards;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiD4C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
55
+ }), "{grid-area:backwards;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiD4C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
56
56
  var SpeedButton = /*#__PURE__*/(0, _base.default)(_button.ButtonV2, {
57
57
  target: "e1gaeajv16",
58
58
  label: "SpeedButton"
59
59
  })(_core.mq.range({
60
60
  until: _core.breakpoints.tabletWide
61
- }), "{grid-area:speed;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAuDoC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
61
+ }), "{grid-area:speed;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAuDoC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
62
62
  var SpeedList = /*#__PURE__*/(0, _base.default)(_dropdownMenu.DropdownContent, {
63
63
  target: "e1gaeajv15",
64
64
  label: "SpeedList"
65
- })("border:1px solid ", _core.colors.brand.lighter, ";padding:5px 10px;justify-content:center;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA6DyC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
65
+ })("border:1px solid ", _core.colors.brand.lighter, ";padding:5px 10px;justify-content:center;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA6DyC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
66
66
  var SpeedValueButton = /*#__PURE__*/(0, _base.default)(_button.ButtonV2, {
67
67
  target: "e1gaeajv14",
68
68
  label: "SpeedValueButton"
69
- })("padding:0 14px;gap:0px;color:", _core.colors.text.light, ";display:flex;justify-content:center;&:hover,&:active,&:focus,&[data-highlighted]{outline:none;color:", _core.colors.text.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmEyC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
69
+ })("padding:0 14px;gap:0px;color:", _core.colors.text.light, ";display:flex;justify-content:center;&:hover,&:active,&:focus,&[data-highlighted]{outline:none;color:", _core.colors.text.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmEyC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
70
70
  var SpeedSelectedMark = /*#__PURE__*/(0, _base.default)("span", {
71
71
  target: "e1gaeajv13",
72
72
  label: "SpeedSelectedMark"
@@ -76,19 +76,19 @@ var SpeedSelectedMark = /*#__PURE__*/(0, _base.default)("span", {
76
76
  } : {
77
77
  name: "1fc4dbs",
78
78
  styles: "align-self:flex-start;border-radius:50%;background:#d1372e;width:6px;height:6px;margin:6px 0 0 2px",
79
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkFqC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
79
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkFqC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
80
80
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
81
81
  });
82
82
  var Time = /*#__PURE__*/(0, _base.default)("div", {
83
83
  target: "e1gaeajv12",
84
84
  label: "Time"
85
- })(_core.fonts.sizes('16px'), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2FuB","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
85
+ })(_core.fonts.sizes('16px'), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2FuB","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
86
86
  var ProgressWrapper = /*#__PURE__*/(0, _base.default)("div", {
87
87
  target: "e1gaeajv11",
88
88
  label: "ProgressWrapper"
89
89
  })("flex:1;display:flex;align-items:center;gap:", _core.spacing.small, ";", _core.mq.range({
90
90
  until: _core.breakpoints.tabletWide
91
- }), "{grid-area:track;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA+FkC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
91
+ }), "{grid-area:track;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA+FkC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
92
92
  var SliderWrapper = /*#__PURE__*/(0, _base.default)(_reactSlider.Root, {
93
93
  target: "e1gaeajv10",
94
94
  label: "SliderWrapper"
@@ -98,13 +98,13 @@ var SliderWrapper = /*#__PURE__*/(0, _base.default)(_reactSlider.Root, {
98
98
  } : {
99
99
  name: "5l00y9",
100
100
  styles: "cursor:pointer;flex:1;position:relative;display:flex;align-items:center;user-select:none;touch-action:none",
101
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAyGwC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
101
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAyGwC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
102
102
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
103
103
  });
104
104
  var StyledTrack = /*#__PURE__*/(0, _base.default)(_reactSlider.Track, {
105
105
  target: "e1gaeajv9",
106
106
  label: "StyledTrack"
107
- })("height:4px;width:100%;background:", _core.colors.brand.lighter, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmHiC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
107
+ })("height:4px;width:100%;background:", _core.colors.brand.lighter, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmHiC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
108
108
  var StyledRange = /*#__PURE__*/(0, _base.default)(_reactSlider.Range, {
109
109
  target: "e1gaeajv8",
110
110
  label: "StyledRange"
@@ -114,7 +114,7 @@ var StyledRange = /*#__PURE__*/(0, _base.default)(_reactSlider.Range, {
114
114
  } : {
115
115
  name: "ufqu4g",
116
116
  styles: "position:absolute;height:4px;background:#5cbc80;border-radius:7px",
117
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA0HiC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
117
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA0HiC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
118
118
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
119
119
  });
120
120
  var StyledThumb = /*#__PURE__*/(0, _base.default)(_reactSlider.SliderThumb, {
@@ -126,7 +126,7 @@ var StyledThumb = /*#__PURE__*/(0, _base.default)(_reactSlider.SliderThumb, {
126
126
  } : {
127
127
  name: "183frwh",
128
128
  styles: "display:block;width:20px;height:20px;background:#5cbc80;border-radius:50%;outline:none",
129
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiIuC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
129
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiIuC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
130
130
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
131
131
  });
132
132
  var VolumeWrapper = /*#__PURE__*/(0, _base.default)(_reactPopover.Root, {
@@ -138,13 +138,13 @@ var VolumeWrapper = /*#__PURE__*/(0, _base.default)(_reactPopover.Root, {
138
138
  } : {
139
139
  name: "16evi40",
140
140
  styles: "position:relative;display:flex;justify-content:center",
141
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA0IyC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
141
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA0IyC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
142
142
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
143
143
  });
144
144
  var VolumeList = /*#__PURE__*/(0, _base.default)(_reactPopover.PopoverContent, {
145
145
  target: "e1gaeajv5",
146
146
  label: "VolumeList"
147
- })("box-shadow:0 14px 20px -5px rgba(32, 88, 143, 0.17);border-radius:60px;background:", _core.colors.white, ";padding:", _core.spacing.small, ";border:1px solid ", _core.colors.brand.lighter, ";height:128px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAgJyC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
147
+ })("box-shadow:0 14px 20px -5px rgba(32, 88, 143, 0.17);border-radius:60px;background:", _core.colors.white, ";padding:", _core.spacing.small, ";border:1px solid ", _core.colors.brand.lighter, ";height:128px;z-index:2;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAgJyC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
148
148
  var VolumeSliderWrapper = /*#__PURE__*/(0, _base.default)(_reactSlider.Root, {
149
149
  target: "e1gaeajv4",
150
150
  label: "VolumeSliderWrapper"
@@ -154,7 +154,7 @@ var VolumeSliderWrapper = /*#__PURE__*/(0, _base.default)(_reactSlider.Root, {
154
154
  } : {
155
155
  name: "1naj2yc",
156
156
  styles: "cursor:pointer;height:100%;position:relative;display:flex;flex-direction:column;align-items:center;user-select:none;touch-action:none",
157
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAyJ8C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
157
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA0J8C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
158
158
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
159
159
  });
160
160
  var VolumeButton = /*#__PURE__*/(0, _base.default)(_button.IconButtonV2, {
@@ -162,19 +162,19 @@ var VolumeButton = /*#__PURE__*/(0, _base.default)(_button.IconButtonV2, {
162
162
  label: "VolumeButton"
163
163
  })(_core.mq.range({
164
164
  until: _core.breakpoints.tabletWide
165
- }), "{grid-area:volume;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAoKyC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
165
+ }), "{grid-area:volume;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAqKyC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
166
166
  var VolumeSliderBackground = /*#__PURE__*/(0, _base.default)(_reactSlider.Track, {
167
167
  target: "e1gaeajv2",
168
168
  label: "VolumeSliderBackground"
169
- })("height:100%;width:5px;background:", _core.colors.brand.lighter, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA0K4C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
169
+ })("height:100%;width:5px;background:", _core.colors.brand.lighter, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2K4C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
170
170
  var VolumeSliderSelected = /*#__PURE__*/(0, _base.default)(_reactSlider.Range, {
171
171
  target: "e1gaeajv1",
172
172
  label: "VolumeSliderSelected"
173
- })("position:absolute;width:5px;background:", _core.colors.brand.secondary, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiL0C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
173
+ })("position:absolute;width:5px;background:", _core.colors.brand.secondary, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkL0C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
174
174
  var VolumeSliderHandle = /*#__PURE__*/(0, _base.default)(_reactSlider.SliderThumb, {
175
175
  target: "e1gaeajv0",
176
176
  label: "VolumeSliderHandle"
177
- })("display:block;width:20px;height:20px;background:", _core.colors.brand.primary, ";border-radius:50%;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAwL8C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
177
+ })("display:block;width:20px;height:20px;background:", _core.colors.brand.primary, ";border-radius:50%;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAyL8C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Root as SliderRoot, Track, Range, SliderThumb } from '@radix-ui/react-slider';\nimport { Root as PopoverRoot, PopoverContent, PopoverTrigger, PopoverPortal } from '@radix-ui/react-popover';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\nimport { ButtonV2, IconButtonV2 } from '@ndla/button';\nimport { DropdownMenu, DropdownContent, DropdownItem, DropdownTrigger } from '@ndla/dropdown-menu';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: ${colors.white};\n  font-family: ${fonts.sans};\n  gap: ${spacing.xsmall};\n  padding: ${spacing.small} ${spacing.normal};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    display: grid;\n    padding: ${spacing.small};\n    grid-template-columns: 1fr repeat(5, auto) 1fr;\n    grid-template-areas:\n      'track  track track     track track     track   track'\n      '.      speed backwards play  forwards  volume  .';\n  }\n`;\n\nconst PlayButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: play;\n  }\n`;\n\nconst Forward15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: forwards;\n  }\n`;\nconst Back15SecButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: backwards;\n  }\n`;\n\nconst SpeedButton = styled(ButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: speed;\n  }\n`;\n\nconst SpeedList = styled(DropdownContent)`\n  border: 1px solid ${colors.brand.lighter};\n  padding: 5px 10px;\n  justify-content: center;\n`;\n\nconst SpeedValueButton = styled(ButtonV2)`\n  padding: 0 14px;\n  gap: 0px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-highlighted] {\n    outline: none;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedSelectedMark = styled.span`\n  align-self: flex-start;\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  ${fonts.sizes('16px')};\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1;\n  display: flex;\n  align-items: center;\n  gap: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: track;\n  }\n`;\n\nconst SliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  flex: 1;\n  position: relative;\n  display: flex;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst StyledTrack = styled(Track)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst StyledRange = styled(Range)`\n  position: absolute;\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst StyledThumb = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  outline: none;\n`;\n\nconst VolumeWrapper = styled(PopoverRoot)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n`;\n\nconst VolumeList = styled(PopoverContent)`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: ${colors.white};\n  padding: ${spacing.small};\n  border: 1px solid ${colors.brand.lighter};\n  height: 128px;\n  z-index: 2;\n`;\n\nconst VolumeSliderWrapper = styled(SliderRoot)`\n  cursor: pointer;\n  height: 100%;\n  position: relative;\n  display: flex;\n  flex-direction: column;\n  align-items: center;\n  user-select: none;\n  touch-action: none;\n`;\n\nconst VolumeButton = styled(IconButtonV2)`\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    grid-area: volume;\n  }\n`;\n\nconst VolumeSliderBackground = styled(Track)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(Range)`\n  position: absolute;\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderHandle = styled(SliderThumb)`\n  display: block;\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ninterface Props {\n  src: string;\n  title: string;\n}\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = value[0];\n    }\n  };\n\n  const handleVolumeSliderChange = (values: number[]) => {\n    if (audioRef.current) {\n      audioRef.current.volume = values[0] / 100;\n      setVolumeValue(values[0]);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton\n          aria-label={t(playing ? t('audio.pause') : t('audio.play'))}\n          colorTheme=\"lighter\"\n          size=\"normal\"\n          onClick={togglePlay}\n        >\n          {playing ? <Pause /> : <Play />}\n        </PlayButton>\n        <Back15SecButton\n          variant=\"ghost\"\n          colorTheme=\"greyLighter\"\n          title={t('audio.controls.rewind15sec')}\n          aria-label={t('audio.controls.rewind15sec')}\n          onClick={() => onSeekSeconds(-15)}\n        >\n          <Back15 />\n        </Back15SecButton>\n        <DropdownMenu>\n          <DropdownTrigger>\n            <SpeedButton\n              shape=\"pill\"\n              variant=\"ghost\"\n              size=\"normal\"\n              colorTheme=\"greyLighter\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n          </DropdownTrigger>\n          <SpeedList side=\"top\">\n            {speedValues.map((speed) => (\n              <DropdownItem key={speed}>\n                <SpeedValueButton\n                  variant=\"ghost\"\n                  colorTheme=\"greyLighter\"\n                  size=\"small\"\n                  onSelect={() => setSpeedValue(speed)}\n                >\n                  {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                </SpeedValueButton>\n              </DropdownItem>\n            ))}\n          </SpeedList>\n        </DropdownMenu>\n        <Forward15SecButton\n          colorTheme=\"greyLighter\"\n          variant=\"ghost\"\n          title={t('audio.controls.forward15sec')}\n          aria-label={t('audio.controls.forward15sec')}\n          onClick={() => onSeekSeconds(15)}\n        >\n          <Forward15 />\n        </Forward15SecButton>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper\n            value={[audioRef.current?.currentTime ?? 0]}\n            defaultValue={[0]}\n            step={1}\n            max={audioRef.current?.duration ?? 0}\n            onValueChange={handleSliderChange}\n          >\n            <StyledTrack>\n              <StyledRange />\n            </StyledTrack>\n            <StyledThumb />\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <PopoverTrigger asChild>\n            <VolumeButton variant=\"ghost\" colorTheme=\"greyLighter\" aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n          </PopoverTrigger>\n          <PopoverPortal>\n            <VolumeList side=\"top\">\n              <VolumeSliderWrapper\n                orientation=\"vertical\"\n                value={[volumeValue]}\n                min={0}\n                defaultValue={[100]}\n                step={1}\n                onValueChange={handleVolumeSliderChange}\n              >\n                <VolumeSliderBackground>\n                  <VolumeSliderSelected />\n                </VolumeSliderBackground>\n                <VolumeSliderHandle />\n              </VolumeSliderWrapper>\n            </VolumeList>\n          </PopoverPortal>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
178
178
  var formatTime = function formatTime(seconds) {
179
179
  var minutes = Math.floor(seconds / 60);
180
180
  var currentSeconds = seconds % 60;