@ndla/ui 4.2.2 → 4.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -36,45 +36,45 @@ var ControlsWrapper = _styled("div", {
36
36
  until: breakpoints.tabletWide
37
37
  }), "{flex-wrap:wrap;}padding:", spacing.small, ";", mq.range({
38
38
  from: breakpoints.tabletWide
39
- }), "{padding:", spacing.small, " ", spacing.normal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAgBkC","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
39
+ }), "{padding:", spacing.small, " ", spacing.normal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAgBkC","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
40
40
 
41
41
  var PlayButton = _styled("button", {
42
42
  target: "e1gaeajv1",
43
43
  label: "PlayButton"
44
44
  })("background:", colors.brand.lighter, ";border:none;display:flex;align-items:center;justify-content:center;padding:0;cursor:pointer;color:", colors.brand.primary, ";width:55px;height:55px;border-radius:50%;transition:", misc.transition["default"], ";margin-right:", spacing.small, ";", mq.range({
45
45
  until: breakpoints.tabletWide
46
- }), "{order:4;margin-left:", spacing.small, ";}&:hover,&:active,&:focus{background:", 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":"AAgCgC","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
46
+ }), "{order:4;margin-left:", spacing.small, ";}&:hover,&:active,&:focus{background:", 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":"AAgCgC","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
47
47
 
48
48
  var ForwardRewindButton = _styled("button", {
49
49
  target: "e1gaeajv2",
50
50
  label: "ForwardRewindButton"
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:", colors.brand.dark, ";font-family:", fonts.sans, ";transition:", misc.transition["default"], ";&:hover{background-color:", colors.brand.greyLighter, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAgEyC","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\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:", colors.brand.dark, ";font-family:", fonts.sans, ";transition:", misc.transition["default"], ";&:hover{background-color:", colors.brand.greyLighter, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAgEyC","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
52
52
 
53
53
  var Forward15SecButton = /*#__PURE__*/_styled(ForwardRewindButton, {
54
54
  target: "e1gaeajv3",
55
55
  label: "Forward15SecButton"
56
56
  })("background-image:url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");", mq.range({
57
57
  until: breakpoints.tabletWide
58
- }), "{order:3;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAqFsD","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
58
+ }), "{order:3;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAqFsD","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
59
59
 
60
60
  var Back15SecButton = /*#__PURE__*/_styled(ForwardRewindButton, {
61
61
  target: "e1gaeajv4",
62
62
  label: "Back15SecButton"
63
63
  })("background-image:url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");", mq.range({
64
64
  until: breakpoints.tabletWide
65
- }), "{order:5;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2FmD","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
65
+ }), "{order:5;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2FmD","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
66
66
 
67
67
  var SpeedWrapper = _styled("div", {
68
68
  target: "e1gaeajv5",
69
69
  label: "SpeedWrapper"
70
70
  })("position:relative;display:flex;justify-content:center;", mq.range({
71
71
  until: breakpoints.tabletWide
72
- }), "{order:2;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkG+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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
72
+ }), "{order:2;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkG+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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
73
73
 
74
74
  var SpeedButton = /*#__PURE__*/_styled(MenuButton, {
75
75
  target: "e1gaeajv6",
76
76
  label: "SpeedButton"
77
- })("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:", colors.brand.greyLighter, ";border-radius:27px;color:", colors.text.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA0GsC","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
77
+ })("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:", colors.brand.greyLighter, ";border-radius:27px;color:", colors.text.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA0GsC","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
78
78
 
79
79
  var SpeedMenu = /*#__PURE__*/_styled(MenuPopover, {
80
80
  target: "e1gaeajv7",
@@ -85,21 +85,21 @@ var SpeedMenu = /*#__PURE__*/_styled(MenuPopover, {
85
85
  } : {
86
86
  name: "1iz23z5",
87
87
  styles: "position:absolute;bottom:36px;z-index:99;",
88
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA6HqC","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
88
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA6HqC","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
89
89
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
90
90
  });
91
91
 
92
92
  var SpeedList = /*#__PURE__*/_styled(MenuItems, {
93
93
  target: "e1gaeajv8",
94
94
  label: "SpeedList"
95
- })("background:#ffffff;border:1px solid ", 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":"AAmImC","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
95
+ })("background:#ffffff;border:1px solid ", 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":"AAmImC","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
96
96
 
97
97
  var SpeedValueButton = /*#__PURE__*/_styled(MenuItem, {
98
98
  target: "e1gaeajv9",
99
99
  label: "SpeedValueButton"
100
100
  })("height:28px;position:relative;background:none;border:0;padding:0 14px;cursor:pointer;font-weight:600;font-size:14px;color:", colors.text.light, ";display:flex;justify-content:center;align-items:center;&:hover,&:active,&:focus,&[data-selected]{background:", colors.brand.greyLighter, ";border-radius:5px;color:", colors.text.primary, ";}", function (props) {
101
101
  return props.selected && "\n color: ".concat(colors.text.primary, ";\n \n ");
102
- }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiJgE","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */");
102
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkJgE","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */");
103
103
 
104
104
  var SpeedSelectedMark = _styled("span", {
105
105
  target: "e1gaeajv10",
@@ -110,7 +110,7 @@ var SpeedSelectedMark = _styled("span", {
110
110
  } : {
111
111
  name: "ilrwre",
112
112
  styles: "border-radius:50%;background:#d1372e;width:6px;height:6px;display:inline-block;align-self:flex-start;margin:6px 0 0 2px;",
113
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA8KqC","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
113
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA+KqC","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
114
114
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
115
115
  });
116
116
 
@@ -123,7 +123,7 @@ var Time = _styled("div", {
123
123
  } : {
124
124
  name: "6xix1i",
125
125
  styles: "font-size:16px;",
126
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAwLuB","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
126
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAyLuB","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
127
127
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
128
128
  });
129
129
 
@@ -132,17 +132,17 @@ var ProgressWrapper = _styled("div", {
132
132
  label: "ProgressWrapper"
133
133
  })("flex:1 1 auto;display:flex;align-items:center;margin:0 ", spacing.small, ";", mq.range({
134
134
  until: breakpoints.tabletWide
135
- }), "{order:1;width:100%;margin:0;margin-bottom:", spacing.normal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4LkC","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
135
+ }), "{order:1;width:100%;margin:0;margin-bottom:", spacing.normal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA6LkC","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
136
136
 
137
137
  var SliderWrapper = _styled("div", {
138
138
  target: "e1gaeajv13",
139
139
  label: "SliderWrapper"
140
- })("cursor:pointer;flex:1 1 auto;margin:0 ", spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAwMgC","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
140
+ })("cursor:pointer;flex:1 1 auto;margin:0 ", spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAyMgC","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
141
141
 
142
142
  var ProgressBackground = /*#__PURE__*/_styled(SliderTrack, {
143
143
  target: "e1gaeajv14",
144
144
  label: "ProgressBackground"
145
- })("height:4px;width:100%;background:", 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":"AA8M8C","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
145
+ })("height:4px;width:100%;background:", 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":"AA+M8C","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
146
146
 
147
147
  var ProgressPlayed = /*#__PURE__*/_styled(SliderRange, {
148
148
  target: "e1gaeajv15",
@@ -153,7 +153,7 @@ var ProgressPlayed = /*#__PURE__*/_styled(SliderRange, {
153
153
  } : {
154
154
  name: "hpepo7",
155
155
  styles: "height:4px;background:#5cbc80;border-radius:7px;",
156
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAqN0C","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
156
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsN0C","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
157
157
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
158
158
  });
159
159
 
@@ -166,7 +166,7 @@ var ProgressHandle = /*#__PURE__*/_styled(SliderHandle, {
166
166
  } : {
167
167
  name: "a7slo3",
168
168
  styles: "width:20px;height:20px;background:#5cbc80;border-radius:50%;top:-8px;",
169
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2N2C","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
169
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4N2C","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
170
170
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
171
171
  });
172
172
 
@@ -175,7 +175,7 @@ var VolumeWrapper = _styled("div", {
175
175
  label: "VolumeWrapper"
176
176
  })("position:relative;display:flex;justify-content:center;", mq.range({
177
177
  until: breakpoints.tabletWide
178
- }), "{order:6;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmOgC","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
178
+ }), "{order:6;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAoOgC","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
179
179
 
180
180
  var WardButtonWrapper = _styled("div", {
181
181
  target: "e1gaeajv18",
@@ -184,12 +184,12 @@ var WardButtonWrapper = _styled("div", {
184
184
  until: breakpoints.tabletWide
185
185
  }), "{", function (props) {
186
186
  return "\n order: ".concat(props.order, ";\n ");
187
- }, "}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4OuD","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
187
+ }, "}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA6OuD","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
188
188
 
189
189
  var VolumeButton = /*#__PURE__*/_styled(MenuButton, {
190
190
  target: "e1gaeajv19",
191
191
  label: "VolumeButton"
192
- })("background-color:inherit;background-image:url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");width:48px;height:48px;border-radius:50%;border:0;background-position:center;background-repeat:no-repeat;cursor:pointer;&:hover,&:active,&:focus,&[aria-expanded='true']{background-color:", colors.brand.greyLighter, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAwPuC","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
192
+ })("background-color:inherit;background-image:url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");width:48px;height:48px;border-radius:50%;border:0;background-position:center;background-repeat:no-repeat;cursor:pointer;&:hover,&:active,&:focus,&[aria-expanded='true']{background-color:", colors.brand.greyLighter, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAyPuC","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
193
193
 
194
194
  var VolumeMenu = /*#__PURE__*/_styled(MenuPopover, {
195
195
  target: "e1gaeajv20",
@@ -200,14 +200,14 @@ var VolumeMenu = /*#__PURE__*/_styled(MenuPopover, {
200
200
  } : {
201
201
  name: "116k0z6",
202
202
  styles: "position:absolute;bottom:52px;z-index:99;",
203
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2QsC","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
203
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4QsC","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
204
204
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
205
205
  });
206
206
 
207
207
  var VolumeList = _styled("div", {
208
208
  target: "e1gaeajv21",
209
209
  label: "VolumeList"
210
- })("box-shadow:0 14px 20px -5px rgba(32,88,143,0.17);border-radius:60px;background:#ffffff;border:1px solid ", 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":"AAiR6B","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
210
+ })("box-shadow:0 14px 20px -5px rgba(32,88,143,0.17);border-radius:60px;background:#ffffff;border:1px solid ", 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":"AAkR6B","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
211
211
 
212
212
  var VolumeSliderWrapper = _styled("div", {
213
213
  target: "e1gaeajv22",
@@ -218,24 +218,24 @@ var VolumeSliderWrapper = _styled("div", {
218
218
  } : {
219
219
  name: "10z7j35",
220
220
  styles: "cursor:pointer;flex:1 1 auto;display:flex;justify-content:center;padding:16px 0;",
221
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA6RsC","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
221
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA8RsC","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */",
222
222
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
223
223
  });
224
224
 
225
225
  var VolumeSliderBackground = /*#__PURE__*/_styled(SliderTrack, {
226
226
  target: "e1gaeajv23",
227
227
  label: "VolumeSliderBackground"
228
- })("height:100%;width:5px;background:", 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":"AAqSkD","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
228
+ })("height:100%;width:5px;background:", 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":"AAsSkD","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
229
229
 
230
230
  var VolumeSliderSelected = /*#__PURE__*/_styled(SliderRange, {
231
231
  target: "e1gaeajv24",
232
232
  label: "VolumeSliderSelected"
233
- })("width:5px;background:", 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":"AA4SgD","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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
233
+ })("width:5px;background:", 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":"AA6SgD","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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
234
234
 
235
235
  var VolumeSliderHandle = /*#__PURE__*/_styled(SliderHandle, {
236
236
  target: "e1gaeajv25",
237
237
  label: "VolumeSliderHandle"
238
- })("width:20px;height:20px;background:", 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":"AAmT+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 } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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\ntype SpeedValueButtonProps = {\n  selected?: boolean;\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      as=\"button\"\n                      key={speed}\n                      selected={speed === speedValue}\n                      onSelect={() => {\n                        setSpeedValue(speed);\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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
238
+ })("width:20px;height:20px;background:", 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":"AAoT+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 } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\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  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M19.8358 20.8769C17.7849 22.6711 15.1504 23.6564 12.4254 23.6482C9.70049 23.6401 7.07191 22.6391 5.03176 20.8326C2.99161 19.0262 1.67975 16.5381 1.34176 13.8342C1.00377 11.1303 1.66282 8.39585 3.19553 6.1428C4.72825 3.88975 7.02955 2.27253 9.66866 1.59387C12.3078 0.915214 15.1037 1.22165 17.5332 2.45581C19.9627 3.68998 21.8591 5.76726 22.8674 8.2988M23.5676 1.29649L23.5676 8.2988L16.5653 8.2988' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  background-image: url(\"data:image/svg+xml,%3Csvg width='25' height='25' viewBox='0 0 25 25' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M5.19004 21.3306C7.24095 23.1248 9.87547 24.11 12.6004 24.1019C15.3254 24.0937 17.954 23.0927 19.9941 21.2863C22.0343 19.4798 23.3461 16.9918 23.6841 14.2879C24.0221 11.5839 23.3631 8.84951 21.8303 6.59646C20.2976 4.3434 17.9963 2.72618 15.3572 2.04753C12.7181 1.36887 9.92213 1.67531 7.49267 2.90947C5.0632 4.14363 3.16681 6.22092 2.15848 8.75246M1.45825 1.75015L1.45825 8.75246L8.46057 8.75246' stroke='%23184673' stroke-width='1.52778' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  background-image: url(\"data:image/svg+xml,%3Csvg width='32' height='32' viewBox='0 0 32 32' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M20 10.6667C20.828 11.2877 21.5 12.0929 21.9628 13.0186C22.4257 13.9443 22.6667 14.9651 22.6667 16C22.6667 17.035 22.4257 18.0557 21.9628 18.9814C21.5 19.9071 20.828 20.7124 20 21.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M23.5999 6.66669C24.9918 7.79155 26.1145 9.21352 26.8858 10.8284C27.6571 12.4434 28.0574 14.2104 28.0574 16C28.0574 17.7897 27.6571 19.5567 26.8858 21.1716C26.1145 22.7865 24.9918 24.2085 23.5999 25.3334' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3Cpath d='M8 20H5.33333C4.97971 20 4.64057 19.8595 4.39052 19.6095C4.14048 19.3594 4 19.0203 4 18.6666V13.3333C4 12.9797 4.14048 12.6406 4.39052 12.3905C4.64057 12.1405 4.97971 12 5.33333 12H8L12.6667 5.99998C12.7832 5.77362 12.9769 5.59641 13.2127 5.50037C13.4484 5.40433 13.7108 5.3958 13.9523 5.47631C14.1939 5.55682 14.3986 5.72107 14.5296 5.93937C14.6607 6.15768 14.7093 6.41564 14.6667 6.66665V25.3333C14.7093 25.5843 14.6607 25.8423 14.5296 26.0606C14.3986 26.2789 14.1939 26.4431 13.9523 26.5237C13.7108 26.6042 13.4484 26.5956 13.2127 26.4996C12.9769 26.4036 12.7832 26.2263 12.6667 26L8 20Z' stroke='%23184673' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'/%3E%3C/svg%3E%0A\");\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  &: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    <>\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            15\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              {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                      {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            15\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            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\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    </>\n  );\n};\n\nexport default Controls;\n"]} */"));
239
239
 
240
240
  var formatTime = function formatTime(seconds) {
241
241
  var minutes = Math.floor(seconds / 60);
@@ -399,7 +399,8 @@ var Controls = function Controls(_ref) {
399
399
  as: "div"
400
400
  }, speedValues.map(function (speed) {
401
401
  return ___EmotionJSX(SpeedValueButton, {
402
- type: "button",
402
+ type: "button" //@ts-ignore
403
+ ,
403
404
  as: "button",
404
405
  key: speed,
405
406
  selected: speed === speedValue,