@ndla/ui 37.1.3 → 37.1.4

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.
@@ -9,6 +9,7 @@ var _base = _interopRequireDefault(require("@emotion/styled/base"));
9
9
  var _react = _interopRequireWildcard(require("react"));
10
10
  var _menuButton = require("@reach/menu-button");
11
11
  var _slider = require("@reach/slider");
12
+ var _react2 = require("@headlessui/react");
12
13
  var _common = require("@ndla/icons/common");
13
14
  var _core = require("@ndla/core");
14
15
  var _reactI18next = require("react-i18next");
@@ -31,47 +32,47 @@ function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringif
31
32
  *
32
33
  */
33
34
  var ControlsWrapper = /*#__PURE__*/(0, _base["default"])("div", {
34
- target: "e1gaeajv25",
35
+ target: "e1gaeajv24",
35
36
  label: "ControlsWrapper"
36
37
  })("border:1px solid ", _core.colors.brand.lighter, ";display:flex;align-items:center;justify-content:center;background:#ffffff;font-family:", _core.fonts.sans, ";", _core.mq.range({
37
38
  until: _core.breakpoints.tabletWide
38
39
  }), "{flex-wrap:wrap;}padding:", _core.spacing.small, ";", _core.mq.range({
39
40
  from: _core.breakpoints.tabletWide
40
- }), "{padding:", _core.spacing.small, " ", _core.spacing.normal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiBkC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
41
+ }), "{padding:", _core.spacing.small, " ", _core.spacing.normal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkBkC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
41
42
  var PlayButton = /*#__PURE__*/(0, _base["default"])("button", {
42
- target: "e1gaeajv24",
43
+ target: "e1gaeajv23",
43
44
  label: "PlayButton"
44
45
  })("background:", _core.colors.brand.lighter, ";border:none;display:flex;align-items:center;justify-content:center;padding:0;cursor:pointer;color:", _core.colors.brand.primary, ";width:55px;height:55px;border-radius:50%;transition:", _core.misc.transition["default"], ";margin-right:", _core.spacing.small, ";", _core.mq.range({
45
46
  until: _core.breakpoints.tabletWide
46
- }), "{order:4;margin-left:", _core.spacing.small, ";}&:hover,&:active,&:focus{background:", _core.colors.brand.primary, ";color:#ffffff;}.c-icon{width:24px;height:24px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiCgC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
47
+ }), "{order:4;margin-left:", _core.spacing.small, ";}&:hover,&:active,&:focus{background:", _core.colors.brand.primary, ";color:#ffffff;}.c-icon{width:24px;height:24px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkCgC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
47
48
  var ForwardRewindButton = /*#__PURE__*/(0, _base["default"])("button", {
48
- target: "e1gaeajv23",
49
+ target: "e1gaeajv22",
49
50
  label: "ForwardRewindButton"
50
- })("background-color:inherit;background-position:center;background-repeat:no-repeat;width:42px;height:42px;border:0;border-radius:50%;cursor:pointer;font-weight:bold;font-size:9px;line-height:23px;color:", _core.colors.brand.dark, ";font-family:", _core.fonts.sans, ";transition:", _core.misc.transition["default"], ";&:hover{background-color:", _core.colors.brand.greyLighter, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiEyC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
51
+ })("background-color:inherit;background-position:center;background-repeat:no-repeat;width:42px;height:42px;border:0;border-radius:50%;cursor:pointer;font-weight:bold;font-size:9px;line-height:23px;color:", _core.colors.brand.dark, ";font-family:", _core.fonts.sans, ";transition:", _core.misc.transition["default"], ";&:hover{background-color:", _core.colors.brand.greyLighter, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkEyC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
51
52
  var Forward15SecButton = /*#__PURE__*/(0, _base["default"])(ForwardRewindButton, {
52
- target: "e1gaeajv22",
53
+ target: "e1gaeajv21",
53
54
  label: "Forward15SecButton"
54
55
  })("svg{fill:", _core.colors.brand.primary, ";width:24px;height:24px;}", _core.mq.range({
55
56
  until: _core.breakpoints.tabletWide
56
- }), "{order:3;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsFsD","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
57
+ }), "{order:3;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAuFsD","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
57
58
  var Back15SecButton = /*#__PURE__*/(0, _base["default"])(ForwardRewindButton, {
58
- target: "e1gaeajv21",
59
+ target: "e1gaeajv20",
59
60
  label: "Back15SecButton"
60
61
  })("svg{fill:", _core.colors.brand.primary, ";width:24px;height:24px;}", _core.mq.range({
61
62
  until: _core.breakpoints.tabletWide
62
- }), "{order:5;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAgGmD","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
63
+ }), "{order:5;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiGmD","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
63
64
  var SpeedWrapper = /*#__PURE__*/(0, _base["default"])("div", {
64
- target: "e1gaeajv20",
65
+ target: "e1gaeajv19",
65
66
  label: "SpeedWrapper"
66
67
  })("position:relative;display:flex;justify-content:center;", _core.mq.range({
67
68
  until: _core.breakpoints.tabletWide
68
- }), "{order:2;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2G+B","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
69
+ }), "{order:2;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4G+B","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
69
70
  var SpeedButton = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuButton, {
70
- target: "e1gaeajv19",
71
+ target: "e1gaeajv18",
71
72
  label: "SpeedButton"
72
- })("height:32px;border:0;background:none;cursor:pointer;font-weight:600;font-size:12px;text-align:center;width:52px;&:hover,&:active,&:focus,&[aria-expanded='true']{background:", _core.colors.brand.greyLighter, ";border-radius:27px;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":"AAmHsC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
73
+ })("height:32px;border:0;background:none;cursor:pointer;font-weight:600;font-size:12px;text-align:center;width:52px;&:hover,&:active,&:focus,&[aria-expanded='true']{background:", _core.colors.brand.greyLighter, ";border-radius:27px;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":"AAoHsC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
73
74
  var SpeedMenu = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuPopover, {
74
- target: "e1gaeajv18",
75
+ target: "e1gaeajv17",
75
76
  label: "SpeedMenu"
76
77
  })(process.env.NODE_ENV === "production" ? {
77
78
  name: "exmt8v",
@@ -79,21 +80,21 @@ var SpeedMenu = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuPopover, {
79
80
  } : {
80
81
  name: "exmt8v",
81
82
  styles: "position:absolute;bottom:36px;z-index:99",
82
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsIqC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
83
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAuIqC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
83
84
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
84
85
  });
85
86
  var SpeedList = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuItems, {
86
- target: "e1gaeajv17",
87
+ target: "e1gaeajv16",
87
88
  label: "SpeedList"
88
- })("background:#ffffff;border:1px solid ", _core.colors.brand.lighter, ";border-radius:5px;padding:5px 10px;display:flex;flex-direction:column;justify-content:center;align-items:stretch;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4ImC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
89
+ })("background:#ffffff;border:1px solid ", _core.colors.brand.lighter, ";border-radius:5px;padding:5px 10px;display:flex;flex-direction:column;justify-content:center;align-items:stretch;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA6ImC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
89
90
  var SpeedValueButton = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuItem, {
90
- target: "e1gaeajv16",
91
+ target: "e1gaeajv15",
91
92
  label: "SpeedValueButton"
92
93
  })("height:28px;position:relative;background:none;border:0;padding:0 14px;cursor:pointer;font-weight:600;font-size:14px;color:", _core.colors.text.light, ";display:flex;justify-content:center;align-items:center;&:hover,&:active,&:focus,&[data-selected]{background:", _core.colors.brand.greyLighter, ";border-radius:5px;color:", _core.colors.text.primary, ";}", function (props) {
93
94
  return props.selected && "\n color: ".concat(_core.colors.text.primary, ";\n \n ");
94
- }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2JgE","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
95
+ }, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4JgE","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
95
96
  var SpeedSelectedMark = /*#__PURE__*/(0, _base["default"])("span", {
96
- target: "e1gaeajv15",
97
+ target: "e1gaeajv14",
97
98
  label: "SpeedSelectedMark"
98
99
  })(process.env.NODE_ENV === "production" ? {
99
100
  name: "1ymok7g",
@@ -101,11 +102,11 @@ var SpeedSelectedMark = /*#__PURE__*/(0, _base["default"])("span", {
101
102
  } : {
102
103
  name: "1ymok7g",
103
104
  styles: "border-radius:50%;background:#d1372e;width:6px;height:6px;display:inline-block;align-self:flex-start;margin:6px 0 0 2px",
104
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAwLqC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
105
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAyLqC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
105
106
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
106
107
  });
107
108
  var Time = /*#__PURE__*/(0, _base["default"])("div", {
108
- target: "e1gaeajv14",
109
+ target: "e1gaeajv13",
109
110
  label: "Time"
110
111
  })(process.env.NODE_ENV === "production" ? {
111
112
  name: "11g4mt0",
@@ -113,25 +114,25 @@ var Time = /*#__PURE__*/(0, _base["default"])("div", {
113
114
  } : {
114
115
  name: "11g4mt0",
115
116
  styles: "font-size:16px",
116
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkMuB","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\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":"AAmMuB","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
117
118
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
118
119
  });
119
120
  var ProgressWrapper = /*#__PURE__*/(0, _base["default"])("div", {
120
- target: "e1gaeajv13",
121
+ target: "e1gaeajv12",
121
122
  label: "ProgressWrapper"
122
123
  })("flex:1 1 auto;display:flex;align-items:center;margin:0 ", _core.spacing.small, ";", _core.mq.range({
123
124
  until: _core.breakpoints.tabletWide
124
- }), "{order:1;width:100%;margin:0;margin-bottom:", _core.spacing.normal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsMkC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
125
+ }), "{order:1;width:100%;margin:0;margin-bottom:", _core.spacing.normal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAuMkC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
125
126
  var SliderWrapper = /*#__PURE__*/(0, _base["default"])("div", {
126
- target: "e1gaeajv12",
127
+ target: "e1gaeajv11",
127
128
  label: "SliderWrapper"
128
- })("cursor:pointer;flex:1 1 auto;margin:0 ", _core.spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkNgC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
129
+ })("cursor:pointer;flex:1 1 auto;margin:0 ", _core.spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmNgC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
129
130
  var ProgressBackground = /*#__PURE__*/(0, _base["default"])(_slider.SliderTrack, {
130
- target: "e1gaeajv11",
131
+ target: "e1gaeajv10",
131
132
  label: "ProgressBackground"
132
- })("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":"AAwN8C","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
133
+ })("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":"AAyN8C","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
133
134
  var ProgressPlayed = /*#__PURE__*/(0, _base["default"])(_slider.SliderRange, {
134
- target: "e1gaeajv10",
135
+ target: "e1gaeajv9",
135
136
  label: "ProgressPlayed"
136
137
  })(process.env.NODE_ENV === "production" ? {
137
138
  name: "1pw5ehb",
@@ -139,11 +140,11 @@ var ProgressPlayed = /*#__PURE__*/(0, _base["default"])(_slider.SliderRange, {
139
140
  } : {
140
141
  name: "1pw5ehb",
141
142
  styles: "height:4px;background:#5cbc80;border-radius:7px",
142
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA+N0C","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
143
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAgO0C","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
143
144
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
144
145
  });
145
146
  var ProgressHandle = /*#__PURE__*/(0, _base["default"])(_slider.SliderHandle, {
146
- target: "e1gaeajv9",
147
+ target: "e1gaeajv8",
147
148
  label: "ProgressHandle"
148
149
  })(process.env.NODE_ENV === "production" ? {
149
150
  name: "11d5yus",
@@ -151,43 +152,31 @@ var ProgressHandle = /*#__PURE__*/(0, _base["default"])(_slider.SliderHandle, {
151
152
  } : {
152
153
  name: "11d5yus",
153
154
  styles: "width:20px;height:20px;background:#5cbc80;border-radius:50%;top:-8px",
154
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAqO2C","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
155
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsO2C","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
155
156
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
156
157
  });
157
- var VolumeWrapper = /*#__PURE__*/(0, _base["default"])("div", {
158
- target: "e1gaeajv8",
158
+ var VolumeWrapper = /*#__PURE__*/(0, _base["default"])(_react2.Popover, {
159
+ target: "e1gaeajv7",
159
160
  label: "VolumeWrapper"
160
161
  })("position:relative;display:flex;justify-content:center;", _core.mq.range({
161
162
  until: _core.breakpoints.tabletWide
162
- }), "{order:6;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA6OgC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
163
+ }), "{order:6;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA8OqC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
163
164
  var WardButtonWrapper = /*#__PURE__*/(0, _base["default"])("div", {
164
- target: "e1gaeajv7",
165
+ target: "e1gaeajv6",
165
166
  label: "WardButtonWrapper"
166
167
  })("position:relative;display:flex;justify-content:center;", _core.mq.range({
167
168
  until: _core.breakpoints.tabletWide
168
169
  }), "{", function (props) {
169
170
  return "\n order: ".concat(props.order, ";\n ");
170
- }, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsPuD","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
171
- var VolumeButton = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuButton, {
172
- target: "e1gaeajv6",
173
- label: "VolumeButton"
174
- })("background-color:inherit;width:48px;height:48px;border-radius:50%;border:0;background-position:center;background-repeat:no-repeat;cursor:pointer;svg{fill:", _core.colors.brand.primary, ";width:32px;height:32px;}&:hover,&:active,&:focus,&[aria-expanded='true']{background-color:", _core.colors.brand.greyLighter, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkQuC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
175
- var VolumeMenu = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuPopover, {
171
+ }, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAuPuD","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
172
+ var VolumeButton = /*#__PURE__*/(0, _base["default"])(_react2.Popover.Button, {
176
173
  target: "e1gaeajv5",
177
- label: "VolumeMenu"
178
- })(process.env.NODE_ENV === "production" ? {
179
- name: "f331yh",
180
- styles: "position:absolute;bottom:52px;z-index:99"
181
- } : {
182
- name: "f331yh",
183
- styles: "position:absolute;bottom:52px;z-index:99",
184
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA0RsC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
185
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
186
- });
187
- var VolumeList = /*#__PURE__*/(0, _base["default"])("div", {
174
+ label: "VolumeButton"
175
+ })("background-color:inherit;width:48px;height:48px;border-radius:50%;border:0;background-position:center;background-repeat:no-repeat;cursor:pointer;svg{fill:", _core.colors.brand.primary, ";width:32px;height:32px;}&:hover,&:active,&:focus,&[aria-expanded='true']{background-color:", _core.colors.brand.greyLighter, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmQ2C","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
176
+ var VolumeList = /*#__PURE__*/(0, _base["default"])(_react2.Popover.Panel, {
188
177
  target: "e1gaeajv4",
189
178
  label: "VolumeList"
190
- })("box-shadow:0 14px 20px -5px rgba(32, 88, 143, 0.17);border-radius:60px;background:#ffffff;border:1px solid ", _core.colors.brand.lighter, ";display:flex;flex-direction:column;justify-content:center;width:32px;height:128px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAgS6B","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
179
+ })("position:absolute;bottom:52px;z-index:99;box-shadow:0 14px 20px -5px rgba(32, 88, 143, 0.17);border-radius:60px;background:#ffffff;border:1px solid ", _core.colors.brand.lighter, ";display:flex;flex-direction:column;justify-content:center;width:32px;height:128px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2RwC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
191
180
  var VolumeSliderWrapper = /*#__PURE__*/(0, _base["default"])("div", {
192
181
  target: "e1gaeajv3",
193
182
  label: "VolumeSliderWrapper"
@@ -197,21 +186,21 @@ var VolumeSliderWrapper = /*#__PURE__*/(0, _base["default"])("div", {
197
186
  } : {
198
187
  name: "bgg7e7",
199
188
  styles: "cursor:pointer;flex:1 1 auto;display:flex;justify-content:center;padding:16px 0",
200
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4SsC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
189
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA0SsC","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
201
190
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
202
191
  });
203
192
  var VolumeSliderBackground = /*#__PURE__*/(0, _base["default"])(_slider.SliderTrack, {
204
193
  target: "e1gaeajv2",
205
194
  label: "VolumeSliderBackground"
206
- })("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":"AAoTkD","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
195
+ })("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":"AAkTkD","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
207
196
  var VolumeSliderSelected = /*#__PURE__*/(0, _base["default"])(_slider.SliderRange, {
208
197
  target: "e1gaeajv1",
209
198
  label: "VolumeSliderSelected"
210
- })("width:5px;background:", _core.colors.brand.secondary, ";border-radius:7px;bottom:0;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2TgD","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
199
+ })("width:5px;background:", _core.colors.brand.secondary, ";border-radius:7px;bottom:0;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAyTgD","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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
211
200
  var VolumeSliderHandle = /*#__PURE__*/(0, _base["default"])(_slider.SliderHandle, {
212
201
  target: "e1gaeajv0",
213
202
  label: "VolumeSliderHandle"
214
- })("width:20px;height:20px;background:", _core.colors.brand.primary, ";border-radius:50%;left:-8px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkU+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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}\n            >\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}\n                  >\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
203
+ })("width:20px;height:20px;background:", _core.colors.brand.primary, ";border-radius:50%;left:-8px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAgU+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 React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Popover } from '@headlessui/react';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\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: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled(Popover)`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(Popover.Button)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeList = styled(Popover.Panel)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\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\ntype 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 [sliderValue, setSliderValue] = useState(0);\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        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\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        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 / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\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 type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}\n          >\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}\n            >\n              {speedValue}x\n            </SpeedButton>\n            <SpeedMenu as=\"div\" portal={false}>\n              <div>\n                <SpeedList as=\"div\">\n                  {speedValues.map((speed) => (\n                    <SpeedValueButton\n                      type=\"button\"\n                      //@ts-ignore\n                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\n                      }}\n                    >\n                      {speed}x{speed === speedValue && <SpeedSelectedMark />}\n                    </SpeedValueButton>\n                  ))}\n                </SpeedList>\n              </div>\n            </SpeedMenu>\n          </Menu>\n        </SpeedWrapper>\n        <WardButtonWrapper order={5}>\n          <Forward15SecButton\n            type=\"button\"\n            title={t('audio.controls.forward15sec')}\n            aria-label={t('audio.controls.forward15sec')}\n            onClick={() => {\n              onSeekSeconds(15);\n            }}\n          >\n            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <VolumeButton aria-label={t('audio.controls.adjustVolume')}>\n            <VolumeUp />\n          </VolumeButton>\n          <VolumeList>\n            <VolumeSliderWrapper>\n              <SliderInput\n                orientation={SliderOrientation.Vertical}\n                onChange={handleVolumeSliderChange}\n                value={volumeValue}\n              >\n                <VolumeSliderBackground as=\"div\">\n                  <VolumeSliderSelected as=\"div\" />\n                  <VolumeSliderHandle as=\"div\" />\n                </VolumeSliderBackground>\n              </SliderInput>\n            </VolumeSliderWrapper>\n          </VolumeList>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
215
204
  var formatTime = function formatTime(seconds) {
216
205
  var minutes = Math.floor(seconds / 60);
217
206
  var currentSeconds = seconds % 60;
@@ -408,36 +397,27 @@ var Controls = function Controls(_ref) {
408
397
  }), (0, _jsxRuntime.jsxs)(Time, {
409
398
  children: ["-", formatTime(remainingTime)]
410
399
  })]
411
- }), (0, _jsxRuntime.jsx)(VolumeWrapper, {
412
- children: (0, _jsxRuntime.jsxs)(_menuButton.Menu, {
413
- children: [(0, _jsxRuntime.jsx)(VolumeButton, {
414
- type: "button",
415
- as: "button",
416
- title: t('audio.controls.adjustVolume'),
417
- "aria-label": t('audio.controls.adjustVolume'),
418
- children: (0, _jsxRuntime.jsx)(_common.VolumeUp, {})
419
- }), (0, _jsxRuntime.jsx)(VolumeMenu, {
420
- as: "div",
421
- portal: false,
422
- children: (0, _jsxRuntime.jsx)(VolumeList, {
423
- children: (0, _jsxRuntime.jsx)(VolumeSliderWrapper, {
424
- children: (0, _jsxRuntime.jsx)(_slider.SliderInput, {
425
- orientation: _slider.SliderOrientation.Vertical,
426
- onChange: handleVolumeSliderChange,
427
- value: volumeValue,
428
- children: (0, _jsxRuntime.jsxs)(VolumeSliderBackground, {
429
- as: "div",
430
- children: [(0, _jsxRuntime.jsx)(VolumeSliderSelected, {
431
- as: "div"
432
- }), (0, _jsxRuntime.jsx)(VolumeSliderHandle, {
433
- as: "div"
434
- })]
435
- })
436
- })
400
+ }), (0, _jsxRuntime.jsxs)(VolumeWrapper, {
401
+ children: [(0, _jsxRuntime.jsx)(VolumeButton, {
402
+ "aria-label": t('audio.controls.adjustVolume'),
403
+ children: (0, _jsxRuntime.jsx)(_common.VolumeUp, {})
404
+ }), (0, _jsxRuntime.jsx)(VolumeList, {
405
+ children: (0, _jsxRuntime.jsx)(VolumeSliderWrapper, {
406
+ children: (0, _jsxRuntime.jsx)(_slider.SliderInput, {
407
+ orientation: _slider.SliderOrientation.Vertical,
408
+ onChange: handleVolumeSliderChange,
409
+ value: volumeValue,
410
+ children: (0, _jsxRuntime.jsxs)(VolumeSliderBackground, {
411
+ as: "div",
412
+ children: [(0, _jsxRuntime.jsx)(VolumeSliderSelected, {
413
+ as: "div"
414
+ }), (0, _jsxRuntime.jsx)(VolumeSliderHandle, {
415
+ as: "div"
416
+ })]
437
417
  })
438
418
  })
439
- })]
440
- })
419
+ })
420
+ })]
441
421
  })]
442
422
  })]
443
423
  });