@ndla/ui 35.0.2 → 35.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/es/Article/ArticleSideBar.js +7 -56
- package/es/AudioPlayer/Controls.js +34 -32
- package/es/Frontpage/FrontpageMultidisciplinarySubject.js +8 -12
- package/es/Frontpage/FrontpageProgramMenu.js +7 -89
- package/es/Frontpage/FrontpageToolbox.js +5 -9
- package/es/locale/messages-en.js +15 -2
- package/es/locale/messages-nb.js +21 -8
- package/es/locale/messages-nn.js +15 -2
- package/es/locale/messages-se.js +14 -1
- package/es/locale/messages-sma.js +14 -1
- package/lib/Article/ArticleSideBar.d.ts +4 -7
- package/lib/Article/ArticleSideBar.js +7 -56
- package/lib/AudioPlayer/Controls.js +33 -31
- package/lib/Embed/conceptComponents.d.ts +1 -1
- package/lib/Frontpage/FrontpageMultidisciplinarySubject.js +8 -12
- package/lib/Frontpage/FrontpageProgramMenu.d.ts +1 -1
- package/lib/Frontpage/FrontpageProgramMenu.js +7 -89
- package/lib/Frontpage/FrontpageToolbox.js +5 -9
- package/lib/MyNdla/Resource/FolderInput.d.ts +1 -1
- package/lib/Resource/resourceComponents.d.ts +1 -1
- package/lib/locale/messages-en.d.ts +13 -0
- package/lib/locale/messages-en.js +15 -2
- package/lib/locale/messages-nb.d.ts +13 -0
- package/lib/locale/messages-nb.js +21 -8
- package/lib/locale/messages-nn.d.ts +13 -0
- package/lib/locale/messages-nn.js +15 -2
- package/lib/locale/messages-se.d.ts +13 -0
- package/lib/locale/messages-se.js +14 -1
- package/lib/locale/messages-sma.d.ts +13 -0
- package/lib/locale/messages-sma.js +14 -1
- package/package.json +21 -21
- package/src/Article/ArticleSideBar.tsx +5 -65
- package/src/AudioPlayer/Controls.tsx +23 -8
- package/src/Frontpage/FrontpageMultidisciplinarySubject.tsx +0 -2
- package/src/Frontpage/FrontpageProgramMenu.tsx +1 -66
- package/src/Frontpage/FrontpageToolbox.tsx +0 -2
- package/src/locale/messages-en.ts +18 -2
- package/src/locale/messages-nb.ts +24 -8
- package/src/locale/messages-nn.ts +18 -2
- package/src/locale/messages-se.ts +17 -1
- package/src/locale/messages-sma.ts +17 -1
- package/es/ComponentCursor/ComponentCursor.js +0 -100
- package/es/ComponentCursor/index.js +0 -2
- package/lib/ComponentCursor/ComponentCursor.d.ts +0 -6
- package/lib/ComponentCursor/ComponentCursor.js +0 -110
- package/lib/ComponentCursor/index.d.ts +0 -2
- package/lib/ComponentCursor/index.js +0 -10
- package/src/ComponentCursor/ComponentCursor.tsx +0 -101
- package/src/ComponentCursor/index.tsx +0 -2
|
@@ -12,6 +12,7 @@ var _slider = require("@reach/slider");
|
|
|
12
12
|
var _common = require("@ndla/icons/common");
|
|
13
13
|
var _core = require("@ndla/core");
|
|
14
14
|
var _reactI18next = require("react-i18next");
|
|
15
|
+
var _action = require("@ndla/icons/action");
|
|
15
16
|
var _jsxRuntime = require("@emotion/react/jsx-runtime");
|
|
16
17
|
function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function _getRequireWildcardCache(nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); }
|
|
17
18
|
function _interopRequireWildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || _typeof(obj) !== "object" && typeof obj !== "function") { return { "default": obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj["default"] = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
|
|
@@ -30,39 +31,39 @@ var ControlsWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
|
30
31
|
until: _core.breakpoints.tabletWide
|
|
31
32
|
}), "{flex-wrap:wrap;}padding:", _core.spacing.small, ";", _core.mq.range({
|
|
32
33
|
from: _core.breakpoints.tabletWide
|
|
33
|
-
}), "{padding:", _core.spacing.small, " ", _core.spacing.normal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
34
|
+
}), "{padding:", _core.spacing.small, " ", _core.spacing.normal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiBkC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
34
35
|
var PlayButton = /*#__PURE__*/(0, _base["default"])("button", {
|
|
35
36
|
target: "e1gaeajv24",
|
|
36
37
|
label: "PlayButton"
|
|
37
38
|
})("background:", _core.colors.brand.lighter, ";border:none;display:flex;align-items:center;justify-content:center;padding:0;cursor:pointer;color:", _core.colors.brand.primary, ";width:55px;height:55px;border-radius:50%;transition:", _core.misc.transition["default"], ";margin-right:", _core.spacing.small, ";", _core.mq.range({
|
|
38
39
|
until: _core.breakpoints.tabletWide
|
|
39
|
-
}), "{order:4;margin-left:", _core.spacing.small, ";}&:hover,&:active,&:focus{background:", _core.colors.brand.primary, ";color:#ffffff;}.c-icon{width:24px;height:24px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
40
|
+
}), "{order:4;margin-left:", _core.spacing.small, ";}&:hover,&:active,&:focus{background:", _core.colors.brand.primary, ";color:#ffffff;}.c-icon{width:24px;height:24px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiCgC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
40
41
|
var ForwardRewindButton = /*#__PURE__*/(0, _base["default"])("button", {
|
|
41
42
|
target: "e1gaeajv23",
|
|
42
43
|
label: "ForwardRewindButton"
|
|
43
|
-
})("background-color:inherit;background-position:center;background-repeat:no-repeat;width:42px;height:42px;border:0;border-radius:50%;cursor:pointer;font-weight:bold;font-size:9px;line-height:23px;color:", _core.colors.brand.dark, ";font-family:", _core.fonts.sans, ";transition:", _core.misc.transition["default"], ";&:hover{background-color:", _core.colors.brand.greyLighter, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
44
|
+
})("background-color:inherit;background-position:center;background-repeat:no-repeat;width:42px;height:42px;border:0;border-radius:50%;cursor:pointer;font-weight:bold;font-size:9px;line-height:23px;color:", _core.colors.brand.dark, ";font-family:", _core.fonts.sans, ";transition:", _core.misc.transition["default"], ";&:hover{background-color:", _core.colors.brand.greyLighter, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAiEyC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
44
45
|
var Forward15SecButton = /*#__PURE__*/(0, _base["default"])(ForwardRewindButton, {
|
|
45
46
|
target: "e1gaeajv22",
|
|
46
47
|
label: "Forward15SecButton"
|
|
47
|
-
})("
|
|
48
|
+
})("svg{fill:", _core.colors.brand.primary, ";width:24px;height:24px;}", _core.mq.range({
|
|
48
49
|
until: _core.breakpoints.tabletWide
|
|
49
|
-
}), "{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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
50
|
+
}), "{order:3;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsFsD","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
50
51
|
var Back15SecButton = /*#__PURE__*/(0, _base["default"])(ForwardRewindButton, {
|
|
51
52
|
target: "e1gaeajv21",
|
|
52
53
|
label: "Back15SecButton"
|
|
53
|
-
})("
|
|
54
|
+
})("svg{fill:", _core.colors.brand.primary, ";width:24px;height:24px;}", _core.mq.range({
|
|
54
55
|
until: _core.breakpoints.tabletWide
|
|
55
|
-
}), "{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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
56
|
+
}), "{order:5;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAgGmD","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
56
57
|
var SpeedWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
57
58
|
target: "e1gaeajv20",
|
|
58
59
|
label: "SpeedWrapper"
|
|
59
60
|
})("position:relative;display:flex;justify-content:center;", _core.mq.range({
|
|
60
61
|
until: _core.breakpoints.tabletWide
|
|
61
|
-
}), "{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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
62
|
+
}), "{order:2;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2G+B","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
62
63
|
var SpeedButton = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuButton, {
|
|
63
64
|
target: "e1gaeajv19",
|
|
64
65
|
label: "SpeedButton"
|
|
65
|
-
})("height:32px;border:0;background:none;cursor:pointer;font-weight:600;font-size:12px;text-align:center;width:52px;&:hover,&:active,&:focus,&[aria-expanded='true']{background:", _core.colors.brand.greyLighter, ";border-radius:27px;color:", _core.colors.text.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
66
|
+
})("height:32px;border:0;background:none;cursor:pointer;font-weight:600;font-size:12px;text-align:center;width:52px;&:hover,&:active,&:focus,&[aria-expanded='true']{background:", _core.colors.brand.greyLighter, ";border-radius:27px;color:", _core.colors.text.primary, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAmHsC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
66
67
|
var SpeedMenu = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuPopover, {
|
|
67
68
|
target: "e1gaeajv18",
|
|
68
69
|
label: "SpeedMenu"
|
|
@@ -72,19 +73,19 @@ var SpeedMenu = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuPopover, {
|
|
|
72
73
|
} : {
|
|
73
74
|
name: "exmt8v",
|
|
74
75
|
styles: "position:absolute;bottom:36px;z-index:99",
|
|
75
|
-
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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
76
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsIqC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
76
77
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
77
78
|
});
|
|
78
79
|
var SpeedList = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuItems, {
|
|
79
80
|
target: "e1gaeajv17",
|
|
80
81
|
label: "SpeedList"
|
|
81
|
-
})("background:#ffffff;border:1px solid ", _core.colors.brand.lighter, ";border-radius:5px;padding:5px 10px;display:flex;flex-direction:column;justify-content:center;align-items:stretch;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
82
|
+
})("background:#ffffff;border:1px solid ", _core.colors.brand.lighter, ";border-radius:5px;padding:5px 10px;display:flex;flex-direction:column;justify-content:center;align-items:stretch;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4ImC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
82
83
|
var SpeedValueButton = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuItem, {
|
|
83
84
|
target: "e1gaeajv16",
|
|
84
85
|
label: "SpeedValueButton"
|
|
85
86
|
})("height:28px;position:relative;background:none;border:0;padding:0 14px;cursor:pointer;font-weight:600;font-size:14px;color:", _core.colors.text.light, ";display:flex;justify-content:center;align-items:center;&:hover,&:active,&:focus,&[data-selected]{background:", _core.colors.brand.greyLighter, ";border-radius:5px;color:", _core.colors.text.primary, ";}", function (props) {
|
|
86
87
|
return props.selected && "\n color: ".concat(_core.colors.text.primary, ";\n \n ");
|
|
87
|
-
}, ";" + (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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
88
|
+
}, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2JgE","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
88
89
|
var SpeedSelectedMark = /*#__PURE__*/(0, _base["default"])("span", {
|
|
89
90
|
target: "e1gaeajv15",
|
|
90
91
|
label: "SpeedSelectedMark"
|
|
@@ -94,7 +95,7 @@ var SpeedSelectedMark = /*#__PURE__*/(0, _base["default"])("span", {
|
|
|
94
95
|
} : {
|
|
95
96
|
name: "1ymok7g",
|
|
96
97
|
styles: "border-radius:50%;background:#d1372e;width:6px;height:6px;display:inline-block;align-self:flex-start;margin:6px 0 0 2px",
|
|
97
|
-
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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
98
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAwLqC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
98
99
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
99
100
|
});
|
|
100
101
|
var Time = /*#__PURE__*/(0, _base["default"])("div", {
|
|
@@ -106,7 +107,7 @@ var Time = /*#__PURE__*/(0, _base["default"])("div", {
|
|
|
106
107
|
} : {
|
|
107
108
|
name: "11g4mt0",
|
|
108
109
|
styles: "font-size:16px",
|
|
109
|
-
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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
110
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkMuB","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
110
111
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
111
112
|
});
|
|
112
113
|
var ProgressWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
@@ -114,15 +115,15 @@ var ProgressWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
|
114
115
|
label: "ProgressWrapper"
|
|
115
116
|
})("flex:1 1 auto;display:flex;align-items:center;margin:0 ", _core.spacing.small, ";", _core.mq.range({
|
|
116
117
|
until: _core.breakpoints.tabletWide
|
|
117
|
-
}), "{order:1;width:100%;margin:0;margin-bottom:", _core.spacing.normal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
118
|
+
}), "{order:1;width:100%;margin:0;margin-bottom:", _core.spacing.normal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsMkC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
118
119
|
var SliderWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
119
120
|
target: "e1gaeajv12",
|
|
120
121
|
label: "SliderWrapper"
|
|
121
|
-
})("cursor:pointer;flex:1 1 auto;margin:0 ", _core.spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
122
|
+
})("cursor:pointer;flex:1 1 auto;margin:0 ", _core.spacing.small, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkNgC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
122
123
|
var ProgressBackground = /*#__PURE__*/(0, _base["default"])(_slider.SliderTrack, {
|
|
123
124
|
target: "e1gaeajv11",
|
|
124
125
|
label: "ProgressBackground"
|
|
125
|
-
})("height:4px;width:100%;background:", _core.colors.brand.lighter, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
126
|
+
})("height:4px;width:100%;background:", _core.colors.brand.lighter, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAwN8C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
126
127
|
var ProgressPlayed = /*#__PURE__*/(0, _base["default"])(_slider.SliderRange, {
|
|
127
128
|
target: "e1gaeajv10",
|
|
128
129
|
label: "ProgressPlayed"
|
|
@@ -132,7 +133,7 @@ var ProgressPlayed = /*#__PURE__*/(0, _base["default"])(_slider.SliderRange, {
|
|
|
132
133
|
} : {
|
|
133
134
|
name: "1pw5ehb",
|
|
134
135
|
styles: "height:4px;background:#5cbc80;border-radius:7px",
|
|
135
|
-
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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
136
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA+N0C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
136
137
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
137
138
|
});
|
|
138
139
|
var ProgressHandle = /*#__PURE__*/(0, _base["default"])(_slider.SliderHandle, {
|
|
@@ -144,7 +145,7 @@ var ProgressHandle = /*#__PURE__*/(0, _base["default"])(_slider.SliderHandle, {
|
|
|
144
145
|
} : {
|
|
145
146
|
name: "11d5yus",
|
|
146
147
|
styles: "width:20px;height:20px;background:#5cbc80;border-radius:50%;top:-8px",
|
|
147
|
-
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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
148
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAqO2C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
148
149
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
149
150
|
});
|
|
150
151
|
var VolumeWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
@@ -152,7 +153,7 @@ var VolumeWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
|
152
153
|
label: "VolumeWrapper"
|
|
153
154
|
})("position:relative;display:flex;justify-content:center;", _core.mq.range({
|
|
154
155
|
until: _core.breakpoints.tabletWide
|
|
155
|
-
}), "{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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
156
|
+
}), "{order:6;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA6OgC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
156
157
|
var WardButtonWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
157
158
|
target: "e1gaeajv7",
|
|
158
159
|
label: "WardButtonWrapper"
|
|
@@ -160,11 +161,11 @@ var WardButtonWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
|
160
161
|
until: _core.breakpoints.tabletWide
|
|
161
162
|
}), "{", function (props) {
|
|
162
163
|
return "\n order: ".concat(props.order, ";\n ");
|
|
163
|
-
}, ";}" + (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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
164
|
+
}, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAsPuD","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
164
165
|
var VolumeButton = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuButton, {
|
|
165
166
|
target: "e1gaeajv6",
|
|
166
167
|
label: "VolumeButton"
|
|
167
|
-
})("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:", _core.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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
168
|
+
})("background-color:inherit;width:48px;height:48px;border-radius:50%;border:0;background-position:center;background-repeat:no-repeat;cursor:pointer;svg{fill:", _core.colors.brand.primary, ";width:32px;height:32px;}&:hover,&:active,&:focus,&[aria-expanded='true']{background-color:", _core.colors.brand.greyLighter, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkQuC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
168
169
|
var VolumeMenu = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuPopover, {
|
|
169
170
|
target: "e1gaeajv5",
|
|
170
171
|
label: "VolumeMenu"
|
|
@@ -174,13 +175,13 @@ var VolumeMenu = /*#__PURE__*/(0, _base["default"])(_menuButton.MenuPopover, {
|
|
|
174
175
|
} : {
|
|
175
176
|
name: "f331yh",
|
|
176
177
|
styles: "position:absolute;bottom:52px;z-index:99",
|
|
177
|
-
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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
178
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA0RsC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
178
179
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
179
180
|
});
|
|
180
181
|
var VolumeList = /*#__PURE__*/(0, _base["default"])("div", {
|
|
181
182
|
target: "e1gaeajv4",
|
|
182
183
|
label: "VolumeList"
|
|
183
|
-
})("box-shadow:0 14px 20px -5px rgba(32, 88, 143, 0.17);border-radius:60px;background:#ffffff;border:1px solid ", _core.colors.brand.lighter, ";display:flex;flex-direction:column;justify-content:center;width:32px;height:128px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
184
|
+
})("box-shadow:0 14px 20px -5px rgba(32, 88, 143, 0.17);border-radius:60px;background:#ffffff;border:1px solid ", _core.colors.brand.lighter, ";display:flex;flex-direction:column;justify-content:center;width:32px;height:128px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAgS6B","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
184
185
|
var VolumeSliderWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
185
186
|
target: "e1gaeajv3",
|
|
186
187
|
label: "VolumeSliderWrapper"
|
|
@@ -190,21 +191,21 @@ var VolumeSliderWrapper = /*#__PURE__*/(0, _base["default"])("div", {
|
|
|
190
191
|
} : {
|
|
191
192
|
name: "bgg7e7",
|
|
192
193
|
styles: "cursor:pointer;flex:1 1 auto;display:flex;justify-content:center;padding:16px 0",
|
|
193
|
-
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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
194
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA4SsC","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */",
|
|
194
195
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
195
196
|
});
|
|
196
197
|
var VolumeSliderBackground = /*#__PURE__*/(0, _base["default"])(_slider.SliderTrack, {
|
|
197
198
|
target: "e1gaeajv2",
|
|
198
199
|
label: "VolumeSliderBackground"
|
|
199
|
-
})("height:100%;width:5px;background:", _core.colors.brand.lighter, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
200
|
+
})("height:100%;width:5px;background:", _core.colors.brand.lighter, ";border-radius:7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAoTkD","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
200
201
|
var VolumeSliderSelected = /*#__PURE__*/(0, _base["default"])(_slider.SliderRange, {
|
|
201
202
|
target: "e1gaeajv1",
|
|
202
203
|
label: "VolumeSliderSelected"
|
|
203
|
-
})("width:5px;background:", _core.colors.brand.secondary, ";border-radius:7px;bottom:0;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
204
|
+
})("width:5px;background:", _core.colors.brand.secondary, ";border-radius:7px;bottom:0;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AA2TgD","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
204
205
|
var VolumeSliderHandle = /*#__PURE__*/(0, _base["default"])(_slider.SliderHandle, {
|
|
205
206
|
target: "e1gaeajv0",
|
|
206
207
|
label: "VolumeSliderHandle"
|
|
207
|
-
})("width:20px;height:20px;background:", _core.colors.brand.primary, ";border-radius:50%;left:-8px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"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    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            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    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
208
|
+
})("width:20px;height:20px;background:", _core.colors.brand.primary, ";border-radius:50%;left:-8px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["Controls.tsx"],"names":[],"mappings":"AAkU+C","file":"Controls.tsx","sourcesContent":["/**\n * Copyright (c) 2021-present, NDLA.\n *\n * This source code is licensed under the GPLv3 license found in the\n * LICENSE file in the root directory of this source tree.\n *\n */\n\nimport React, { useEffect, useRef, useState } from 'react';\nimport styled from '@emotion/styled';\nimport { Menu, MenuButton, MenuItem, MenuPopover, MenuItems, MenuItemProps } from '@reach/menu-button';\nimport { SliderInput, SliderTrack, SliderRange, SliderHandle, SliderOrientation } from '@reach/slider';\nimport { Play, Pause, VolumeUp } from '@ndla/icons/common';\nimport { breakpoints, colors, fonts, misc, mq, spacing } from '@ndla/core';\nimport { useTranslation } from 'react-i18next';\nimport { Back15, Forward15 } from '@ndla/icons/action';\n\nconst ControlsWrapper = styled.div`\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  background: #ffffff;\n  font-family: ${fonts.sans};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    flex-wrap: wrap;\n  }\n  padding: ${spacing.small};\n  ${mq.range({ from: breakpoints.tabletWide })} {\n    padding: ${spacing.small} ${spacing.normal};\n  }\n`;\n\nconst PlayButton = styled.button`\n  background: ${colors.brand.lighter};\n  border: none;\n  display: flex;\n  align-items: center;\n  justify-content: center;\n  padding: 0;\n  cursor: pointer;\n  color: ${colors.brand.primary};\n  width: 55px;\n  height: 55px;\n  border-radius: 50%;\n  transition: ${misc.transition.default};\n  margin-right: ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 4;\n    margin-left: ${spacing.small};\n  }\n\n  &:hover,\n  &:active,\n  &:focus {\n    background: ${colors.brand.primary};\n    color: #ffffff;\n  }\n\n  .c-icon {\n    width: 24px;\n    height: 24px;\n  }\n`;\n\nconst ForwardRewindButton = styled.button`\n  background-color: inherit;\n  background-position: center;\n  background-repeat: no-repeat;\n  width: 42px;\n  height: 42px;\n  border: 0;\n  border-radius: 50%;\n  cursor: pointer;\n  font-weight: bold;\n  font-size: 9px;\n  line-height: 23px;\n  color: ${colors.brand.dark};\n  font-family: ${fonts.sans};\n  transition: ${misc.transition.default};\n\n  &:hover {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst Forward15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 3;\n  }\n`;\nconst Back15SecButton = styled(ForwardRewindButton)`\n  svg {\n    fill: ${colors.brand.primary};\n    width: 24px;\n    height: 24px;\n  }\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 5;\n  }\n`;\n\nconst SpeedWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 2;\n  }\n`;\nconst SpeedButton = styled(MenuButton)`\n  height: 32px;\n  border: 0;\n  background: none;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 12px;\n  text-align: center;\n  width: 52px;\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 27px;\n    color: ${colors.text.primary};\n  }\n`;\n\nconst SpeedMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 36px;\n  z-index: 99;\n`;\n\nconst SpeedList = styled(MenuItems)`\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  border-radius: 5px;\n  padding: 5px 10px;\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  align-items: stretch;\n`;\n\ninterface SpeedValueButtonProps extends MenuItemProps {\n  selected?: boolean;\n}\n\nconst SpeedValueButton = styled(MenuItem)<SpeedValueButtonProps>`\n  height: 28px;\n  position: relative;\n  background: none;\n  border: 0;\n  padding: 0 14px;\n  cursor: pointer;\n  font-weight: 600;\n  font-size: 14px;\n  color: ${colors.text.light};\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  &:hover,\n  &:active,\n  &:focus,\n  &[data-selected] {\n    background: ${colors.brand.greyLighter};\n    border-radius: 5px;\n    color: ${colors.text.primary};\n  }\n  ${(props) =>\n    props.selected &&\n    `\n    color: ${colors.text.primary};\n    \n  `}\n`;\n\nconst SpeedSelectedMark = styled.span`\n  border-radius: 50%;\n  background: #d1372e;\n  width: 6px;\n  height: 6px;\n  display: inline-block;\n  align-self: flex-start;\n  margin: 6px 0 0 2px;\n`;\n\nconst Time = styled.div`\n  font-size: 16px;\n`;\n\nconst ProgressWrapper = styled.div`\n  flex: 1 1 auto;\n  display: flex;\n  align-items: center;\n  margin: 0 ${spacing.small};\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 1;\n    width: 100%;\n    margin: 0;\n    margin-bottom: ${spacing.normal};\n  }\n`;\nconst SliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  margin: 0 ${spacing.small};\n`;\n\nconst ProgressBackground = styled(SliderTrack)`\n  height: 4px;\n  width: 100%;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst ProgressPlayed = styled(SliderRange)`\n  height: 4px;\n  background: #5cbc80;\n  border-radius: 7px;\n`;\n\nconst ProgressHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: #5cbc80;\n  border-radius: 50%;\n  top: -8px;\n`;\n\nconst VolumeWrapper = styled.div`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    order: 6;\n  }\n`;\n\nconst WardButtonWrapper = styled.div<{ order: number }>`\n  position: relative;\n  display: flex;\n  justify-content: center;\n  ${mq.range({ until: breakpoints.tabletWide })} {\n    ${(props) =>\n      `\n    order: ${props.order};\n  `}\n  }\n`;\n\nconst VolumeButton = styled(MenuButton)`\n  background-color: inherit;\n  width: 48px;\n  height: 48px;\n  border-radius: 50%;\n  border: 0;\n  background-position: center;\n  background-repeat: no-repeat;\n  cursor: pointer;\n\n  svg {\n    fill: ${colors.brand.primary};\n    width: 32px;\n    height: 32px;\n  }\n\n  &:hover,\n  &:active,\n  &:focus,\n  &[aria-expanded='true'] {\n    background-color: ${colors.brand.greyLighter};\n  }\n`;\n\nconst VolumeMenu = styled(MenuPopover)`\n  position: absolute;\n  bottom: 52px;\n  z-index: 99;\n`;\n\nconst VolumeList = styled.div`\n  box-shadow: 0 14px 20px -5px rgba(32, 88, 143, 0.17);\n  border-radius: 60px;\n  background: #ffffff;\n  border: 1px solid ${colors.brand.lighter};\n  display: flex;\n  flex-direction: column;\n  justify-content: center;\n  width: 32px;\n  height: 128px;\n`;\n\nconst VolumeSliderWrapper = styled.div`\n  cursor: pointer;\n  flex: 1 1 auto;\n  display: flex;\n  justify-content: center;\n  padding: 16px 0;\n`;\n\nconst VolumeSliderBackground = styled(SliderTrack)`\n  height: 100%;\n  width: 5px;\n  background: ${colors.brand.lighter};\n  border-radius: 7px;\n`;\n\nconst VolumeSliderSelected = styled(SliderRange)`\n  width: 5px;\n  background: ${colors.brand.secondary};\n  border-radius: 7px;\n  bottom: 0;\n`;\n\nconst VolumeSliderHandle = styled(SliderHandle)`\n  width: 20px;\n  height: 20px;\n  background: ${colors.brand.primary};\n  border-radius: 50%;\n  left: -8px;\n`;\n\nconst formatTime = (seconds: number) => {\n  const minutes = Math.floor(seconds / 60);\n  const currentSeconds = seconds % 60;\n\n  const formattedSeconds = currentSeconds < 10 ? `0${currentSeconds}` : currentSeconds;\n  return `${minutes}:${formattedSeconds}`;\n};\n\nconst speedValues = [0.5, 0.75, 1, 1.25, 1.5, 1.75, 2];\n\ntype Props = {\n  src: string;\n  title: string;\n};\n\nconst Controls = ({ src, title }: Props) => {\n  const { t } = useTranslation();\n  const [speedValue, setSpeedValue] = useState(1);\n  const [volumeValue, setVolumeValue] = useState(100);\n  const [sliderValue, setSliderValue] = useState(0);\n  const [currentTime, setCurrentTime] = useState(0);\n  const [remainingTime, setRemainingTime] = useState(0);\n  const [playing, setPlaying] = useState(false);\n  const audioRef = useRef<HTMLAudioElement>(null);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      audioRef.current.playbackRate = speedValue;\n    }\n  }, [speedValue]);\n\n  useEffect(() => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      const handleTimeUpdate = () => {\n        const { currentTime, duration } = audioElement;\n        const percent = Math.round((currentTime / duration) * 100);\n        setSliderValue(percent);\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleLoadedMetaData = () => {\n        const { currentTime, duration } = audioElement;\n        setCurrentTime(Math.round(currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n        setRemainingTime(Math.round(duration - currentTime));\n      };\n\n      const handleTimeEnded = () => {\n        setPlaying(false);\n      };\n\n      audioElement.addEventListener('timeupdate', handleTimeUpdate);\n      audioElement.addEventListener('loadedmetadata', handleLoadedMetaData);\n      audioElement.addEventListener('ended', handleTimeEnded);\n      return () => {\n        audioElement.removeEventListener('timeupdate', handleTimeUpdate);\n        audioElement.removeEventListener('loadedmetadata', handleLoadedMetaData);\n        audioElement.removeEventListener('ended', handleTimeEnded);\n      };\n    }\n  }, []);\n\n  const togglePlay = () => {\n    if (audioRef.current) {\n      const audioElement = audioRef.current;\n      if (!playing) {\n        audioElement.play();\n      } else {\n        audioElement.pause();\n      }\n      setPlaying(!playing);\n    }\n  };\n\n  const onSeekSeconds = (seconds: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime += seconds;\n    }\n  };\n\n  const handleSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.currentTime = (value / 100) * audioRef.current.duration;\n    }\n  };\n\n  const handleVolumeSliderChange = (value: number) => {\n    if (audioRef.current) {\n      audioRef.current.volume = value / 100;\n      setVolumeValue(value);\n    }\n  };\n\n  return (\n    <div>\n      {/* eslint-disable-next-line jsx-a11y/media-has-caption */}\n      <audio ref={audioRef} src={src} title={title} preload=\"metadata\" />\n      <ControlsWrapper>\n        <PlayButton type=\"button\" onClick={togglePlay} title=\"play\" aria-label=\"play\">\n          <span aria-hidden>\n            {playing ? (\n              <Pause role=\"img\" aria-label=\"Pause\" title=\"Pause\" />\n            ) : (\n              <Play role=\"img\" aria-label=\"Play\" title=\"Play\" />\n            )}\n          </span>\n        </PlayButton>\n        <WardButtonWrapper order={3}>\n          <Back15SecButton\n            type=\"button\"\n            title={t('audio.controls.rewind15sec')}\n            aria-label={t('audio.controls.rewind15sec')}\n            onClick={() => {\n              onSeekSeconds(-15);\n            }}>\n            <Back15 />\n          </Back15SecButton>\n        </WardButtonWrapper>\n\n        <SpeedWrapper>\n          <Menu>\n            <SpeedButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.selectSpeed')}\n              aria-label={t('audio.controls.selectSpeed')}>\n              {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            <Forward15 />\n          </Forward15SecButton>\n        </WardButtonWrapper>\n        <ProgressWrapper>\n          <Time>{formatTime(currentTime)}</Time>\n          <SliderWrapper>\n            <SliderInput onChange={handleSliderChange} value={sliderValue}>\n              <ProgressBackground as=\"div\">\n                <ProgressPlayed as=\"div\" />\n                <ProgressHandle as=\"div\" />\n              </ProgressBackground>\n            </SliderInput>\n          </SliderWrapper>\n          <Time>-{formatTime(remainingTime)}</Time>\n        </ProgressWrapper>\n        <VolumeWrapper>\n          <Menu>\n            {/* @ts-ignore */}\n            <VolumeButton\n              type=\"button\"\n              as=\"button\"\n              title={t('audio.controls.adjustVolume')}\n              aria-label={t('audio.controls.adjustVolume')}>\n              <VolumeUp />\n            </VolumeButton>\n            <VolumeMenu as=\"div\" portal={false}>\n              <VolumeList>\n                <VolumeSliderWrapper>\n                  <SliderInput\n                    orientation={SliderOrientation.Vertical}\n                    onChange={handleVolumeSliderChange}\n                    value={volumeValue}>\n                    <VolumeSliderBackground as=\"div\">\n                      <VolumeSliderSelected as=\"div\" />\n                      <VolumeSliderHandle as=\"div\" />\n                    </VolumeSliderBackground>\n                  </SliderInput>\n                </VolumeSliderWrapper>\n              </VolumeList>\n            </VolumeMenu>\n          </Menu>\n        </VolumeWrapper>\n      </ControlsWrapper>\n    </div>\n  );\n};\n\nexport default Controls;\n"]} */"));
|
|
208
209
|
var formatTime = function formatTime(seconds) {
|
|
209
210
|
var minutes = Math.floor(seconds / 60);
|
|
210
211
|
var currentSeconds = seconds % 60;
|
|
@@ -338,7 +339,7 @@ var Controls = function Controls(_ref) {
|
|
|
338
339
|
onClick: function onClick() {
|
|
339
340
|
onSeekSeconds(-15);
|
|
340
341
|
},
|
|
341
|
-
children:
|
|
342
|
+
children: (0, _jsxRuntime.jsx)(_action.Back15, {})
|
|
342
343
|
})
|
|
343
344
|
}), (0, _jsxRuntime.jsx)(SpeedWrapper, {
|
|
344
345
|
children: (0, _jsxRuntime.jsxs)(_menuButton.Menu, {
|
|
@@ -380,7 +381,7 @@ var Controls = function Controls(_ref) {
|
|
|
380
381
|
onClick: function onClick() {
|
|
381
382
|
onSeekSeconds(15);
|
|
382
383
|
},
|
|
383
|
-
children:
|
|
384
|
+
children: (0, _jsxRuntime.jsx)(_action.Forward15, {})
|
|
384
385
|
})
|
|
385
386
|
}), (0, _jsxRuntime.jsxs)(ProgressWrapper, {
|
|
386
387
|
children: [(0, _jsxRuntime.jsx)(Time, {
|
|
@@ -407,7 +408,8 @@ var Controls = function Controls(_ref) {
|
|
|
407
408
|
type: "button",
|
|
408
409
|
as: "button",
|
|
409
410
|
title: t('audio.controls.adjustVolume'),
|
|
410
|
-
"aria-label": t('audio.controls.adjustVolume')
|
|
411
|
+
"aria-label": t('audio.controls.adjustVolume'),
|
|
412
|
+
children: (0, _jsxRuntime.jsx)(_common.VolumeUp, {})
|
|
411
413
|
}), (0, _jsxRuntime.jsx)(VolumeMenu, {
|
|
412
414
|
as: "div",
|
|
413
415
|
portal: false,
|